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

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

利用java實現動態(tài)代理的方法

這篇文章將為大家詳細講解有關利用java實現動態(tài)代理的方法,文章內容質量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關知識有一定的了解。

創(chuàng)新互聯致力于互聯網品牌建設與網絡營銷,包括成都做網站、成都網站設計、成都外貿網站建設、SEO優(yōu)化、網絡推廣、整站優(yōu)化營銷策劃推廣、電子商務、移動互聯網營銷等。創(chuàng)新互聯為不同類型的客戶提供良好的互聯網應用定制及解決方案,創(chuàng)新互聯核心團隊10多年專注互聯網開發(fā),積累了豐富的網站經驗,為廣大企業(yè)客戶提供一站式企業(yè)網站建設服務,在網站建設行業(yè)內樹立了良好口碑。

java 動態(tài)代理的方法總結

AOP的攔截功能是由java中的動態(tài)代理來實現的。說白了,就是在目標類的基礎上增加切面邏輯,生成增強的目標類(該切面邏輯或者在目標類函數執(zhí)行之前,或者目標類函數執(zhí)行之后,或者在目標類函數拋出異常時候執(zhí)行。不同的切入時機對應不同的Interceptor的種類,如BeforeAdviseInterceptor,AfterAdviseInterceptor以及ThrowsAdviseInterceptor等)。

那么動態(tài)代理是如何實現將切面邏輯(advise)織入到目標類方法中去的呢?下面我們就來詳細介紹并實現AOP中用到的兩種動態(tài)代理。

AOP的源碼中用到了兩種動態(tài)代理來實現攔截切入功能:jdk動態(tài)代理和cglib動態(tài)代理。兩種方法同時存在,各有優(yōu)劣。jdk動態(tài)代理是由Java內部的反射機制來實現的,cglib動態(tài)代理底層則是借助asm來實現的??偟膩碚f,反射機制在生成類的過程中比較高效,而asm在生成類之后的相關執(zhí)行過程中比較高效(可以通過將asm生成的類進行緩存,這樣解決asm生成類過程低效問題)。還有一點必須注意:jdk動態(tài)代理的應用前提,必須是目標類基于統(tǒng)一的接口。如果沒有上述前提,jdk動態(tài)代理不能應用。由此可以看出,jdk動態(tài)代理有一定的局限性,cglib這種第三方類庫實現的動態(tài)代理應用更加廣泛,且在效率上更有優(yōu)勢。。

1、定義接口和實現

package com.meituan.hyt.test3.service; 
 
 
public interface UserService { 
  public String getName(int id); 
 
  public Integer getAge(int id); 
} 

package com.meituan.hyt.test3.service.impl; 
 
import com.meituan.hyt.test3.service.UserService; 
 
 
public class UserServiceImpl implements UserService { 
  @Override 
  public String getName(int id) { 
    System.out.println("------getName------"); 
    return "Tom"; 
  } 
 
  @Override 
  public Integer getAge(int id) { 
    System.out.println("------getAge------"); 
    return 10; 
  } 
} 

2、jdk動態(tài)代理實現

package com.meituan.hyt.test3.jdk; 
 
import java.lang.reflect.InvocationHandler; 
import java.lang.reflect.Method; 
 
 
public class MyInvocationHandler implements InvocationHandler { 
  private Object target; 
 
  MyInvocationHandler() { 
    super(); 
  } 
 
  MyInvocationHandler(Object target) { 
    super(); 
    this.target = target; 
  } 
 
  @Override 
  public Object invoke(Object o, Method method, Object[] args) throws Throwable { 
    if("getName".equals(method.getName())){ 
      System.out.println("++++++before " + method.getName() + "++++++"); 
      Object result = method.invoke(target, args); 
      System.out.println("++++++after " + method.getName() + "++++++"); 
      return result; 
    }else{ 
      Object result = method.invoke(target, args); 
      return result; 
    } 
 
  } 
} 
package com.meituan.hyt.test3.jdk; 
 
import com.meituan.hyt.test3.service.UserService; 
import com.meituan.hyt.test3.service.impl.UserServiceImpl; 
 
