一呢蔫、簡介
代理模式是23種設(shè)計(jì)模式的一種,他是指一個(gè)對(duì)象A通過持有另一個(gè)對(duì)象B鹏氧,可以具有B同樣的行為的模式渤涌。為了對(duì)外開放協(xié)議,B往往實(shí)現(xiàn)了一個(gè)接口把还,A也會(huì)去實(shí)現(xiàn)接口实蓬。但是B是“真正”實(shí)現(xiàn)類,A則比較“虛”吊履,他借用了B的方法去實(shí)現(xiàn)接口的方法安皱。A雖然是“偽軍”,但它可以增強(qiáng)B艇炎,在調(diào)用B的方法前后都做些其他的事情酌伊。Spring AOP就是使用了動(dòng)態(tài)代理完成了代碼的動(dòng)態(tài)“織入”。
二缀踪、靜態(tài)代理和動(dòng)態(tài)代理
靜態(tài)代理:類A寫死持有B居砖,就是B的靜態(tài)代理;
動(dòng)態(tài)代理:如果A代理的對(duì)象是不確定的驴娃,就是動(dòng)態(tài)代理奏候。目前動(dòng)態(tài)代理分:JDK動(dòng)態(tài)代理,CGLIB動(dòng)態(tài)代理唇敞。
三蔗草、JDK動(dòng)態(tài)代理
- 3.1 簡介
jdk動(dòng)態(tài)代理是jre提供給我們的類庫,可以直接使用厚棵。不需要第三方提供 - 3.2 代碼實(shí)現(xiàn)
- 3.2.1 定義接口并實(shí)現(xiàn)接口
/**
* 定義接口
*
* @author <a href="jian.huang@bintools.cn">yunzhe</a>
* @version 1.0.0 2019-07-08-下午8:15
*/
public interface IUserManager {
void addUser(String id,String password);
}
import com.bintools.inteface.IUserManager;
/**
* 接口實(shí)現(xiàn)
*
* @author <a href="jian.huang@bintools.cn">yunzhe</a>
* @version 1.0.0 2019-07-08-下午8:16
*/
public class IUserManagerImpl implements IUserManager {
@Override
public void addUser(String id, String password) {
System.out.println("=========添加用戶信息成功。蔼紧。婆硬。。==========");
}
}
- 3.2.2 實(shí)現(xiàn)jdk代理類奸例,主要是要實(shí)現(xiàn)InvocatinHandler類
mport java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* jdk代理模式
*
* @author <a href="jian.huang@bintools.cn">yunzhe</a>
* @version 1.0.0 2019-07-08-下午8:17
*/
public class JDKProxy implements InvocationHandler {
/** 需要代理的目標(biāo)對(duì)象**/
private Object targetObject;
/***
* 將目標(biāo)對(duì)象傳入進(jìn)行代理
* @param targetObject
* @return
*/
public Object newProxy(Object targetObject){
this.targetObject = targetObject;
//返回代理對(duì)象
return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),targetObject.getClass().getInterfaces(),this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//此部分進(jìn)行一定的邏輯處理
checkPopedom();
//設(shè)置返回方法的返回值
Object ret =null;
ret = method.invoke(targetObject,args);
return ret;
}
private void checkPopedom() {
System.out.println("========邏輯處理已完成=======");
}
}
- 3.2.3 定義客戶端
import com.bintools.inteface.IUserManager;
import com.bintools.inteface.impl.IUserManagerImpl;
import com.bintools.jdk.JDKProxy;
/**
* 客戶端
*
* @author <a href="jian.huang@bintools.cn">yunzhe</a>
* @version 1.0.0 2019-07-08-下午8:22
*/
public class Client {
public static void main(String[] args) {
System.out.println("**************JDKPROXY*************");
JDKProxy jdkProxy = new JDKProxy();
IUserManager iUserManager = (IUserManager) jdkProxy.newProxy(new IUserManagerImpl());
iUserManager.addUser("100","one234#$%");
}
}
-
3.2.4 執(zhí)行結(jié)果信息
jdk動(dòng)態(tài)代理.jpg
從上可知:JDK代理是不需要第三方支持的彬犯,只需要JDK環(huán)境姐可以進(jìn)行代理。使用條件:
1) 實(shí)現(xiàn)InvocationHandler
2 ) 使用Proxy.newProxyInstance產(chǎn)生代理對(duì)象
3)被代理的對(duì)象必須要實(shí)現(xiàn)接口 3.2.5 CGLIB動(dòng)態(tài)代理
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
/**
* TODO
*
* @author <a href="jian.huang@bintools.cn">yunzhe</a>
* @version 1.0.0 2019-07-08-下午8:36
*/
public class CglibProxy implements MethodInterceptor {
/**CGLIB 需要代理的對(duì)象**/
private Object targetObject;
/**創(chuàng)建代理對(duì)象**/
public Object createProxyObject(Object targetObject){
this.targetObject = targetObject;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(targetObject.getClass());
enhancer.setCallback(this);
Object proxyO = enhancer.create();
return proxyO;
}
@Override
public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
Object obj = null;
/**檢測(cè)當(dāng)前方法是否是addUser**/
if("addUser".equals(method.getName())){
checkPepedom();
}
obj = method.invoke(targetObject,args);
return obj;
}
/**可以做檢測(cè)或其他處理**/
private void checkPepedom() {
System.out.println("*******代理方法為addUser,進(jìn)行業(yè)務(wù)處理*******");
}
}
結(jié)果展示:
從上可知:CGlib必須依賴與CGLIB的類庫查吊,待它需要類來實(shí)現(xiàn)任何接口代理的指定類生成一個(gè)子類谐区,覆蓋其中方法,是一種繼承但是針對(duì)接口編程的環(huán)境下推進(jìn)使用JDK代理逻卖。
四宋列、JDK與CGLIB動(dòng)態(tài)代理
- 4.1 JDK動(dòng)態(tài)代理
利用攔截器(攔截器必須實(shí)現(xiàn)InvocationHanlder)加上反射機(jī)制生成一個(gè)實(shí)現(xiàn)代理的匿名類 - 4.2 CGLIB動(dòng)態(tài)代理
利用第三方類庫,對(duì)代理對(duì)象類的class文件加載進(jìn)來评也,通過修改字節(jié)嗎生成子類來處理 - 4.3 JDK動(dòng)態(tài)代理與CGLIB字節(jié)碼生成的區(qū)別
1)JDK動(dòng)態(tài)代理只能對(duì)實(shí)現(xiàn)接口的類生成代理炼杖,而不能針對(duì)類
2)CGLIB是針對(duì)類實(shí)現(xiàn)代理灭返,主要是指定的類生成子類,覆蓋其中的方法坤邪。并覆蓋其中方法實(shí)現(xiàn)增強(qiáng)熙含,但是因?yàn)椴捎美^承,所以類不要用final修飾