Java 動態代理的多種實現方式

一、動態代理簡介

優勢:在不修改源碼的情況下,對目標方法進行相應的增強。

作用:完成程序功能之間的松耦合。

二、動態代理的多種實現

  • JDK代理:基於接口的動態代理技術(缺點,目標對象必須有接口,如果沒有接口,則無法完成動態代理的實現)
  • cglib代理:基於父類的動態代理技術

兩者的區別如圖所示:

1. 基於JDK的實現

目標接口類:

public interface TargetInterface {
	public void save();
	
	public void print(String str);
}

目標類:

public class Target implements TargetInterface{
	
	public void save() {
		System.out.println("save running...");
	}
	
	public void print(String str) {
		System.out.println(str);
	}

}

增強類:

public class Advice {
	public void before() {
		System.out.println("前置增強");
	}

	public void after() {
		System.out.println("後置增強");
	}
}

測試類:

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

public class ProxyTest {

	public static void main(String[] args) {
		
		//目標對象
		final Target target = new Target();
		
		//增強對象
		final Advice advice = new Advice();
		
		TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(
				target.getClass().getClassLoader(), 					//目標對象類加載器
				target.getClass().getInterfaces(), 						//目標對象相同的接口字節碼對象數組
				new InvocationHandler() {
					//調用代理對象的任何方法,實質執行的都是invoke方法
					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
						advice.before();								//前置增強
						Object invoke = method.invoke(target, args);	//執行目標方法
						advice.after();									//後置增強
						System.out.println();
						return invoke;
					}
				});
		
		//代理對象的方法測試
		proxyInstance.save();
		
		proxyInstance.print("JDK動態代理");
	}

}

運行截圖:

2. 基於cglib的實現

需要導入Jar包,如果是maven項目,則在pom.xml文件加入如下配置:

<dependency>
	<groupId>org.springframework</groupId>
	<artifactId>spring-context</artifactId>
	<version>4.2.4.RELEASE</version>
</dependency>

目標類:

public class Target {
	public void save() {
		System.out.println("save running...");
	}
	
	public void print(String str) {
		System.out.println(str);
	}
}

增強類:

public class Advice {
	
	public void before() {
		System.out.println("前置增強");
	}

	public void after() {
		System.out.println("後置增強");
	}

}

測試類:

import java.lang.reflect.Method;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

public class ProxyTest {

	public static void main(String[] args) {
		final Target target = new Target();
		final Advice advice = new Advice();
		
		//返回值就是動態生成的代理對象,基於cglib
		//創建增強器
		Enhancer enhancer = new Enhancer();
		
		//設置父類(目標)
		enhancer.setSuperclass(Target.class);
		
		//設置回調
		enhancer.setCallback(new MethodInterceptor() {
			public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{
				advice.before();
				Object invoke = method.invoke(target, obj);
				advice.after();
				System.out.println();
				return invoke;
			}
		});
		
		//創建代理對象
		Target proxy = (Target)enhancer.create();
		
		//測試代理方法
		proxy.save();
		proxy.print("基於cglib實現動態規劃");
		
	}

}

運行截圖:

三、為什麼要有基於cglib的實現

使用JDK動態代理實現時,最大限制是被增強對象必須實現接口,並且增強的方法隻能是接口中聲明的方法。但在實際的項目中,可能總是存在對不實現業務接口的對象進行增強的需求,這時JDK動態代理將無能為力。

四、兩種方式的適用場景

JDK動態代理

優點

  • 不依賴第三方jar包, 使用方便
  • 隨著JDK的升級,JDK動態代理的性能在穩步提升

缺點

  • 隻能代理實現瞭接口的類
  • 執行速度較慢

適用場景

  • 如果你的程序需要頻繁、反復地創建代理對象,則JDK動態代理在性能上更占優。

cglib

優點

由於是動態生成字節碼實現代理,因此代理對象的執行速度較快, 約為JDK動態代理的1.5 ~ 2倍
可以代理沒有實現接口的對象

缺點

  • 不能代理final類
  • 動態生成字節碼雖然執行較快,但是生成速度很慢,根據網上一些人的測試結果,cglib創建代理對象的速度要比JDK慢10 ~ 15倍。

適用場景

  • 不需要頻繁創建代理對象的應用,如Spring中默認的單例bean,隻需要在容器啟動時生成一次代理對象。

以上就是Java 動態代理的多種實現方式的詳細內容,更多關於Java 動態代理的實現的資料請關註WalkonNet其它相關文章!

推薦閱讀: