靜態代理、JDK動態代理及Cglib動態代理各自存在的缺點及代碼示例

1.靜態代理:

  • 基本描述:靜態代理代理的是某一個接口的實例對象,以接口的形式對外展示。當然如果不需要以接口的形式對外展示,直接代理對象即可。
  • 缺點:只能代理某一種類型,要想代理其他類型,需要修改代碼。
package unittest.proxy;
interface StaticInterface{
	void method();
}
class Target implements StaticInterface{
	@Override
	public void method() {
		System.out.println("StaticInterface");
	}
}
/**
 * @author Dylaniou
 * 基本描述:靜態代理代理的是某一個接口的實例對象,以接口的形式對外展示。當然如果不需要以接口的形式對外展示,直接代理對象即可。
 * 缺點:只能代理某一種類型,要想代理其他類型,需要修改代碼。
 */
public class JdkStaticProxy implements StaticInterface{
	Target target;
	JdkStaticProxy(Target target){
		this.target = target;
	}
	@Override
	public void method() {
		System.out.println("before");
		target.method();
		System.out.println("after");
	}
	public static void main(String[] args) {
		StaticInterface staticInterface = (StaticInterface)(new JdkStaticProxy(new Target()));
		staticInterface.method();
	}
}

2.JDK動態代理

  • 基本描述:jdk動態代理可以代理的接口的實例對象,以接口的形式對外展示。不侷限於某個類型。
  • 缺點:必須要有接口。
package unittest.proxy;

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

interface DynamicInterface{
	void method();
}
class DynamicProxyInterfaceImp implements DynamicInterface{
	@Override
	public void method() {
		System.out.println("DynamicProxyInterfaceImp");
	}
}
/**
 * @author Dylaniou
 * 基本描述:jdk動態代理可以代理的接口的實例對象,以接口的形式對外展示。不侷限於某個類型。
 * 缺點:必須要有接口。
 */
public class JdkDynamicProxy implements InvocationHandler{
	Object target;
	JdkDynamicProxy(Object target){
		this.target = target;
	}
	
	public Object getProxyOfTarget(){
		return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
	}
	
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("before");
		method.invoke(target, args);
		System.out.println("after");
		return null;
	}
	public static void main(String[] args) {
		DynamicProxyInterfaceImp target = new DynamicProxyInterfaceImp();
		DynamicInterface dynamicInterface = (DynamicInterface)(new JdkDynamicProxy(target)).getProxyOfTarget();
		dynamicInterface.method();
	}
}

3.Cglib動態代理

  • 基本描述:Cglib動態代理可以直接代理實例對象,不是必須要接口。當然想以接口的形式對外展示也是完全OK的。
  • 缺點:Cglib動態代理需要依賴第三方jar包,比如我本地用的就是mybatis-3.5.4裏的cglib-3.3.0.jar和asm-7.1.jar。
package unittest.proxy;

import java.lang.reflect.Method;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

class CglibInstance{
	void method(){
		System.out.println("CglibInstance");
	};
}
/**
 * @author Dylaniou
 * 基本描述:Cglib動態代理可以直接代理實例對象,不是必須要接口。當然想以接口的形式對外展示也是完全OK的。
 */
//public class CglibProxy{
public class CglibProxy implements MethodInterceptor{
	public Object getProxyOfTarget(Class cls){
		Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(cls);
		enhancer.setCallback(this);
		return enhancer.create();
	}
	@Override
	public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
		System.out.println("before");
		 methodProxy.invokeSuper(proxy, args);
		System.out.println("after");
		return null;
	}
	public static void main(String[] args) {
		CglibInstance cglibInterface  = (CglibInstance)(new CglibProxy().getProxyOfTarget(CglibInstance.class));
		cglibInterface.method();
		/*Enhancer enhancer = new Enhancer();
		enhancer.setSuperclass(CglibInstance.class);
		enhancer.setCallback(new MethodInterceptor(){
			public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
				System.out.println("before");
				Object result = methodProxy.invokeSuper(proxy, args);
				System.out.println("after");
				return result;
			}
		});
		((CglibInstance) enhancer.create()).method();*/
	}
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章