范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

Spring中经典的9种设计模式

  简单工厂(非23种设计模式中的一种)
  实现方式:BeanFactory。Spring中的BeanFactory就是简单工厂模式的体现,根据传入一个唯一的标识来获得Bean对象,但是否是在传入参数后创建还是传入参数前创建这个要根据具体情况来定。
  实质:由一个工厂类根据传入的参数,动态决定应该创建哪一个产品类。
  实现原理:
  bean容器的启动阶段:  读取bean的xml配置文件,将bean元素分别转换成一个BeanDefinition对象。  然后通过BeanDefinitionRegistry将这些bean注册到beanFactory中,保存在它的一个ConcurrentHashMap中。  将BeanDefinition注册到了beanFactory之后,在这里Spring为我们提供了一个扩展的切口,允许我们通过实现接口BeanFactoryPostProcessor 在此处来插入我们定义的代码。典型的例子就是:PropertyPlaceholderConfigurer,我们一般在配置数据库的dataSource时使用到的占位符的值,就是它注入进去的。
  容器中bean的实例化阶段,实例化阶段主要是通过反射或者CGLIB对bean进行实例化,在这个阶段Spring又给我们暴露了很多的扩展点:  各种的Aware接口,比如 BeanFactoryAware,对于实现了这些Aware接口的bean,在实例化bean时Spring会帮我们注入对应的BeanFactory的实例。  BeanPostProcessor接口,实现了BeanPostProcessor接口的bean,在实例化bean时Spring会帮我们调用接口中的方法。  InitializingBean接口,实现了InitializingBean接口的bean,在实例化bean时Spring会帮我们调用接口中的方法。  DisposableBean接口,实现了BeanPostProcessor接口的bean,在该bean死亡时Spring会帮我们调用接口中的方法。
  设计意义:
  松耦合。可以将原来硬编码的依赖,通过Spring这个beanFactory这个工长来注入依赖,也就是说原来只有依赖方和被依赖方,现在我们引入了第三方——Spring这个beanFactory,由它来解决bean之间的依赖问题,达到了松耦合的效果。  bean的额外处理。通过Spring接口的暴露,在实例化bean的阶段我们可以进行一些额外的处理,这些额外的处理只需要让bean实现对应的接口即可,那么spring就会在bean的生命周期调用我们实现的接口来处理该bean。
  工厂方法
  实现方式:FactoryBean接口。
  实现原理:实现了FactoryBean接口的bean是一类叫做factory的bean。其特点是,spring会在使用getBean()调用获得该bean时,会自动调用该bean的getObject()方法,所以返回的不是factory这个bean,而是这个bean.getOjbect()方法的返回值。
  例子:
  典型的例子有Spring与MyBatis的结合。  代码示例:  说明:我们看上面该bean,因为实现了FactoryBean接口,所以返回的不是SqlSessionFactoryBean的实例,而是它的 SqlSessionFactoryBean.getObject()的返回值。
  单例模式
  Spring依赖注入Bean实例默认是单例的。
  Spring的依赖注入(包括lazy-init方式)都是发生在AbstractBeanFactory的getBean里。getBean的doGetBean方法调用getSingleton进行bean的创建。
  分析getSingleton()方法:  public Object getSingleton(String beanName){     //参数true设置标识允许早期依赖     return getSingleton(beanName,true); } protected Object getSingleton(String beanName, boolean allowEarlyReference) {     //检查缓存中是否存在实例     Object singletonObject = this.singletonObjects.get(beanName);     if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {         //如果为空,则锁定全局变量并进行处理。         synchronized (this.singletonObjects) {             //如果此bean正在加载,则不处理             singletonObject = this.earlySingletonObjects.get(beanName);             if (singletonObject == null && allowEarlyReference) {                   //当某些方法需要提前初始化的时候则会调用addSingleFactory 方法将对应的ObjectFactory初始化策略存储在singletonFactories                 ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);                 if (singletonFactory != null) {                     //调用预先设定的getObject方法                     singletonObject = singletonFactory.getObject();                     //记录在缓存中,earlysingletonObjects和singletonFactories互斥                     this.earlySingletonObjects.put(beanName, singletonObject);                     this.singletonFactories.remove(beanName);                 }             }         }     }     return (singletonObject != NULL_OBJECT ? singletonObject : null); }
  getSingleton()过程图(ps:Spring依赖注入时,使用了双重判断加锁的单例模式):
  单例模式定义:保证一个类仅有一个实例,并提供一个访问它的全局访问点。
  Spring对单例的实现:Spring中的单例模式完成了后半句话,即提供了全局的访问点BeanFactory。但没有从构造器级别去控制单例,这是因为Spring管理的是任意的Java对象。
  适配器模式
  实现方式:SpringMVC中的适配器HandlerAdatper。
  实现原理:HandlerAdatper根据Handler规则执行不同的Handler。
  实现过程:DispatcherServlet根据HandlerMapping返回的handler,向HandlerAdatper发起请求,处理Handler。HandlerAdapter根据规则找到对应的Handler并让其执行,执行完毕后Handler会向HandlerAdapter返回一个ModelAndView,最后由HandlerAdapter向DispatchServelet返回一个ModelAndView。
  实现意义:HandlerAdatper使得Handler的扩展变得容易,只需要增加一个新的Handler和一个对应的HandlerAdapter即可。因此Spring定义了一个适配接口,使得每一种Controller有一种对应的适配器实现类,让适配器代替controller执行相应的方法。这样在扩展Controller时,只需要增加一个适配器类就完成了SpringMVC的扩展了。
  装饰器模式
  实现方式:Spring中用到的包装器模式在类名上有两种表现:一种是类名中含有Wrapper,另一种是类名中含有Decorator。
  实质:
  动态地给一个对象添加一些额外的职责。  就增加功能来说,Decorator模式相比生成子类更为灵活。
  代理模式
  实现方式:AOP底层,就是动态代理模式的实现。
  动态代理:在内存中构建的,不需要手动编写代理类  静态代理:需要手工编写代理类,代理类引用被代理对象。
  实现原理:切面在应用运行的时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象创建动态的创建一个代理对象。SpringAOP就是以这种方式织入切面的。
  织入:把切面应用到目标对象并创建新的代理对象的过程。
  观察者模式
  实现方式:Spring的事件驱动模型使用的是观察者模式 ,Spring中Observer模式常用的地方是listener的实现。
  具体实现:事件机制的实现需要三个部分,即:事件源、事件、事件监听器。
  ApplicationEvent抽象类[事件]
  继承自JDK的EventObject,所有的事件都需要继承ApplicationEvent,并且通过构造器参数source得到事件源。
  该类的实现类ApplicationContextEvent表示ApplicaitonContext的容器事件。
  代码:  public abstract class ApplicationEvent extends EventObject {     private static final long serialVersionUID = 7099057708183571937L;     private final long timestamp;     public ApplicationEvent(Object source) {     super(source);     this.timestamp = System.currentTimeMillis();     }     public final long getTimestamp() {         return this.timestamp;     } }
  ApplicationListener接口[事件监听器]
  继承自JDK的EventListener,所有的监听器都要实现这个接口。
  这个接口只有一个onApplicationEvent()方法,该方法接受一个ApplicationEvent或其子类对象作为参数,在方法体中,可以通过不同对Event类的判断来进行相应的处理。
  当事件触发时所有的监听器都会收到消息。
  代码:      public interface ApplicationListener extends                EventListener {                  void onApplicationEvent(E event); }
  ApplicationContext接口[事件源]
  ApplicationContext是Spring中的全局容器,翻译过来是"应用上下文"。
  实现了ApplicationEventPublisher接口。
  职责:负责读取bean的配置文档,管理bean的加载,维护bean之间的依赖关系,可以说是负责bean的整个生命周期,再通俗一点就是我们平时所说的IOC容器。
  代码:  public interface ApplicationEventPublisher {         void publishEvent(ApplicationEvent event); }     public void publishEvent(ApplicationEvent event) {     Assert.notNull(event, "Event must not be null");     if (logger.isTraceEnabled()) {          logger.trace("Publishing event in " + getDisplayName() + ": " + event);     }     getApplicationEventMulticaster().multicastEvent(event);     if (this.parent != null) {     this.parent.publishEvent(event);     } }
  ApplicationEventMulticaster抽象类[事件源中publishEvent方法需要调用其方法getApplicationEventMulticaster]
  属于事件广播器,它的作用是把Applicationcontext发布的Event广播给所有的监听器。
  代码:  public abstract class AbstractApplicationContext extends DefaultResourceLoader     implements ConfigurableApplicationContext, DisposableBean {       private ApplicationEventMulticaster applicationEventMulticaster;       protected void registerListeners() {       // Register statically specified listeners first.       for (ApplicationListener<?> listener : getApplicationListeners()) {       getApplicationEventMulticaster().addApplicationListener(listener);       }       // Do not initialize FactoryBeans here: We need to leave all regular beans       // uninitialized to let post-processors apply to them!       String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);       for (String lisName : listenerBeanNames) {       getApplicationEventMulticaster().addApplicationListenerBean(lisName);       }     }   }
  策略模式
  实现方式:Spring框架的资源访问Resource接口。该接口提供了更强的资源访问能力,Spring框架本身大量使用了Resource接口来访问底层资源。
  Resource接口介绍
  source接口是具体资源访问策略的抽象,也是所有资源访问类所实现的接口。
  Resource接口主要提供了如下几个方法:
  getInputStream():定位并打开资源,返回资源对应的输入流。每次调用都返回新的输入流。调用者必须负责关闭输入流。  exists():返回Resource所指向的资源是否存在。  isOpen():返回资源文件是否打开,如果资源文件不能多次读取,每次读取结束应该显式关闭,以防止资源泄漏。  getDescription():返回资源的描述信息,通常用于资源处理出错时输出该信息,通常是全限定文件名或实际URL。  getFile:返回资源对应的File对象。  getURL:返回资源对应的URL对象。
  最后两个方法通常无须使用,仅在通过简单方式访问无法实现时,Resource提供传统的资源访问的功能。
  Resource接口本身没有提供访问任何底层资源的实现逻辑,针对不同的底层资源,Spring将会提供不同的Resource实现类,不同的实现类负责不同的资源访问逻辑。
  Spring为Resource接口提供了如下实现类:
  UrlResource:访问网络资源的实现类。  ClassPathResource:访问类加载路径里资源的实现类。  FileSystemResource:访问文件系统里资源的实现类。  ServletContextResource:访问相对于ServletContext路径里的资源的实现类.  InputStreamResource:访问输入流资源的实现类。  ByteArrayResource:访问字节数组资源的实现类。
  这些Resource实现类,针对不同的的底层资源,提供了相应的资源访问逻辑,并提供便捷的包装,以利于客户端程序的资源访问。
  模版方法模式
  经典模板方法定义:
  父类定义了骨架(调用哪些方法及顺序),某些特定方法由子类实现。
  最大的好处:代码复用,减少重复代码。除了子类要实现的特定方法,其他方法及方法调用顺序都在父类中预先写好了。
  所以父类模板方法中有两类方法:
  共同的方法:所有子类都会用到的代码  不同的方法:子类要覆盖的方法,分为两种:  抽象方法:父类中的是抽象方法,子类必须覆盖  钩子方法:父类中是一个空方法,子类继承了默认也是空的  注:为什么叫钩子,子类可以通过这个钩子(方法),控制父类,因为这个钩子实际是父类的方法(空方法)!
  Spring模板方法模式实质:是模板方法模式和回调模式的结合,是Template Method不需要继承的另一种实现方式。Spring几乎所有的外接扩展都采用这种模式。
  具体实现:JDBC的抽象和对Hibernate的集成,都采用了一种理念或者处理方式,那就是模板方法模式与相应的Callback接口相结合。
  采用模板方法模式是为了以一种统一而集中的方式来处理资源的获取和释放,以JdbcTemplate为例:  public abstract class JdbcTemplate {        public final Object execute(String sql){           Connection con=null;           Statement stmt=null;           try{               con=getConnection();               stmt=con.createStatement();               Object retValue=executeWithStatement(stmt,sql);               return retValue;           }catch(SQLException e){                ...           }finally{               closeStatement(stmt);               releaseConnection(con);           }       }        protected abstract Object executeWithStatement(Statement   stmt, String sql);   }
  引入回调原因:JdbcTemplate是抽象类,不能够独立使用,我们每次进行数据访问的时候都要给出一个相应的子类实现,这样肯定不方便,所以就引入了回调。
  回调代码:  public interface StatementCallback{       Object doWithStatement(Statement stmt);   }
  利用回调方法重写JdbcTemplate方法:  public class JdbcTemplate {       public final Object execute(StatementCallback callback){           Connection con=null;           Statement stmt=null;           try{               con=getConnection();               stmt=con.createStatement();               Object retValue=callback.doWithStatement(stmt);               return retValue;           }catch(SQLException e){               ...           }finally{               closeStatement(stmt);               releaseConnection(con);           }       }        ...//其它方法定义   }
  Jdbc使用方法如下:  JdbcTemplate jdbcTemplate=...;       final String sql=...;       StatementCallback callback=new StatementCallback(){       public Object=doWithStatement(Statement stmt){           return ...;       }   }     jdbcTemplate.execute(callback);
  为什么JdbcTemplate没有使用继承?
  因为这个类的方法太多,但是我们还是想用到JdbcTemplate已有的稳定的、公用的数据库连接,那么我们怎么办呢?我们可以把变化的东西抽出来作为一个参数传入JdbcTemplate的方法中。但是变化的东西是一段代码,而且这段代码会用到JdbcTemplate中的变量。怎么办?那我们就用回调对象吧。在这个回调对象中定义一个操纵JdbcTemplate中变量的方法,我们去实现这个方法,就把变化的东西集中到这里了。然后我们再传入这个回调对象到JdbcTemplate,从而完成了调用。
  原文链接:https://blog.csdn.net/caoxiaohong1005/article/details/80039656

游戏王ssb01新卡魔女术闪刀姬魔妖,不得不说k社抓住了商机各位决斗者,大家好,这里是社长!游戏王圣诞礼盒ssb01是今年的圣诞礼盒,相对于以前的几个圣诞礼盒,社长认为还是少些火候的,因此这个盒子的价格也没有前几年抄的那么高。今天社长就来带英雄联盟史上分最高职业选手?小黄人四个号上韩服千分大家好,这里是嘟咕电竞。韩服排行一直是衡量职业选手个人能力水平的标准之一。各大俱乐部都会根据韩服rank分数排行来培养青训选手。众所周知,每个大杀四方的职业选手之前都是出名的ran阿水眼含泪花,赛后自曝带病登场,白色月牙1支队伍坑了我们前言12月26日晚上,英雄联盟LPL德杯正式落下帷幕,最终TES完成让2追3,成功击败FPX卫冕德杯冠军。这已经不是TES首次上演让2追3的好戏,他们之前在S10世界赛上也完成过让终于找到战队了,宝蓝将有新ad,还是跟jkl并称的双子星选手?引子目前各大战队的比赛阵容基本确定,就算有些队伍没公布完全体阵容,我们也差不多知道了。不过,有一位选手的去向迟迟不清楚,他就是宝蓝。之前ning直播的时候说,baolan想确定的话55岁老阿姨玩CSGO,坚持3000多小时,已成为游戏狙神?最近,一位叫作娟姨的网红悄然走红,虽然粉丝不多,但依然吸引了不少游戏爱好者的关注,原因就在于娟姨是一位55岁的老阿姨,但她却坚持直播游戏,而且已经玩CSGO这款游戏超过3000多个新丧尸神作?消逝的光芒2游戏前瞻如果说2022年最令人期待的丧尸题材游戏是哪款?那毫无疑问是消逝的光芒2。在丧尸题材游戏泛滥成灾的今天,消逝的光芒系列游戏一直历久弥新,其开创性的第一人称跑酷机制让它备受玩家欢迎,神鸡营输给刘氏家族,韩涵坐等德华亲自上阵,结果德华不敢了?大家期待已久的KPI杯总决赛已经正式结束,在神鸡营和刘氏家族的对阵中神鸡营直接被碾压,几乎没有一点还手之力,就像是曾经故辞带队被梦之队吊打一样,德华都忍不住想要亲自上场,结果韩涵已永劫无间世界冠军赛三排决赛第4场大混战三队共同斩获荣光北京时间12月26日,永劫无间世界冠军赛决赛DAY3开战。第3局开局,总分排名第一的ADA爆冷出局,只拿到一个击败分,为后续比赛增加了悬念。其他各队都想抓住这个机会,尽可能多拿分。黄子韬妥妥的电竞男主,乱战中豪取四杀太秀了,被他圈粉了12月26日下午,英雄联盟手游职业资格赛全国大赛赛道中,黄子韬为大家呈现了一场令人热血沸腾的明星表演赛,激烈的比赛斗智斗勇,黄子韬热血出战,向大家展现了电竞的魅力。据了解黄子韬接触艺帝帝艺高人胆大?带PEL首位女选手比赛,力压4AMST拿下战神榜12月26日,由斗鱼独播的和平精英黄金大奖赛迎来了S3赛季的揭幕战,经常关注和平精英赛事的小伙伴们应该对黄金大奖赛十分熟悉了,作为国内最具含金量的电竞赛事之一,举办到了第三季赛制也最终幻想XVI预计2022年春季到来最终幻想XVI制作人吉田直树在博客上发布了有关游戏开发的公告。吉田引用新冠问题,包括远程工作如何对游戏开发管道的关键方面产生影响,解释说有关该游戏的消息已推迟到今年发布,预计将在2
游戏赚钱套路动人心刷头条有各式各样的广告,最近喜欢搜星际争霸比赛视频观看,广告也升级为游戏广告,广告语让人觉得像是在撒钱,就下载一款爷爷的花园玩玩。安装好app,微信登录,播种一颗种子,立马收割,几抓周一般在孩子多大时进行蚂蚁庄园6月1日答案最新蚂蚁庄园6月1日的问题目前已经给玩家们带来了,今天的问题之一就是抓周一般在孩子多大时进行?玩家每天都可以在蚂蚁庄园当中通过回答问题的方式来得到饲料奖励,那么今天问题的答案是什么呢?海市蜃楼之馆ampampamp第七境中文版制作中开发商Novectacle宣布海市蜃楼之馆游魂归梦第七境中文版制作中,其中海市蜃楼之馆游魂归梦曾获MC满分评价。以下是中文版宣传片视频地址以下为官方介绍游魂归梦是海市蜃楼之馆系列在LOL海内外网友热议KT春季赛表现围绕Smeb重建队伍果然没用!在刚刚的LCK春季常规赛上,KT在面临自己生死大关一战HLE时,未能把握自己的命运,最终以02落败,那么这样一来KT现在在联赛的位置就非常的尴尬了。LCK春季赛目前的排名中JAG是亚索用W挡掉谁的R技能最有成就感?大家好我是祖安的未来,一个最讨厌亚索风墙的召唤师,今天就跟大家聊一聊被风墙挡掉什么技能的时候最气。1女枪的R大龙池里两阵人马正打得火热,各个英雄也都是八仙过海,各显神通,此时都已用LCK赛区保级成功率微乎其微,昔日冠军KT是能否闯过这道坎!2019年LCK春季赛也在3月31日落下帷幕,与LPL赛区不同的是,在LCK打不好的话就要保级,不管你曾经有过多少荣誉,打的多好都要面临这道坎!目前LCK季后赛保级赛名额也都已经全虎牙再引进一高人气大佬,粉丝散文诗告白我们有新家了苦瓜原创,翻版必究!苦瓜电竞让你体验不一样的电竞主播圈!作为国内高人气直播平台,虎牙直播总是频繁的带给网友们太多的惊喜,不管是涉及到那个领域,在虎牙大家庭中,你总会找到一个自己最爱有钱人的世界不懂,土豪出2w买玩家昵称,玩家回绝天美还只能看看玩王者荣耀的朋友们都知道,昵称在游戏里面就是代表着一个人的身份,因为荣耀中的昵称只能出现一次,这也是为了保证各位玩家能有一个良好的竞技环境,这样一来也方便了天美官方的各项管理。随着国产游戏等于质量差?这款国产游戏在国外口碑爆炸!在国内游戏领域存在着这么一个现象,许多国内工作室做游戏完全是以赚钱为目的,从而忽略了对游戏可玩性以及游戏质量的提升。要么我们玩到的是建模粗糙手感极差的游戏要么游戏打着免费的旗号却充还玩王昭君?赶紧换,甄姬才是最强中路法师摘要小伙伴们,大家好,我是唐唐,欢迎收看本期的英雄攻略。王昭君在这个赛季可谓是香饽饽,小伙伴们在打排位的时候经常遇到,她的团控能力非常强悍,清理兵线更是一绝,即使嬴政,也得对她侧目游戏的寒冬过去了?那些游戏产业亲历者的所思所想深响原创作者马小军如果以2001年,盛大游戏引入传奇为元年的话,今年刚好是中国网络游戏的第18年在寒冬与初春的交替之际,中国网络游戏迎来了它的成人礼。18年前,中国游戏玩家的数量在