在Java Spring框架中使用的設(shè)計模式有哪些
1 簡單工廠模式
當(dāng)A對象需要調(diào)用B對象的方法時,我們需要在A中new一個B的實例,我們把這種方式叫作硬編碼耦合,缺點是一旦需求發(fā)生變化,比如需要使用C類來代替B時,就要改寫A類的方法。
假如應(yīng)用中有上千個類以硬編碼的方式耦合了B,改就很頭疼。
于是有了簡單工廠模式,又叫靜態(tài)工廠方法,就是由一個工廠類根據(jù)傳入?yún)?shù),動態(tài)決定應(yīng)該創(chuàng)建哪個產(chǎn)品類。
Spring中的BeanFactory就是簡單工廠模式的體現(xiàn),BeanFactory是Spring IOC容器中的一個核心接口,它的定義如下:

可以通過它的具體實現(xiàn)類(如ClassPathXmlApplicationContext)獲取Bean:
BeanFactory bf = new ClassPathXmlApplicationContext("spring.xml");
User userBean = (User) bf.getBean("userBean");
使用者無需自己new,而是通過工廠類的方法getBean獲取對象實例,這就是簡單工廠模式,只不過Spring是用反射創(chuàng)建Bean。
2 工廠方法模式
簡單工廠中,由工廠類進行所有的邏輯判斷、實例創(chuàng)建。
如果不想在工廠類中進行判斷,可為不同產(chǎn)品提供不同工廠,不同工廠生產(chǎn)不同產(chǎn)品,每個工廠都只對應(yīng)一個相應(yīng)對象,這就是工廠方法模式。
Spring FactoryBean,工廠Bean

