動态代理
1、動态代理
- 動态代理和靜态代理角色一樣
- 動态代理的代理類是動态生成的,不是我們直接寫好的
- 動态代理分為兩大類:基于接口的動态代理,基于類的動态代理
- 基于接口---JDK動态代理
- 基于類:cglib
- java位元組碼實作:JAVAssist
我們這裡使用JDK的原生代碼來實作,其餘的道理都是一樣的!
JDK的動态代理需要了解兩個類:
核心:InvocationHandler 和 Proxy,打開JDK幫助文檔看看
【InvocationHandler:調用處理程式】
- 每個代理執行個體都有一個關聯的調用處理程式。 當在代理執行個體上調用方法時,方法調用将被編碼并分派到其調用處理程式的
方法。invoke
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable;
// 參數解釋:
// proxy -- 代理角色
// method -- 代理角色調用的方法
// args -- 代理角色調用的方法傳遞的參數
【Proxy : 代理】
-
提供了建立動态代理類和執行個體的靜态方法,它也是由這些方法建立的所有動态代理類的超類。Proxy
- 為某個接口建立代理執行個體
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
throws IllegalArgumentException
// 參數解釋:
// loader -- 類加載器
// interfaces -- 代理類實作的接口清單
// h -- 調用處理程式
代碼實作
抽象角色和真實角色和之前的一樣!
1、Rent . java 即抽象角色
package com.edgar.demo03;
// 抽象角色:租房
public interface Rent {
void rent();
}
2、LangLord. java 即真實角色
package com.edgar.demo03;
//真實角色: 房東,房東要出租房子
public class LangLord implements Rent {
@Override
public void rent() {
System.out.println("房東要出租房子!");
}
}
3、ProxyInvocationHandler. java 即代理角色
package com.edgar.demo03;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
// 等會我們會用這個類,自動生成代理類
public class ProxyInvocationHandler implements InvocationHandler {
// 被代理的接口
private Rent rent;
public void setRent(Rent rent) {
this.rent = rent;
}
// 生成得到代理類
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
this.rent.getClass().getInterfaces(),this);
}
// 處理代理執行個體,并傳回結果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//動态代理的本質,就是使用反射機制實作!
this.seeHouse();
Object result = method.invoke(this.rent, args);
this.fare();
return result;
}
public void seeHouse(){
System.out.println("中介帶看房子");
}
public void fare(){
System.out.println("收中介費");
}
}
4、Client . java
package com.edgar.demo03;
public class Client {
public static void main(String[] args) {
// 真實角色
LangLord langLord = new LangLord();
// 代理執行個體的調用程式
ProxyInvocationHandler pih = new ProxyInvocationHandler();
// 将真實角色放置進去
pih.setRent(langLord);
// 動态生成對應的代理類!
Rent proxy = (Rent) pih.getProxy();
proxy.rent();
}
核心:一個動态代理 , 一般代理某一類業務 , 一個動态代理可以代理多個實作類,代理的是接口!
2、動态代理再了解
我們來使用動态代理實作代理我們之前寫的UserService!
我們也可以編寫一個通用的動态代理實作的類!所有的代理對象設定為Object即可!
package com.edgar.demo03;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
// 等會我們會用這個類,自動生成代理類
public class ProxyInvocationHandler implements InvocationHandler {
// 被代理的接口
private Object target;
public void setTarget(Object target) {
this.target = target;
}
// 生成得到代理類
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
this.target.getClass().getInterfaces(),this);
}
// 處理代理執行個體,并傳回結果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//動态代理的本質,就是使用反射機制實作!
this.log(method.getName());
Object result = method.invoke(this.target, args);
return result;
}
public void log(String methodName){
System.out.println("執行了"+methodName+"方法");
}
}
測試!
package com.edgar.demo03;
import com.edgar.demo02.UserService;
import com.edgar.demo02.UserServiceImpl;
public class Client {
public static void main(String[] args) {
// 真實角色
UserService userService = new UserServiceImpl();
// 代理執行個體的調用程式
ProxyInvocationHandler pih = new ProxyInvocationHandler();
// 将真實角色放置進去
pih.setTarget(userService);
// 動态生成代理類
UserService proxy = (UserService) pih.getProxy();
proxy.delete();
}
}
測試,增删改查,檢視結果!
動态代理的好處
靜态代理有的它都有,靜态代理沒有的,它也有!
- 可以使得我們的真實角色更加純粹 . 不再去關注一些公共的事情 .
- 公共的業務由代理來完成 . 實作了業務的分工 ,
- 公共業務發生擴充時變得更加集中和友善 .
- 一個動态代理 , 一般代理某一類業務
- 一個動态代理可以代理多個實作類,代理的是接口!