真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些

這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)公司!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序開發(fā)、集團企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了橫峰免費建站歡迎大家使用!

代理模式是java中最常用的設(shè)計模式之一,尤其是在spring框架中廣泛應(yīng)用。對于java的代理模式,一般可分為:靜態(tài)代理、動態(tài)代理、以及CGLIB實現(xiàn)動態(tài)代理。

對于上述三種代理模式,分別進行說明。

1.靜態(tài)代理

靜態(tài)代理其實就是在程序運行之前,提前寫好被代理方法的代理類,編譯后運行。在程序運行之前,class已經(jīng)存在。
下面我們實現(xiàn)一個靜態(tài)代理demo:

JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些

靜態(tài)代理

定義一個接口Target

package com.test.proxy;

public interface Target {

  public String execute();
}

TargetImpl 實現(xiàn)接口Target

package com.test.proxy;

public class TargetImpl implements Target {

  @Override
  public String execute() {
    System.out.println("TargetImpl execute!");
    return "execute";
  }
}

代理類

package com.test.proxy;

public class Proxy implements Target{

  private Target target;

  public Proxy(Target target) {
    this.target = target;
  }

  @Override
  public String execute() {
    System.out.println("perProcess");
    String result = this.target.execute();
    System.out.println("postProcess");
    return result;
  }
}

測試類:

package com.test.proxy;

public class ProxyTest {

  public static void main(String[] args) {

    Target target = new TargetImpl();
    Proxy p = new Proxy(target);
    String result = p.execute();
    System.out.println(result);
  }

}

運行結(jié)果:

perProcess
TargetImpl execute!
postProcess
execute

靜態(tài)代理需要針對被代理的方法提前寫好代理類,如果被代理的方法非常多則需要編寫很多代碼,因此,對于上述缺點,通過動態(tài)代理的方式進行了彌補。

2.動態(tài)代理

動態(tài)代理主要是通過反射機制,在運行時動態(tài)生成所需代理的class.

JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些

動態(tài)代理

接口

package com.test.dynamic;

public interface Target {

  public String execute();
}

實現(xiàn)類

package com.test.dynamic;

public class TargetImpl implements Target {

  @Override
  public String execute() {
    System.out.println("TargetImpl execute!");
    return "execute";
  }
}

代理類

package com.test.dynamic;


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class DynamicProxyHandler implements InvocationHandler{

  private Target target;

  public DynamicProxyHandler(Target target) {
    this.target = target;
  }

  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("========before==========");
    Object result = method.invoke(target,args);
    System.out.println("========after===========");
    return result;
  }
}

測試類

package com.test.dynamic;

import java.lang.reflect.Proxy;

public class DynamicProxyTest {

  public static void main(String[] args) {
    Target target = new TargetImpl();
    DynamicProxyHandler handler = new DynamicProxyHandler(target);
    Target proxySubject = (Target) Proxy.newProxyInstance(TargetImpl.class.getClassLoader(),TargetImpl.class.getInterfaces(),handler);
    String result = proxySubject.execute();
    System.out.println(result);
  }

}

運行結(jié)果:

========before==========
TargetImpl execute!
========after===========
execute

無論是動態(tài)代理還是靜態(tài)帶領(lǐng),都需要定義接口,然后才能實現(xiàn)代理功能。這同樣存在局限性,因此,為了解決這個問題,出現(xiàn)了第三種代理方式:cglib代理。

3.cglib代理

CGLib采用了非常底層的字節(jié)碼技術(shù),其原理是通過字節(jié)碼技術(shù)為一個類創(chuàng)建子類,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,順勢織入橫切邏輯。JDK動態(tài)代理與CGLib動態(tài)代理均是實現(xiàn)Spring AOP的基礎(chǔ)。

JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些

cglib動態(tài)代理

目標(biāo)類

package com.test.cglib;

public class Target {

  public String execute() {
    String message = "-----------test------------";
    System.out.println(message);
    return message;
  }
}

通用代理類:

package com.test.cglib;

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class MyMethodInterceptor implements MethodInterceptor{

  @Override
  public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
    System.out.println(">>>>MethodInterceptor start...");
    Object result = proxy.invokeSuper(obj,args);
    System.out.println(">>>>MethodInterceptor ending...");
    return "result";
  }
}

測試類

package com.test.cglib;

import net.sf.cglib.proxy.Enhancer;

public class CglibTest {

  public static void main(String ... args) {
    System.out.println("***************");
    Target target = new Target();
    CglibTest test = new CglibTest();
    Target proxyTarget = (Target) test.createProxy(Target.class);
    String res = proxyTarget.execute();
    System.out.println(res);
  }

  public Object createProxy(Class targetClass) {
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(targetClass);
    enhancer.setCallback(new MyMethodInterceptor());
    return enhancer.create();
  }

}

執(zhí)行結(jié)果:

***************
>>>>MethodInterceptor start...
-----------test------------
>>>>MethodInterceptor ending...
result

代理對象的生成過程由Enhancer類實現(xiàn),大概步驟如下:

1、生成代理類Class的二進制字節(jié)碼;

2、通過Class.forName加載二進制字節(jié)碼,生成Class對象;

3、通過反射機制獲取實例構(gòu)造,并初始化代理類對象。

上述就是小編為大家分享的JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


網(wǎng)站題目:JAVA中靜態(tài)代理與動態(tài)代理的區(qū)別有哪些
轉(zhuǎn)載注明:http://weahome.cn/article/gepsse.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部