设计模式浅析之动态代理模式

近期在看Spring以及Mybatis的相关书籍,其中Spring的AOP(Aspect Oriented Programming)面向切面编程的实现使用了动态代理模式。本文简述个人对于动态代理模式中的动态代理技术中的两种的理解。

java中动态代理技术有JDK,CGLIB,Javassist,ASM。其中最常用的两种就是本文要说的JDK,CGLIB。

JDK动态代理:是JDK自带的功能,它必须借助接口才能产生动代理对象。

CGLIB(Code Generation Library):它是第三方提供的一个技术。CGLIB项目是一个开源的项目,扩展性非常强。

1.什么是动态代理模式。

动态代理的意义在于生成一个代理对象,来代理真实对象,从而控制外部对真实对象的访问。

举个简单的例子来看:代理对象就相当于公司的前台,真实对象就相当于公司的高管。有人来访一般会在前台登记,前台会判断来访者是否可以会见高管,然后前台会让来访者进行登记,访问高管,访问之后再在对来访记录进行登记。

可以用一张图来理解:

javassist 修改方法 javassist cglib_动态代理

外部的访问可以理解成来访者,前台可以比作代理对象,高管是真实对象。在来访者访问到高管之前,可能会有一些安来访登记的操作,以及访问之后来访登记的结束记录。如果利用一个前台代理高管的一个工作,高管只需要接受来访者的到访,同时还可以去除一些没有必要的访问。

代理模式中的前台作为代理对象,那么他需要和高管即真实对象之间,建立一个代理关系,不能说A公司的前台,给B公司高管处理来访的人员登记吧。这样就会出问题。所以代理必须满足下面的条件:

1、代理对象必须和真实对象之间建立代理关系。A公司的前台,给B公司高管处理来访的人员登记,需要在A公司离职,在B公司入职。

2、实现代理对象的代理逻辑方法。前台咨询来访者是不是有预约、以及来访登记等等。

使用的场景:

javassist 修改方法 javassist cglib_JAVA_02

2、JDK动态代理

JDK动态代理是java.lang.reflect.*包中提供的方式,它必须借助一个接口才能产生代理对象。

//代理对象需要借助的接口
public interface HelloWorld {
	public void sayHelloWorld();
}

//真实对象,实现接口
public class HelloWorldImpl implements HelloWorld {

	@Override
	public void sayHelloWorld() {
		System.out.println("Hello World");
	}

}

按照代理必须满足的条件,要先建立起来代理对象和真实对象之间的代理关系,然后实现代理逻辑。

JDK动态代理中,要实现代理逻辑类必须实现java.lang.reflect.InvocationHandler接口,它定义了一个invoke方法,并提供接口数组用于下挂代理对象。

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

public class JdkProxyExample implements InvocationHandler {

	// 真实对象
	private Object target = null;

	/**
	 * 建立代理对象和真实对象的代理关系,并返回代理对象
	 * @param target真实对象
	 * @return 代理对象
	 */
	public Object bind(Object target) {
		this.target = target;
		return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
	}

	/**
	 * 代理方法逻辑
	 * 
	 * @param proxy --代理对象
	 * @param method --当前调度方法
	 * @param args --当前方法参数
	 * @return 代理结果返回
	 * @throws Throwable 异常
	 */
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("进入代理逻辑方法");
		System.out.println("在调度真实对象之前的服务");
		Object obj = method.invoke(target, args);// 相当于调用sayHelloWorld方法
		System.out.println("在调度真实对象之后的服务");
		return obj;
	}
}

❤建立代理对象和真实对象的关系。使用bind方法完成,方法里首先使用target类保存真实对象,然后通过代码生成代理对象。

newProxyInstance方法包含的3个参数:

  • 第一个是类加载器,采用了target本身的类加载器。
  • 第二个是把生成的动态代理对象下挂在哪些接口下,这个写法就是放在target实现的接口HelloWorld下。
  • 第三个是定义实现方法逻辑的代理类,this鞭尸当前对象,它必须实现InvocationHandler接口的invoke方法,它就是代理逻辑方法的实现方法。

❤实现代理逻辑的方法。invoke方法可以实现代理逻辑。

invoke方法包含的3个参数:

  • proxy,代理对象,就是bind方法生成的对象。就是公司的前台。
  • method,当前调度的方法。(如果不知道的话,可以去看下java的反射,后续可以下一篇问,说一说。)
  • args,调度方法的参数。

测试一下:

public static void testJdkProxy() {
		JdkProxyExample jdk = new JdkProxyExample();
		// 绑定关系,因为挂在接口HelloWorld下,所以声明代理对象HelloWorld proxy
		HelloWorld proxy = (HelloWorld) jdk.bind(new HelloWorldImpl());
		// 注意,此时HelloWorld对象已经是一个代理对象,它会进入代理的逻辑方法invoke里
		proxy.sayHelloWorld();
	}

测试运行结果:

javassist 修改方法 javassist cglib_javassist 修改方法_03

 

3、CGLIB动态代理

如果一个类没有实现的接口那么就不能,那么用JDK动态代理技术是没有办法实现代理的。如果说高管没有一个稳定的办公场所,就比方说这个公司是一个初创公司比较小,没有资金花再办公场所的建设上。

这时候只能采用第三方技术,CGLIB动态代理。它的优势在于不需要提供接口,是要一个非抽象类就可以实现动态代理。

看这个一个非抽象类:

public class ReflectServiceImpl {
	public void sayHello(String name) {
		System.err.println("Hello " + name);
	}

}

使用CGLIB动态代理代码如下:

import java.lang.reflect.Method;

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

public class CglibProxyExample implements MethodInterceptor {
	/**
	 * 生成CGLIB代理对象
	 * @param cls -- Class类
	 * @return Class类的CGLIB代理对象
	 */
	public Object getProxy(Class cls) {
		// CGLIB enhancer增强类对象
		Enhancer enhancer = new Enhancer();
		// 设置增强类型
		enhancer.setSuperclass(cls);
		// 定义代理逻辑对象为当前对象,要求当前对象实现MethodInterceptor方法
		enhancer.setCallback(this);
		// 生成并返回代理对象
		return enhancer.create();
	}

	/**
	 * 代理逻辑方法
	 * @param proxy 代理对象
	 * @param method 方法
	 * @param args 方法参数
	 * @param methodProxy 方法代理
	 * @return 代理逻辑返回
	 * @throws Throwable异常
	 */
	@Override
	public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
		System.err.println("调用真实对象前");
		// CGLIB反射调用真实对象方法
		Object result = methodProxy.invokeSuper(proxy, args);
		System.err.println("调用真实对象后");
		return result;
	}
}

这里用了CGLIB的Enhance,通过设置超类的方法(setSuperclass),然后通过setCallback方法设置为哪个类做代理。

这里的两个必要条件:

1、getProxy方法,建立代理对象和真实对象之间的关系

2、intercept方法,实现代理逻辑。

测试一下:

public static void tesCGLIBProxy() {
	    CglibProxyExample cpe = new CglibProxyExample();
	    ReflectServiceImpl obj = (ReflectServiceImpl)cpe.getProxy(ReflectServiceImpl.class);
	    obj.sayHello("张三");
	}

测试结果:

javassist 修改方法 javassist cglib_JAVA_04

动态代理模式,我个人理解也就这个多。其中说的来访者、前台还有高管,纯属个人理解,如果觉得不合适的话,欢迎留言指正。

本文是看书《Java EE互联网轻量级框架整合开发》,个人感觉可以一起分享出来。