定義一個類UserFactoryBean實現(xiàn)FactoryBean接口,主要是在getObject方法里new一個User對象。
這樣通過getBean(id) 獲得的是該工廠所產(chǎn)生的User實例,而非UserFactoryBean本身實例:
BeanFactory bf = new ClassPathXmlApplicationContext("user.xml");
User userBean = (User) bf.getBean("userFactoryBean");
3 單例模式
一個類在整個系統(tǒng)運行過程中,只允許產(chǎn)生一個實例。
Spring Bean默認(rèn)是單例模式。Spring 通過單例注冊表(HashMap)方式:
public class DefaultSingletonBeanRegistry {
// 使用 ConcurrentHashMap 保存各種單實例對象
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<String, Object>;
protected Object getSingleton(String beanName) {
// 先到 Map 中拿Object
Object singletonObject = singletonObjects.get(beanName);
// 若沒拿到通過反射創(chuàng)建一個對象實例,并添加到HashMap中
if (singletonObject == null) {
singletonObjects.put(beanName,
Class.forName(beanName).newInstance());
}
// 返回對象實例
return singletonObjects.get(beanName);
}
}
4 代理模式
它與被代理對象實現(xiàn)相同接口,客戶端必須通過代理才能與被代理的目標(biāo)類進行交互,而代理一般在交互的過程中(交互前后),進行某些特定處理,比如在調(diào)用這個方法前做前置處理,調(diào)用這個方法后做后置處理。
好處
可以在目標(biāo)對象業(yè)務(wù)功能的基礎(chǔ)上添加一些公共的邏輯,比如我們想給目標(biāo)對象加入日志、權(quán)限管理和事務(wù)控制等功能,我們就可以使用代理類來完成,而沒必要修改目標(biāo)類,從而使得目標(biāo)類保持穩(wěn)定。
符合開閉原則,不要隨意修改別人寫好的代碼或方法。
代理又分為
靜態(tài)代理
需要定義接口,被代理對象(目標(biāo)對象)與代理對象(Proxy)一起實現(xiàn)相同接口:
// 抽象接口
public interface IStudentDao {
void save();
}
// 目標(biāo)對象
public class StudentDao implements IStudentDao {
public void save() {
System.out.println("保存成功");
}
}
// 代理對象
public class StudentDaoProxy implements IStudentDao{
//持有目標(biāo)對象的引用
private IStudentDao target;
public StudentDaoProxy(IStudentDao target){
this.target = target;
}
//在目標(biāo)功能對象方法的前后加入事務(wù)控制
public void save() {
System.out.println("開始事務(wù)");
target.save();//執(zhí)行目標(biāo)對象的方法
System.out.println("提交事務(wù)");
}
}
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象
StudentDao target = new StudentDao();
//創(chuàng)建代理對象,把目標(biāo)對象傳給代理對象,建立代理關(guān)系
StudentDaoProxy proxy = new StudentDaoProxy(target);
//執(zhí)行的是代理的方法
proxy.save();
}
動態(tài)代理
Spring AOP采用動態(tài)代理,即代理類在程序運行時由JVM動態(tài)創(chuàng)建。
靜態(tài)代理的例子中,代理類(StudentDaoProxy)是自定義的,在程序運行之前就已經(jīng)編譯完成。
而動態(tài)代理,代理類并不是在Java代碼中定義,而是在運行時根據(jù)我們在Java代碼中的“指示”動態(tài)生成的。
怎么“指示”JDK去動態(tài)地生成代理類呢?
在Java的java.lang.reflect包里提供了一個Proxy類和一個InvocationHandler接口,通過這個類和這個接口可以生成動態(tài)代理對象:
1.定義一個InvocationHandler類,將需要擴展的邏輯集中放到這個類中。
比如下面的例子模擬了添加事務(wù)控制:
public class MyInvocationHandler implements InvocationHandler {
private Object obj;
public MyInvocationHandler(Object obj){
this.obj=obj;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("開始事務(wù)");
Object result = method.invoke(obj, args);
System.out.println("開始事務(wù)");
return result;
}
}
2.使用 Proxy#newProxyInstance 動態(tài)創(chuàng)建代理對象
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象StudentDao
IStudentDao stuDAO = new StudentDao();
//創(chuàng)建MyInvocationHandler對象
InvocationHandler handler = new MyInvocationHandler(stuDAO);
//使用Proxy.newProxyInstance動態(tài)的創(chuàng)建代理對象stuProxy
IStudentDao stuProxy = (IStudentDao)
Proxy.newProxyInstance(stuDAO.getClass().getClassLoader(), stuDAO.getClass().getInterfaces(), handler);
//動用代理對象的方法
stuProxy.save();
}
動態(tài)代理的優(yōu)勢在于可以很方便地對代理類的函數(shù)進行統(tǒng)一的處理,而不用修改每個代理類中的方法。
Spring實現(xiàn)了通過動態(tài)代理對類進行方法級別的切面增強,即動態(tài)生成目標(biāo)對象的代理類,并在代理類的方法中設(shè)置攔截器,通過執(zhí)行攔截器中的邏輯增強了代理方法的功能,從而實現(xiàn)AOP。
到此這篇關(guān)于在Java Spring框架中使用的設(shè)計模式有哪些的文章就介紹到這了,更多相關(guān)Java Spring 設(shè)計模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring?boot?mybatis日志輸出到控制臺的方法實踐
在開發(fā)過程中我們往往需要打印出SQL語句,這樣就方便我們監(jiān)控問題,本文主要介紹了spring?boot?mybatis日志輸出到控制臺的方法實踐,具有一定的參考價值,感興趣的可以了解一下2024-05-05
詳解Springboot之Logback的使用學(xué)習(xí)
Logback是SpringBoot內(nèi)置的日志處理框架,你會發(fā)現(xiàn)spring-boot-starter其中包含了spring-boot-starter-logging,該依賴內(nèi)容就是Spring Boot默認(rèn)的日志框架logback,本文詳細(xì)介紹了該框架 ,需要的朋友可以參考下2021-05-05
Java發(fā)送http請求的示例(get與post方法請求)
這篇文章主要介紹了Java發(fā)送http請求的示例(get與post方法請求),幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2021-01-01
springcloud Zuul動態(tài)路由的實現(xiàn)
這篇文章主要介紹了springcloud Zuul動態(tài)路由的實現(xiàn),詳細(xì)的介紹了什么是Zuu及其動態(tài)路由的實現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-11-11
Springmvc conver實現(xiàn)原理及用法解析
這篇文章主要介紹了Springmvc conver實現(xiàn)原理及用法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10
Spring中的@RestController注解詳細(xì)解析
這篇文章主要介紹了Spring中的@RestController注解詳細(xì)解析,@RestController 是 Spring Framework 中的一個注解,用于標(biāo)識一個類為 RESTful Web 服務(wù)的控制器(Controller),處理 HTTP 請求并返回相應(yīng)的數(shù)據(jù),2024-01-01
SpringBoot利用限速器RateLimiter實現(xiàn)單機限流的示例代碼
本文主要介紹了SpringBoot利用限速器RateLimiter實現(xiàn)單機限流的示例代碼,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01

