Java代理深入講解之靜態(tài)代理
什么是代理
代理就是給目標(biāo)對(duì)象一個(gè)代理對(duì)象,并由代理對(duì)象控制目標(biāo)的引用。
為什么要使用代理模式
1、通過引入代理對(duì)象的方式,可以間接的訪問目標(biāo)對(duì)象,避免直接訪問目標(biāo)對(duì)象給系統(tǒng)帶來不必要的復(fù)雜性。
2、通過代理對(duì)象可以對(duì)原有的業(yè)務(wù)進(jìn)行業(yè)務(wù)增強(qiáng)處理。
舉例:如果我們需要買國(guó)外的某一件商品A,這個(gè)時(shí)候我們一般有兩個(gè)途徑要么直接去國(guó)外買,要么可以找一些代購人員幫我們?nèi)ベ徺I。在這種情況下,我們由于直接去國(guó)外買,實(shí)在是太耗軟妹幣,而且還要花時(shí)間等等,這個(gè)時(shí)候我們最優(yōu)的選擇就是找代購購買,這樣也幫我們省去了很多麻煩的事情。

代理模式類圖

代碼示例
抽象對(duì)象:
public interface ITargetFactoryService {
void sale(String name);
}
目標(biāo)對(duì)象:
@Slf4j
public class TargetFactoryServiceImpl implements ITargetFactoryService {
@Override
public void sale(String name) {
log.info(name+"購買了商品A");
}
}
代理對(duì)象:
@Slf4j
public class ProxyImpl implements ITargetFactoryService {
public ITargetFactoryService service;
public ProxyImpl(ITargetFactoryService service){
super();
this.service = service;
}
@Override
public void sale(String name) {
before();
service.sale("代購");
after();
}
/**
* 后置增強(qiáng)
*/
private void after() {
log.info("代購在購買后得到了市場(chǎng)調(diào)研結(jié)果");
}
/**
* 前置增強(qiáng)
*/
private void before() {
log.info("代購在購買前做了市場(chǎng)調(diào)研");
}
}
測(cè)試類:
@Slf4j
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class, JdbcTemplateAutoConfiguration.class})
public class SpsringJdbcApplication {
public static void main(String[] args) {
TargetFactoryServiceImpl service = new TargetFactoryServiceImpl();
ProxyImpl proxy = new ProxyImpl(service);
proxy.sale("代購");
SpringApplication.run(SpsringJdbcApplication.class, args);
}
}
測(cè)試結(jié)果:

我們可以在代碼示例中清晰的看到,在代理類中,代理對(duì)象包含了目標(biāo)對(duì)象,并且在業(yè)務(wù)處理上進(jìn)行了一定的業(yè)務(wù)擴(kuò)展,但是卻和目標(biāo)對(duì)象繼承于同一個(gè)接口。但是此擴(kuò)展基于Spring AOP來講,以更加專業(yè)的叫法為前置增強(qiáng)、后置增強(qiáng)。
此類代理便是我們常說的靜態(tài)代理,靜態(tài)代理適合在業(yè)務(wù)比較簡(jiǎn)單,實(shí)現(xiàn)類少,需求變化不頻繁,但是卻要對(duì)原有目標(biāo)服務(wù)對(duì)象功能進(jìn)行擴(kuò)展,并且不去修改原有服務(wù),這個(gè)時(shí)候我們就可以選擇使用靜態(tài)代理。
靜態(tài)代理的缺點(diǎn)
如果此時(shí)我們業(yè)務(wù)需要進(jìn)行擴(kuò)展,我們的代購?fù)瑢W(xué)在經(jīng)過市場(chǎng)調(diào)研以后,發(fā)現(xiàn)商品B更加受大家歡迎,這個(gè)時(shí)候我們就需要對(duì)自己的業(yè)務(wù)進(jìn)行擴(kuò)展了,怎么擴(kuò)展呢?一起接著往下看。
抽象對(duì)象:
public interface ITargetFactoryBService {
void saleB(String name);
}
目標(biāo)對(duì)象:
@Slf4j
public class ITargetFactoryBServiceImpl implements ITargetFactoryBService {
@Override
public void saleB(String name) {
log.info(name + "購買了商品B");
}
}
代理對(duì)象:
@Slf4j
public class ProxyTwoImpl implements ITargetFactoryService, ITargetFactoryBService {
public ITargetFactoryService service;
public ITargetFactoryBService bService;
public ProxyTwoImpl(ITargetFactoryService service,ITargetFactoryBService bService){
super();
this.service = service;
this.bService = bService;
}
@Override
public void sale(String name) {
before();
service.sale("代購");
after();
}
@Override
public void saleB(String name) {
before();
bService.saleB("代購");
after();
}
/**
* 后置增強(qiáng)
*/
private void after() {
log.info("代購在購買后得到了市場(chǎng)調(diào)研結(jié)果");
}
/**
* 前置增強(qiáng)
*/
private void before() {
log.info("代購在購買前做了市場(chǎng)調(diào)研");
}
}
測(cè)試類:
@Slf4j
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class, JdbcTemplateAutoConfiguration.class})
public class SpsringJdbcApplication {
public static void main(String[] args) {
TargetFactoryServiceImpl service = new TargetFactoryServiceImpl();
ITargetFactoryBServiceImpl bService = new ITargetFactoryBServiceImpl();
ProxyTwoImpl proxy2 = new ProxyTwoImpl(service, bService);
proxy2.sale("代購");
proxy2.saleB("代購");
SpringApplication.run(SpsringJdbcApplication.class, args);
}
}
結(jié)果:

