自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

三分鐘吃透代理技術(shù)!

開發(fā) 前端
如果目標(biāo)對象至少實(shí)現(xiàn)了一個(gè)接口,那么就用JDK動(dòng)態(tài)代理,所有由目標(biāo)對象實(shí)現(xiàn)的接口將全部都被代理。如果目標(biāo)對象沒有實(shí)現(xiàn)任何接口,就是個(gè)類,那么就用CGLIB代理。

最近有網(wǎng)友問了我一些問題,什么是代理,又該在什么地方使用。結(jié)合之前的討論,這篇文章我們一起細(xì)致的講解一下關(guān)于代理的一些問題。

在 Java 中,代理通常分為兩類:

  • 靜態(tài)代理
  • 動(dòng)態(tài)代理

兩者技術(shù)實(shí)現(xiàn)是不一樣的,具體有什么區(qū)別呢?下面我們一起來看看。

一、靜態(tài)代理

我們先說靜態(tài)代理的實(shí)現(xiàn)方式,為什么不推薦使用靜態(tài)代理?

1.繼承方式實(shí)現(xiàn)代理(靜態(tài)代理中的繼承代理)

//目標(biāo)對象

public class UserImpl {
    public void system(){
        System.out.println("輸出測試");
    }
}

//代理對象
public class Proxy extends UserImpl {

    @Override
    public void system() {
        super.system();
        System.out.println("增強(qiáng)之后的輸出");
    }
}

//測試類

public class TestMain {
    public static void main(String[] args) {
        UserImpl user = new Proxy();
        user.system();
    }
}

靜態(tài)代理可以看出來一點(diǎn)問題吧?

每次代理都要實(shí)現(xiàn)一個(gè)類,導(dǎo)致項(xiàng)目中代碼很多;你每次想要代理,都要去實(shí)現(xiàn)一個(gè)類,代碼就會成堆的增加,然后你就會發(fā)現(xiàn)項(xiàng)目的類就會越來越多,就會導(dǎo)致你們的項(xiàng)目顯得很臃腫。而且代碼的復(fù)用性太低了,并且耦合度非常高,這個(gè)我們所說的高內(nèi)聚低耦合是相悖的。

二、動(dòng)態(tài)代理

我們在看一下這個(gè)動(dòng)態(tài)代理:

//接口類
public interface Italk {
    public void talk(String msg);
}
//實(shí)現(xiàn)類
public class People implements Italk {
        public String username;
        public String age;
        public String getName() {
        return username;
    }
    public void setName(String name) {
        this.username= name;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }
    public People(String name1, String age1) {
        this.username= name1;
        this.age = age1;
    }
    public void talk(String msg) {
        System.out.println(msg+"!你好,我是"+username+",我年齡是"+age);
     }
}
//代理類
public class TalkProxy implements Italk {
        Italk talker;
    public TalkProxy (Italk talker) {
        //super();
        this.talker=talker;
    }
    public void talk(String msg) {
        talker.talk(msg);
    }
    public void talk(String msg,String singname) {
        talker.talk(msg);
        sing(singname);
    }
    private void sing(String singname){
         System.out.println("唱歌:"+singname);
    }
}
//測試
public class MyProxyTest {
    //代理模式
    public static void main(String[] args) {
    //不需要執(zhí)行額外方法的
    Italk people1=new People("湖海散人","18");
    people1.talk("No ProXY Test");
    System.out.println("-----------------------------");
    //需要執(zhí)行額外方法的
    TalkProxy talker=new TalkProxy(people1);
    talker.talk("ProXY Test","七里香");
    }
}

代碼解析如下:

一個(gè) Italk 接口,有空的方法 talk()(說話),所有的 people 對象都實(shí)現(xiàn)(implements)這個(gè)接口,實(shí)現(xiàn) talk() 方法,前端有很多地方都將 people 實(shí)例化,執(zhí)行 talk 方法,后來發(fā)現(xiàn)這些前端里有一些除了要說話以外還要唱歌(sing),那么我們既不能在 Italk 接口里增加 sing() 方法,又不能在每個(gè)前端都增加 sing 方法,我們只有增加一個(gè)代理類 talkProxy ,這個(gè)代理類里實(shí)現(xiàn) talk 和 sing 方法,然后在需要 sing 方法的客戶端調(diào)用代理類即可,

這也是實(shí)現(xiàn)動(dòng)態(tài)代理的方式,是通過實(shí)現(xiàn)(implements)的方式來實(shí)現(xiàn)的,這種方法的優(yōu)點(diǎn),在編碼時(shí),代理邏輯與業(yè)務(wù)邏輯互相獨(dú)立,各不影響,沒有侵入,沒有耦合。

