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

打死也要记住的Spring中9种设计模式!!!

  作者:iCoding91
  来源:http://t.csdn.cn/MxOO5
  目录 1.简单工厂(非23种设计模式中的一种) 2.工厂方法 3.单例模式 4.适配器模式 5.装饰器模式 6.代理模式 7.观察者模式 8.策略模式 9.模版方法模式 1.简单工厂(非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。 [非常重要] 2.工厂方法
  实现方式:
  FactoryBean接口。
  实现原理:
  实现了FactoryBean接口的bean是一类叫做factory的bean。其特点是,spring会在使用getBean()调用获得该bean时,会自动调用该bean的getObject()方法,所以返回的不是factory这个bean,而是这个bean.getOjbect()方法的返回值。
  例子:
  典型的例子有spring与mybatis的结合。
  代码示例:
  说明:
  我们看上面该bean,因为实现了FactoryBean接口,所以返回的不是 SqlSessionFactoryBean 的实例,而是它的 SqlSessionFactoryBean.getObject() 的返回值。 3.单例模式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对象。 4.适配器模式
  实现方式:
  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的扩展了。 5.装饰器模式
  实现方式:
  Spring中用到的包装器模式在类名上有两种表现:一种是类名中含有Wrapper,另一种是类名中含有Decorator。
  实质:
  动态地给一个对象添加一些额外的职责。
  就增加功能来说,Decorator模式相比生成子类更为灵活。 6.代理模式
  实现方式:
  AOP底层,就是动态代理模式的实现。动态代理:在内存中构建的,不需要手动编写代理类 静态代理:需要手工编写代理类,代理类引用被代理对象。
  实现原理:
  切面在应用运行的时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象创建动态的创建一个代理对象。SpringAOP就是以这种方式织入切面的。
  织入:把切面应用到目标对象并创建新的代理对象的过程。 7.观察者模式
  实现方式:
  spring的事件驱动模型使用的是 观察者模式 ,Spring中Observer模式常用的地方是listener的实现。
  具体实现:
  事件机制的实现需要三个部分,事件源,事件,事件监听器
  (1)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;     } }
  (2)ApplicationListener接口 [事件监听器] 继承自jdk的EventListener,所有的监听器都要实现这个接口。 这个接口只有一个onApplicationEvent()方法,该方法接受一个ApplicationEvent或其子类对象作为参数,在方法体中,可以通过不同对Event类的判断来进行相应的处理。 当事件触发时所有的监听器都会收到消息。 代码: public interface ApplicationListener extends EventListener {      void onApplicationEvent(E event); }
  (3)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);     } }
  (4)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);     }   } }8.策略模式
  实现方式:
  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 实现类,针对不同的的底层资源,提供了相应的资源访问逻辑,并提供便捷的包装,以利于客户端程序的资源访问。 9.模版方法模式
  经典模板方法定义:
  父类定义了骨架(调用哪些方法及顺序),某些特定方法由子类实现。
  最大的好处:代码复用,减少重复代码。除了子类要实现的特定方法,其他方法及方法调用顺序都在父类中预先写好了。
  所以父类模板方法中有两类方法:
  共同的方法:  所有子类都会用到的代码
  不同的方法:   子类要覆盖的方法,分为两种: 抽象方法:父类中的是抽象方法,子类必须覆盖 钩子方法:父类中是一个空方法,子类继承了默认也是空的
  注:为什么叫钩子,子类可以通过这个钩子(方法),控制父类,因为这个钩子实际是父类的方法(空方法)!
  Spring模板方法模式实质:
  是模板方法模式和回调模式的结合,是Template Method不需要继承的另一种实现方式。Spring几乎所有的外接扩展都采用这种模式。
  具体实现:
  JDBC的抽象和对Hibernate的集成,都采用了一种理念或者处理方式,那就是模板方法模式与相应的Callback接口相结合。
  采用模板方法模式是为了以一种统一而集中的方式来处理资源的获取和释放,以JdbcTempalte为例: 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,从而完成了调用。