我們可以看到,在實(shí)現(xiàn)業(yè)務(wù)擴(kuò)展的時(shí)候,需要對(duì)原有的代理類進(jìn)行修改,如果后期我們需要擴(kuò)展的業(yè)務(wù)較多的時(shí)候,這個(gè)類將變的更加繁雜,大量的繼承以及方法重寫,以至于牽一發(fā)而動(dòng)全身,所以在這種業(yè)務(wù)擴(kuò)展性高、業(yè)務(wù)變化頻繁的情況下我們不建議使用靜態(tài)代理。
靜態(tài)代理總結(jié):
1、違反Java設(shè)計(jì)模式開閉原則,即:程序?qū)ν鈹U(kuò)展開放,對(duì)修改關(guān)閉。當(dāng)需求進(jìn)行變更時(shí),我們應(yīng)該是新增代碼塊來實(shí)現(xiàn),而不是在原來的代碼中進(jìn)行修改實(shí)現(xiàn)。
2、擴(kuò)展性很差。
3、可維護(hù)性差。
4、代碼耦合度高。
總結(jié)
到此這篇關(guān)于Java代理深入講解之靜態(tài)代理的文章就介紹到這了,更多相關(guān)Java靜態(tài)代理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java中List數(shù)組用逗號(hào)分隔開轉(zhuǎn)成字符串2種方法
在我們?nèi)粘i_發(fā)中,在前后端交互的時(shí)候會(huì)遇到多個(gè)id或其他字段存放到一個(gè)字段中,這時(shí)我們會(huì)遇到一個(gè)List(集合)---->String(單個(gè)字段),這篇文章主要給大家介紹了關(guān)于java中List數(shù)組用逗號(hào)分隔開轉(zhuǎn)成字符串的2種方法,需要的朋友可以參考下2023-10-10
SpringMVC中的HandlerMapping和HandlerAdapter詳解
這篇文章主要介紹了SpringMVC中的HandlerMapping和HandlerAdapter詳解,在Spring MVC中,HandlerMapping(處理器映射器)用于確定請(qǐng)求處理器對(duì)象,請(qǐng)求處理器可以是任何對(duì)象,只要它們使用了@Controller注解或注解@RequestMapping,需要的朋友可以參考下2023-08-08
關(guān)于pom.xml中maven無法下載springcloud包問題
小編遇到這樣一個(gè)問題spring-cloud-starter-feign,spring-cloud-starter-eureka 一直無法下載,maven倉庫中包路徑顯示為unknown,怎么解決呢?下面小編給大家?guī)砹藀om.xml中maven無法下載springcloud包問題,需要的朋友可以參考下2022-08-08
java?web項(xiàng)目Session獲取不到問題及解決
這篇文章主要介紹了java?web項(xiàng)目Session獲取不到問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01
spring boot集成rabbitmq的實(shí)例教程
這篇文章主要給大家介紹了關(guān)于spring boot集成rabbitmq的相關(guān)資料,springboot集成RabbitMQ非常簡(jiǎn)單,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11
spring boot如何使用AOP統(tǒng)一處理web請(qǐng)求
這篇文章主要介紹了spring boot如何使用AOP統(tǒng)一處理web請(qǐng)求,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
Java中的Vector和ArrayList區(qū)別及比較
這篇文章主要介紹了Java中的Vector和ArrayList區(qū)別及比較,本文從API、同步、數(shù)據(jù)增長(zhǎng)、使用模式4個(gè)方面總結(jié)了它們之間的不同之處,需要的朋友可以參考下2015-03-03
如何在JDK 9中更簡(jiǎn)潔使用 try-with-resources 語句
本文詳細(xì)介紹了自 JDK 7 引入的 try-with-resources 語句的原理和用法,以及介紹了 JDK 9 對(duì) try-with-resources 的改進(jìn),使得用戶可以更加方便、簡(jiǎn)潔的使用 try-with-resources 語句。,需要的朋友可以參考下2019-06-06
java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序詳解
這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序,結(jié)合具體實(shí)例形式分析了java樹形選擇排序的原理、實(shí)現(xiàn)技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-05-05

