成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

來自讀者的面試題:談談Spring用到了哪些設計模式?

開發 架構
所謂代理,是指它與被代理對象實現了相同的接口,客戶端必須通過代理才能與被代理的目標類進行交互,而代理一般在交互的過程中(交互前后),進行某些特定的處理,比如在調用這個方法前做前置處理,調用這個方法后做后置處理。

[[402520]]

本文轉載自微信公眾號「月伴飛魚」,作者日常加油站。轉載本文請聯系月伴飛魚公眾號。

代理模式

所謂代理,是指它與被代理對象實現了相同的接口,客戶端必須通過代理才能與被代理的目標類進行交互,而代理一般在交互的過程中(交互前后),進行某些特定的處理,比如在調用這個方法前做前置處理,調用這個方法后做后置處理。

代理又分為靜態代理和動態代理兩種方式,Spring的AOP采用的是動態代理的方式

Spring通過動態代理對類進行方法級別的切面增強,動態生成目標對象的代理類,并在代理類的方法中設置攔截器,通過執行攔截器中的邏輯增強了代理方法的功能,從而實現AOP。

策略模式

我們前面講到,Spring AOP是通過動態代理來實現的。

具體到代碼實現,Spring支持兩種動態代理實現方式,一種是JDK提供的動態代理實現方式,另一種是Cglib提供的動態代理實現方式。

Spring會在運行時動態地選擇不同的動態代理實現方式。這個應用場景實際上就是策略模式的典型應用場景。

我們只需要定義一個策略接口,讓不同的策略類都實現這一個策略接口。對應到Spring源碼,AopProxy是策略接口,JdkDynamicAopProxy、CglibAopProxy是兩個實現了AopProxy接口的策略類。

其中,AopProxy接口的定義如下所示:

在策略模式中,策略的創建一般通過工廠方法來實現。對應到Spring源碼,AopProxyFactory是一個工廠類接口,DefaultAopProxyFactory是一個默認的工廠類,用來創建AopProxy對象。

源碼如下所示:

策略模式的典型應用場景,一般是通過環境變量、狀態值、計算結果等動態地決定使用哪個策略。

對應到Spring源碼中,我們可以參看剛剛給出的DefaultAopProxyFactory類中的createAopProxy()函數的代碼實現。

其中,第10行代碼是動態選擇哪種策略的判斷條件。

裝飾器模式

我們知道,緩存一般都是配合數據庫來使用的。如果寫緩存成功,但數據庫事務回滾了,那緩存中就會有臟數據。

為了解決這個問題,我們需要將緩存的寫操作和數據庫的寫操作,放到同一個事務中,要么都成功,要么都失敗。

實現這樣一個功能,Spring使用到了裝飾器模式。

TransactionAwareCacheDecorator增加了對事務的支持,在事務提交、回滾的時候分別對Cache的數據進行處理。

TransactionAwareCacheDecorator實現Cache接口,并且將所有的操作都委托給targetCache來實現,對其中的寫操作添加了事務功能。這是典型的裝飾器模式的應用場景和代碼實現。

單例模式

單例模式是指一個類在整個系統運行過程中,只允許產生一個實例

在Spring中,Bean可以被定義為兩種模式:Prototype(多例)和Singleton(單例),Spring Bean默認是單例模式。

那Spring是如何實現單例模式的呢?