台州市黄岩区旅游业微改造精提升解锁文旅新业态随着疫情的阴霾云开雾散,旅游市场需求开发模式发生巨大变化。房车旅游作为一种文化一种生活方式,越来越受到市民的青睐,成为后疫情时代消费热门元素。2023年,台州市黄岩区着力放大央视跨糖尿病患者的健康早餐食谱您是否在早上与空腹血糖升高作斗争?你可能认为我只是整夜禁食!我睡了8个小时没吃东西!我的血糖是怎么升高的?我们对此有一个术语叫做黎明现象。如果这种情况偶尔发生,那真的没什么大不了的清淡爽口家常菜,有助于老年人血糖控制糖尿病患者需忌口,有些东西可以多吃,有些东西则必须少量。营养师表示,糖尿病患者家属可以准备清淡爽口家常菜,如香菜竹笋汤所含蔬菜类配料珊瑚草等都富含纤维质,可望延缓血糖上升,营养师建心电血糖轻松测,didoE55SProMax健康运动戴手上随着智能可穿戴设备的不断普及,我们大多数人已经使用过智能手表手环之类的产品,说实话,因为有了手环手表这些产品,为我们日常的运动健康带来很多的新体验。但普通智能手表功能相对单一,并且患糖尿病22年的病友总结出的一套控制血糖的有效方法糖尿病是一种极为普遍的慢性疾病,如果血糖不能得到有效地控制,就可能引发许多并发症,严重危害患者健康。身为一位糖尿病患者22年的老病友,我总结出一套控制血糖的有效方法,现在想与大家分卫健委2023最新指南,推荐控血糖吃这7种主食要控血糖,建议吃哪些主食?今年卫健委对外发布的成人糖尿病食养指南(2023版),针对全国华北东北西北华东华中西南华南地区,又分别针对春夏秋冬,都制定的3天食谱,也就是每个地区都12无缝对接抢救新生哈医大一院多学科完成高难度产时产房手术成功抢救先天性脐疝胎儿近日,在哈医大一院群力院区的产科手术室内出现超燃的一幕,一位孕产妇在这边剖宫产娩出一位宝宝后,宝宝还没有断脐连着胎盘就开始进行检查,检查评估后立刻被抱到旁边紧邻的手术台上完成了一场生小孩要收红包?收了红包还不干事?这个妇产科主任现在怎么样?妇产科医生,就是熟练掌握妇产科相关专业基础理论知识,守护着女性的健康,迎接每一个小生命的降生,为未来人口的身体质量把好第一关。就在昨天,苏州有一家医院的妇产科主任医师被人举报明码实孩子发热怎么办孩子发烧,是家长们最常遇到的问题之一。如果您不幸遇到了这种情况,以下是一些建议,希望能对您有所帮助1。观察体温首先要观察孩子的体温,确认是否确实发烧。正常情况下孩子的体温为3637上海,七个高知家庭抱团育儿本文转自果壳童学馆(guokrkid)作者周育楠这或许是全上海独一份的魔方课,授课的是年仅九岁的男孩晨晨。此刻他穿着蓝色上衣,站在白板前,娴熟转动着手中的魔方。那些色彩斑斓的小方块33岁还在拼头胎,因为我走了很多弯路,所以把这件事告诉姐妹们!门诊是很多女性朋友因怀不上而苦恼,而也有很多女性朋友虽然能怀上,但是怀上后总是保不住,老流产,我觉得这比怀不上更难受,更痛苦。这种情况就称之为胎停育。胎停育是怎么一回事呢?如果在孕
布克眼红了!詹娜与新男友两人同骑一匹马!举止亲密,旁若无人聚焦体坛资讯,关注名人花边。NBA常规赛进入尾声,各路豪强都在一张季后赛门票奋力厮杀。菲尼克斯太阳队自从杜兰特加盟之后,太阳之子布克便迎来强援,拥有了再次冲击总冠军的资本。只不过初云浮亏损王背后的温氏兄弟前年亏134亿,全家仍是亿万富豪对于普通人而言,学习成功人士的发家之道,进而掌握他们的财富密码,这是一条现实可行的途径,俗话说,站在巨人的肩膀上,才能看得更远,我们通过对上百名亿万富豪的创业故事进行了分析,结果表北京市文化市场综合执法总队发布旅游防范须知来源人民网日前,北京市文化市场综合执法总队发布旅游防范须知,提醒广大市民假日出行需谨慎,警惕九大雷与坑。旅游出行擦亮眼,非法组团套路深游客须警惕社会上一些没有旅行社资质的公司组织个(体育)NBA综合湖人险胜爵士勇士力克雷霆新华社华盛顿4月4日电20222023赛季美职篮(NBA)常规赛4日展开多场争夺,洛杉矶湖人135133加时险胜犹他爵士,金州勇士136125击败俄克拉荷马雷霆。第四节还剩1分43世界花卉大观园5万平方米赏花面积100多项传统文化节目亮相穿汉服赏国风舞蹈游花海4月5日至9日,第三届北京花朝汉服文化节在世界花卉大观园开启。5万平方米的赏花面积,结合100多项传统文化节目2000多种国风商品展示,让广大市民可以尽情赏古被全网搜视频,她有多无辜,他就有多可恶!体坛溯本清源迫在眉睫最近张继科与景甜频频登上热搜,这件事原本是好几年前的事了,但随着记者李微敖的爆料,这件往事被公开在了所有人的面前。很多人把这一事件对比陈冠希的艳照门,但我个人认为这件事比艳照门事件好风好水好生活怎么选?尽在A。O。史密斯AILiNK智能家居时代,在家就能享受好风好水的生活虽然看似容易,但却耗费很大精力。除了要添置各样的智能家居家电,还要进行精心布局设计,让智能家电之间实现互联互通,联动家居环境和生活方式。而这缺乏行动力的人生,有多可怕!狼是天生的行动者,而不是被动的幻想者!狼道中有个小故事,两个人站在一个十字路口,前面大路小路纵横交错,犹如一张理不清的网。一个自以为很有智慧的人走到网前,很优雅地弯下腰,一点一点地碧桂园再出发杨惠妍翻起新一页2023年3月,对走过30余年的碧桂园而言,是重要的时间节点此前低调到全网难见公开报道的杨惠妍走到台前,执掌公司经营,甚至亮相2022年业绩交流会,自信从容地谈其对公司中长期战略开新河街道消防车开进幼儿园孩子们零距离学消防来源津滨海客户端津滨海讯(记者田敏)近日,新河街道南瑞社区居委会联合南益育树家幼儿园开展了消防车进幼儿园活动。在活动现场,消防员用浅显易懂的语言为孩子们一一介绍了消防装备,如消防水被忽视的生死教育,该如何向孩子讲述死亡?中新网北京4月5日电(刘欢)你都没有经历过死亡,怎么进行生死教育?清华大学附属北京清华长庚医院疼痛科主任路桂军从事生死教育与安宁疗护工作多年,在这样被人质疑过后,他在2021年的清