import java.lang.reflect.InvocationHandler; 
import java.lang.reflect.Proxy; 
 
 
public class Main1 { 
  public static void main(String[] args) { 
    UserService userService = new UserServiceImpl(); 
    InvocationHandler invocationHandler = new MyInvocationHandler(userService); 
    UserService userServiceProxy = (UserService)Proxy.newProxyInstance(userService.getClass().getClassLoader(), 
        userService.getClass().getInterfaces(), invocationHandler); 
    System.out.println(userServiceProxy.getName(1)); 
    System.out.println(userServiceProxy.getAge(1)); 
  } 
} 

運行結果

++++++before getName++++++
------getName------
++++++after getName++++++
Tom
------getAge------
10

3、cglib動態(tài)代理實現

Cglib是一個優(yōu)秀的動態(tài)代理框架,它的底層使用ASM在內存中動態(tài)的生成被代理類的子類,使用CGLIB即使代理類沒有實現任何接口也可以實現動態(tài)代理功能。CGLIB具有簡單易用,它的運行速度要遠遠快于JDK的Proxy動態(tài)代理:

CGLIB的核心類:

 net.sf.cglib.proxy.Enhancer – 主要的增強類
  net.sf.cglib.proxy.MethodInterceptor – 主要的方法攔截類,它是Callback接口的子接口,需要用戶實現
  net.sf.cglib.proxy.MethodProxy – JDK的java.lang.reflect.Method類的代理類,可以方便的實現對源對象方法的調用,如使用:
  Object o = methodProxy.invokeSuper(proxy, args);//雖然第一個參數是被代理對象,也不會出現死循環(huán)的問題。

net.sf.cglib.proxy.MethodInterceptor接口是最通用的回調(callback)類型,它經常被基于代理的AOP用來實現攔截(intercept)方法的調用。這個接口只定義了一個方法
public Object intercept(Object object, java.lang.reflect.Method method,
Object[] args, MethodProxy proxy) throws Throwable;

第一個參數是代理對像,第二和第三個參數分別是攔截的方法和方法的參數。原來的方法可能通過使用java.lang.reflect.Method對象的一般反射調用,或者使用 net.sf.cglib.proxy.MethodProxy對象調用。net.sf.cglib.proxy.MethodProxy通常被首選使用,因為它更快。

package com.meituan.hyt.test3.cglib; 
 
 
import net.sf.cglib.proxy.MethodInterceptor; 
import net.sf.cglib.proxy.MethodProxy; 
 
import java.lang.reflect.Method; 
 
 
public class CglibProxy implements MethodInterceptor { 
  @Override 
  public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { 
    System.out.println("++++++before " + methodProxy.getSuperName() + "++++++"); 
    System.out.println(method.getName()); 
    Object o1 = methodProxy.invokeSuper(o, args); 
    System.out.println("++++++before " + methodProxy.getSuperName() + "++++++"); 
    return o1; 
  } 
} 
package com.meituan.hyt.test3.cglib; 
 
import com.meituan.hyt.test3.service.UserService; 
import com.meituan.hyt.test3.service.impl.UserServiceImpl; 
import net.sf.cglib.proxy.Enhancer; 
 
 
 
public class Main2 { 
  public static void main(String[] args) { 
    CglibProxy cglibProxy = new CglibProxy(); 
 
    Enhancer enhancer = new Enhancer(); 
    enhancer.setSuperclass(UserServiceImpl.class); 
    enhancer.setCallback(cglibProxy); 
 
    UserService o = (UserService)enhancer.create(); 
    o.getName(1); 
    o.getAge(1); 
  } 
} 

運行結果:

++++++before CGLIB$getName$0++++++
getName
------getName------
++++++before CGLIB$getName$0++++++
++++++before CGLIB$getAge$1++++++
getAge
------getAge------
++++++before CGLIB$getAge$1++++++

關于利用java實現動態(tài)代理的方法就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。


本文題目:利用java實現動態(tài)代理的方法
文章鏈接:http://weahome.cn/article/peishc.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部