答案是通過單例注冊表的方式,具體來說就是使用了HashMap。簡化代碼如下:

  1. public class DefaultSingletonBeanRegistry { 
  2.      
  3.     //使用了線程安全容器ConcurrentHashMap,保存各種單實例對象 
  4.     private final Map singletonObjects = new ConcurrentHashMap; 
  5.  
  6.     protected Object getSingleton(String beanName) { 
  7.     //先到HashMap中拿Object 
  8.     Object singletonObject = singletonObjects.get(beanName); 
  9.      
  10.     //如果沒拿到通過反射創建一個對象實例,并添加到HashMap中 
  11.     if (singletonObject == null) { 
  12.       singletonObjects.put(beanName, 
  13.                            Class.forName(beanName).newInstance()); 
  14.    } 
  15.     
  16.    //返回對象實例 
  17.    return singletonObjects.get(beanName); 
  18.   } 

上面的代碼邏輯比較清晰,先到HashMap去拿單實例對象,沒拿到就創建一個添加到HashMap。

簡單工廠模式

有這樣一個場景:

當A對象需要調用B對象的方法時,我們需要在A中new一個B的實例,它的缺點是一旦需求發生變化,比如需要使用C類來代替B時,就要改寫A類的方法。

假如應用中有100個類以的方式耦合了B,那改起來就費勁了。

使用簡單工廠模式:

簡單工廠模式又叫靜態工廠方法,其實質是由一個工廠類根據傳入的參數,動態決定應該創建哪一個產品類。

其中Spring中的BeanFactory就是簡單工廠模式的體現,BeanFactory是Spring IOC容器中的一個核心接口,它的定義如下:

我們可以通過它的具體實現類(比如ClassPathXmlApplicationContext)來獲取Bean:

  1. BeanFactory bf = new ClassPathXmlApplicationContext("spring.xml"); 
  2. FlyFish flyFishBean = (FlyFish) bf.getBean("flyfishBean"); 

從上面代碼可以看到,使用者不需要自己來new對象,而是通過工廠類的方法getBean來獲取對象實例,這是典型的簡單工廠模式,只不過Spring是用反射機制來創建Bean的。

工廠方法模式

在簡單工廠中,由工廠類進行所有的邏輯判斷、實例創建;如果不想在工廠類中進行判斷,可以為不同的產品提供不同的工廠,不同的工廠生產不同的產品,每一個工廠都只對應一個相應的對象,這就是工廠方法模式。

Spring中的FactoryBean就是這種思想的體現,FactoryBean可以理解為工廠Bean,先來看看它的定義:

我們定義一個類FlyFishFactoryBean來實現FactoryBean接口,主要是在getObject方法里new一個FlyFish對象。這樣我們通過getBean(id) 獲得的是該工廠所產生的FlyFish的實例,而不是FlyFishFactoryBean本身的實例,像下面這樣:

  1. BeanFactory bf = new ClassPathXmlApplicationContext("spring.xml"); 
  2. FlyFish flyFishBean = (FlyFish) bf.getBean("flyfishBean"); 

觀察者模式

Spring中實現的觀察者模式包含三部分:Event事件(相當于消息)、Listener監聽者(相當于觀察者)、Publisher發送者(相當于被觀察者)

我們通過一個例子來看下Spring提供的觀察者模式是怎么使用的

  1. // Event事件 
  2. public class DemoEvent extends ApplicationEvent { 
  3.   private String message; 
  4.  
  5.   public DemoEvent(Object source, String message) { 
  6.     super(source); 
  7.   } 
  8.  
  9.   public String getMessage() { 
  10.     return this.message; 
  11.   } 
  12.  
  13. // Listener監聽者 
  14. @Component 
  15. public class DemoListener implements ApplicationListener { 
  16.   @Override 
  17.   public void onApplicationEvent(DemoEvent demoEvent) { 
  18.     String message = demoEvent.getMessage(); 
  19.     System.out.println(message); 
  20.   } 
  21.  
  22. // Publisher發送者 
  23. @Component 
  24. public class DemoPublisher { 
  25.   @Autowired 
  26.   private ApplicationContext applicationContext; 
  27.  
  28.   public void publishEvent(DemoEvent demoEvent) { 
  29.     this.applicationContext.publishEvent(demoEvent); 
  30.   } 

從代碼中,我們可以看出,主要包含三部分工作:

  • 定義一個繼承ApplicationEvent的事件(DemoEvent);
  • 定義一個實現了ApplicationListener的監聽器(DemoListener);
  • 定義一個發送者(DemoPublisher),發送者調用ApplicationContext來發送事件消息。

在Spring的實現中,觀察者注冊到了哪里呢?又是如何注冊的呢?

Spring把觀察者注冊到了ApplicationContext對象中。

實際上,具體到源碼來說,ApplicationContext只是一個接口,具體的代碼實現包含在它的實現類AbstractApplicationContext中。我把跟觀察者模式相關的代碼,如下。你只需要關注它是如何發送事件和注冊監聽者就好。

從上面的代碼中,我們發現,真正的消息發送,實際上是通過ApplicationEventMulticaster這個類來完成的。

下面這個類的源碼我只摘抄了最關鍵的一部分,也就是multicastEvent()這個消息發送函數,它通過線程池,支持異步非阻塞、同步阻塞這兩種類型的觀察者模式。

借助Spring提供的觀察者模式的骨架代碼,如果我們要在Spring下實現某個事件的發送和監聽,只需要做很少的工作,定義事件、定義監聽器、往ApplicationContext中發送事件就可以了,剩下的工作都由Spring框架來完成。

實際上,這也體現了Spring框架的擴展性,也就是在不需要修改任何代碼的情況下,擴展新的事件和監聽。

模板模式

我們經常在面試中被問到的一個問題:

請你說下Spring Bean的創建過程包含哪些主要的步驟。

這其中就涉及模板模式。它也體現了Spring的擴展性。利用模板模式,Spring能讓用戶定制Bean的創建過程。

下面是Spring Bean的整個生命周期,一張圖,清晰明了:

如果你仔細看過源碼會發現,實際上,這里的模板模式的實現,并不是標準的抽象類的實現方式,而是有點類似Callback回調的實現方式,也就是將要執行的函數封裝成對象(比如,初始化方法封裝成InitializingBean對象),傳遞給模板(BeanFactory)來執行。

觀察者模式和模板模式,這兩種模式能夠幫助我們創建擴展點,讓框架的使用者在不修改源碼的情況下,基于擴展點定制化框架功能。

適配器模式

在Spring MVC中,定義一個Controller最常用的方式是,通過@Controller注解來標記某個類是Controller類,通過@RequesMapping注解來標記函數對應的URL

不過,我們還可以通過讓類實現Controller接口或者Servlet接口,來定義一個Controller。

針對這三種定義方式,我寫了三段示例代碼,如下所示:

  1. // 方法一:通過@Controller、@RequestMapping來定義 
  2. @Controller 
  3. public class DemoController { 
  4.     @RequestMapping("/FlyFish"
  5.     public ModelAndView getEmployeeName() { 
  6.         ModelAndView model = new ModelAndView("FlyFish");         
  7.         model.addObject("message""FlyFish");        
  8.         return model;  
  9.     }   
  10.  
  11. // 方法二:實現Controller接口 + xml配置文件:配置DemoController與URL的對應關系 
  12. public class DemoController implements Controller { 
  13.     @Override 
  14.     public ModelAndView handleRequest(HttpServletRequest req, HttpServletResponse resp) throws Exception { 
  15.         ModelAndView model = new ModelAndView("FlyFish"); 
  16.         model.addObject("message""FlyFish"); 
  17.         return model; 
  18.     } 
  19.  
  20. // 方法三:實現Servlet接口 + xml配置文件:配置DemoController類與URL的對應關系 
  21. public class DemoServlet extends HttpServlet { 
  22.   @Override 
  23.   protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { 
  24.     this.doPost(req, resp); 
  25.   } 
  26.    
  27.   @Override 
  28.   protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { 
  29.     resp.getWriter().write("Hello World."); 
  30.   } 

在應用啟動的時候,Spring容器會加載這些Controller類,并且解析出URL對應的處理函數,封裝成Handler對象,存儲到HandlerMapping對象中。當有請求到來的時候,DispatcherServlet從HanderMapping中,查找請求URL對應的Handler,然后調用執行Handler對應的函數代碼,最后將執行結果返回給客戶端。

但是,不同方式定義的Controller,其函數的定義(函數名、入參、返回值等)是不統一的。

DispatcherServlet調用的是service()方法,DispatcherServlet需要根據不同類型的Controller,調用不同的函數。

Spring利用適配器模式,我們將不同方式定義的Controller類中的函數,適配為統一的函數定義。

我們再具體看下Spring的代碼實現。

Spring定義了統一的接口HandlerAdapter,并且對每種Controller定義了對應的適配器類。

這些適配器類包括:AnnotationMethodHandlerAdapter、SimpleControllerHandlerAdapter、SimpleServletHandlerAdapter等。

 

在DispatcherServlet類中,我們就不需要區分對待不同的Controller對象了,統一調用HandlerAdapter的handle()函數就可以了

 

責任編輯:武曉燕 來源: 月伴飛魚
相關推薦

2023-07-11 08:50:34

2020-03-18 09:43:37

開發技能代碼

2019-05-29 17:20:07

Spring設計模式Java

2022-09-21 09:01:27

Spring設計模式框架,

2024-11-26 14:29:48

2021-06-08 07:04:46

Dubbo設計模式

2020-01-02 15:43:29

Spring設計策略

2021-04-23 14:14:46

設計模式對象

2015-07-13 09:45:32

阿里校招

2025-05-09 09:05:00

Spring框架設計模式

2018-05-10 15:48:47

面試面試官Java

2020-11-05 10:01:35

系統設計軟件

2024-10-11 17:09:27

2014-09-19 11:17:48

面試題

2020-06-04 14:40:40

面試題Vue前端

2021-10-27 17:57:35

設計模式場景

2010-04-27 13:49:04

Oracle數據庫

2023-11-13 07:37:36

JS面試題線程

2011-03-24 13:27:37

SQL

2024-05-11 08:33:45

JavaJVM虛擬機
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 九九久久在线看 | www.中文字幕av | 色资源在线 | 日本爱爱| 国产色网 | 亚洲一区二区久久 | 免费在线一区二区 | 日韩欧美在线视频播放 | 国产传媒在线观看 | 欧美日韩视频在线播放 | 亚洲精彩视频 | 永久网站| 国产精品视频免费观看 | 精品在线一区二区 | 国产精品久久久久久久岛一牛影视 | 欧美视频在线看 | 精品日韩在线观看 | 亚洲福利网站 | 在线免费av观看 | 欧美精品 在线观看 | 久久久久久久久国产 | 亚洲天堂精品一区 | 午夜免费| 日韩一区二区三区视频在线播放 | 91色在线视频 | 精品国产不卡一区二区三区 | 很很干很很日 | 成人免费一级视频 | 亚洲国产精品一区 | 精品91久久 | 日韩欧美一区二区三区免费观看 | 欧美日韩不卡合集视频 | 免费特级黄毛片 | 91精品导航 | 国产精品国产精品国产专区不片 | 九色国产 | 亚洲成av| 蜜桃在线视频 | 99爱在线观看| 成人午夜免费在线视频 | 爱爱视频网 |