最近在看spring的aop,由于aop底层原理使用动态代理实现的,所以对动态代理做一番研究。
动态代理分为两种,jdk动态代理和CGlib动态代理
1、jdk动态代理
jdk动态代理的一个要求是必须只能为接口创建代理实例。接下来来看代码:
- 定义一个普通接口
package com.wisdom.jdkproxy;
/*****************************************************************
*
* 定义一个普通接口
*
* @author wisdom
* @date 2018年3月14日下午2:00:39
****************************************************************/
public interface User {
/**
* 定义方法吃
*/
public void eat();
/**
* 定义方法玩
*/
public void play();
}
- 接口的实现类
package com.wisdom.jdkproxy;
/*****************************************************************
*
* 用户实现类
*
* @author wisdom
* @date 2018年3月14日下午2:01:17
****************************************************************/
public class UserImpl implements User {
@Override
public void eat() {
System.out.println("吃鸡");
}
@Override
public void play() {
System.out.println("玩游戏");
}
}
- 获取代理对象
package com.wisdom.jdkproxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/*****************************************************************
*
* 获取代理对象,对eat方法进行增强
*
* @author wisdom
* @date 2018年3月14日下午2:01:33
****************************************************************/
public class UserProxy {
public static User getProxy(User user) {
User proxy = (User) Proxy.newProxyInstance(user.getClass().getClassLoader(), user.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if("eat".equals(method.getName())) {
System.out.println("记录日志");
}
return method.invoke(user, args);
}
});
return proxy;
}
}
- 测试
package com.wisdom.jdkproxy;
/*****************************************************************
*
* 测试类
*
* @author wisdom
* @date 2018年3月14日下午2:01:48
****************************************************************/
public class UserTest {
public static void main(String[] args) {
User user = new UserImpl();
User proxy = UserProxy.getProxy(user);
proxy.eat();
// proxy.play();
}
}
- 输出结果
2、CGlib动态代理
CGlib动态代理解决了jdk动态代理只能为接口创建代理实例的缺陷。
- 定义一个普通类
package com.wisdom.cglibproxy;
/*****************************************************************
*
* 定义一个普通类
*
* @author wisdom
* @date 2018年3月14日下午1:59:36
****************************************************************/
public class Person {
/**
* 定义eat方法
*/
public void eat() {
System.out.println("吃鸡");
}
/**
* 定义play方法
*/
public void play() {
System.out.println("玩游戏");
}
}
- 创建代理对象
package com.wisdom.cglibproxy;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
/*****************************************************************
*
* cglib动态代理
*
* @author wisdom
* @date 2018年3月14日下午1:59:01
****************************************************************/
public class PersonProxy {
public static Person getProxy() {
Enhancer enhancer = new Enhancer();
// 设置父类
enhancer.setSuperclass(Person.class);
// 设置回调函数
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
if ("eat".equals(method.getName())) {
System.out.println("记录日志");
}
return proxy.invokeSuper(obj, args);
}
});
// 生成代理对象
Person proxy = (Person) enhancer.create();
return proxy;
}
}
- 测试类
package com.wisdom.cglibproxy;
/**
* 测试类
*
* @author wisdom
* @date 2018年1月11日
*/
/*****************************************************************
*
* 测试类
*
* @author w15309
* @date 2018年3月14日下午1:59:59
****************************************************************/
public class PersonTest {
public static void main(String[] args) {
Person proxy = PersonProxy.getProxy();
proxy.eat();
// proxy.play();
}
}
- 输出结果:
还在学习中·····