動態代理的兩種實現方式
那麼動態代理是如何實現將切面邏輯(advise)織入到目標類方法中去的呢?下面我們就來詳細介紹並實現AOP中用到的兩種動態代理。
AOP的原始碼中用到了兩種動態代理來實現攔截切入功能:jdk動態代理和cglib動態代理。兩種方法同時存在,各有優劣。jdk動態代理是由Java內部的反射機制來實現的,cglib動態代理底層則是藉助asm來實現的。總的來說,反射機制在生成類的過程中比較高效,而asm在生成類之後的相關執行過程中比較高效(可以通過將asm生成的類進行快取,這樣解決asm生成類過程低效問題)。還有一點必須注意:jdk動態代理的應用前提,必須是目標類基於統一的介面。如果沒有上述前提,jdk動態代理不能應用。
1、定義介面和實現
[java] view plain copy print?- package com.meituan.hyt.test3.service;
- publicinterface UserService {
- public String getName(int id);
- public Integer getAge(int id);
- }
-
package
- import com.meituan.hyt.test3.service.UserService;
- publicclass UserServiceImpl implements UserService {
- @Override
- public String getName(int id) {
- System.out.println("------getName------");
- return"Tom";
- }
- @Override
-
public
- System.out.println("------getAge------");
- return10;
- }
- }
2、jdk動態代理實現
[java] view plain copy print?- package com.meituan.hyt.test3.jdk;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- publicclass 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;
- }
- }
- }
[java] view plain copy print?
- 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;
- publicclass Main1 {
- publicstaticvoid 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動態代理實現
Cglib是一個優秀的動態代理框架,它的底層使用ASM在記憶體中動態的生成被代理類的子類,使用CGLIB即使代理類沒有實現任何介面也可以實現動態代理功能。CGLIB具有簡單易用,它的執行速度要遠遠快於JDK的Proxy動態代理:
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);//雖然第一個引數是被代理物件,也不會出現死迴圈的問題。
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;
- publicclass 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;
- }
- }
[java] view plain copy print?
- 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;
- publicclass Main2 {
- publicstaticvoid 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++++++