三、cgLib代理

還有一種是cgLib的代理,這種代理則是適合那些沒有接口抽象的類代理,而Java 動(dòng)態(tài)代理適合于那些有接口抽象的類代理。

我們來通過代碼了解一下他到底是怎么玩的。

/**
 * 業(yè)務(wù)類,
 */
public class TestService {
    public TestService() {
        System.out.println("TestService的構(gòu)造");
    }
    /**
     * 該方法不能被子類覆蓋,Cglib是無法代理final修飾的方法的
     */
    final public String sayOthers(String name) {
        System.out.println("TestService:sayOthers>>"+name);
        return null;
    }
    public void sayHello() {
        System.out.println("TestService:sayHello");
    }
}
/**
 * 自定義MethodInterceptor
 */
public class MethodInterceptorTest implements MethodInterceptor {
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("======插入前置通知======");
        Object object = methodProxy.invokeSuper(o, objects);
        System.out.println("======插入后者通知======");
        return object;
    }
}
/**
 * 測試
 */
public class Client {
    public static void main(String[] args) {
        // 代理類class文件存入本地磁盤方便我們反編譯查看源碼
        System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\code");
        // 通過CGLIB動(dòng)態(tài)代理獲取代理對象的過程
        Enhancer enhancer = new Enhancer();
        // 設(shè)置enhancer對象的父類
        enhancer.setSuperclass(TestService.class);
        // 設(shè)置enhancer的回調(diào)對象
        MethodInterceptorTest t =  new  MethodInterceptorTest();
        enhancer.setCallback(t);
        // 創(chuàng)建代理對象
        TestService proxy= (TestService)enhancer.create();
        // 通過代理對象調(diào)用目標(biāo)方法
        proxy.sayHello();
    }
}

運(yùn)行結(jié)果:

CGLIB debugging enabled, writing to 'D:\code'
TestService的構(gòu)造
======插入前置通知======
TestService:sayHello
======插入后者通知======

實(shí)現(xiàn)CGLIB動(dòng)態(tài)代理必須實(shí)現(xiàn)MethodInterceptor(方法攔截器)接口,

這個(gè)接口只有一個(gè)intercept()方法,這個(gè)方法有4個(gè)參數(shù):

  • obj表示增強(qiáng)的對象,即實(shí)現(xiàn)這個(gè)接口類的一個(gè)對象;
  • method表示要被攔截的方法;
  • args表示要被攔截方法的參數(shù);
  • proxy表示要觸發(fā)父類的方法對象;

四、代理的使用

那么什么時(shí)候使用靜態(tài)態(tài)代理,什么時(shí)候使用動(dòng)態(tài)代理和cgLib代理呢?

一般情況靜態(tài)代理是很少是用的,因?yàn)樗麑Υa的復(fù)用性或者說是耦合度都非常不友好,不推薦使用。

如果目標(biāo)對象至少實(shí)現(xiàn)了一個(gè)接口,那么就用JDK動(dòng)態(tài)代理,所有由目標(biāo)對象實(shí)現(xiàn)的接口將全部都被代理。

如果目標(biāo)對象沒有實(shí)現(xiàn)任何接口,就是個(gè)類,那么就用CGLIB代理。

責(zé)任編輯:武曉燕 來源: Java極客技術(shù)
相關(guān)推薦

2024-10-15 09:18:30

2024-05-16 11:13:16

Helm工具release

2009-11-09 12:55:43

WCF事務(wù)

2022-02-17 09:24:11

TypeScript編程語言javaScrip

2021-04-20 13:59:37

云計(jì)算

2023-12-27 08:15:47

Java虛擬線程

2024-01-16 07:46:14

FutureTask接口用法

2024-08-30 08:50:00

2013-06-28 14:30:26

棱鏡計(jì)劃棱鏡棱鏡監(jiān)控項(xiàng)目

2020-06-30 10:45:28

Web開發(fā)工具

2021-12-17 07:47:37

IT風(fēng)險(xiǎn)框架

2021-02-03 14:31:53

人工智能人臉識別

2024-01-12 07:38:38

AQS原理JUC

2024-07-05 09:31:37

2009-11-05 16:04:19

Oracle用戶表

2020-06-29 07:42:20

邊緣計(jì)算云計(jì)算技術(shù)

2023-12-04 18:13:03

GPU編程

2025-02-24 10:40:55

2024-04-01 09:59:08

消息隊(duì)列通信微服務(wù)

2022-03-26 09:06:40

ActorCSP模型
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號