★学习目标:
1、熟悉Spring中两种动态代理方式的区别
2、掌握JDK动态代理
★思考任务:
1、什么是JDK动态代理?
★任务学习:
★知识要点:
JDK动态代理是通过java.lang.reflect.Proxy类来实现的,我们可以调用Proxy类的newProxyInstance()方法来创建代理对象。对于使用业务接口的类,Spring默认会使用JDK动态代理来实现AOP。
接下来,通过一个案例来演示Spring 中JDK 动态代理的实现过程,具体步骤如下。
( 1 )创建一个名为chapter03 的Web 项目,导入Spring 框架所需JAR 包到项目的lib 目录中,并发布到类路径。
(2 )在src 目录下,创建一个com.itheima.jdk 包,在该包下创建接口UserDao ,并在该接口中编写添加和删除的方法。
package com.itheima.jdk;
public interface UserDao {
public void addUser();
public void deleteUser();
}
(3 )在com . itheima . jdk 包中,创建UserDao 接口的实现类UserDaolmpl ,分别实现接口中的方法,并在每个方法中添加一条输出语句。
// 目标类
@Repository("userDao")
public class UserDaoImpl implements UserDao {
public void addUser() {
//int i = 10/0;
System.out.println("添加用户");
}
public void deleteUser() {
System.out.println("删除用户");
}
}
(4 )在src 目录下,创建一个com. itheima.aspect 包,并在该包下创建切面类MyAspect ,在该类中定义一个模拟权限检查的方法和一个模拟记录日志的方法,这两个方法就表示切面中的通知。
//切面类:可以存在多个通知Advice(即增强的方法)
public class MyAspect {
public void check_Permissions(){
System.out.println("模拟检查权限...");
}
public void log(){
System.out.println("模拟记录日志...");
}
}
(5 )在com.itheima.jdk 包下,创建代理类JdkProxy ,该类需要实现InvocationHandler 接口,并编写代理方法。在代理方法中,需要通过Proxy 类实现动态代理,
/**
* JDK代理类
*/
public class JdkProxy implements InvocationHandler{
// 声明目标类接口
private UserDao userDao;
// 创建代理方法
public Object createProxy(UserDao userDao) {
this.userDao = userDao;
// 1.类加载器
ClassLoader classLoader = JdkProxy.class.getClassLoader();
// 2.被代理对象实现的所有接口
Class[] clazz = userDao.getClass().getInterfaces();
// 3.使用代理类,进行增强,返回的是代理后的对象
return Proxy.newProxyInstance(classLoader,clazz,this);
}
/*
* 所有动态代理类的方法调用,都会交由invoke()方法去处理
* proxy 被代理后的对象
* method 将要被执行的方法信息(反射)
* args 执行方法时需要的参数
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
// 声明切面
MyAspect myAspect = new MyAspect();
// 前增强
myAspect.check_Permissions();
// 在目标类上调用方法,并传入参数
Object obj = method.invoke(userDao, args);
// 后增强
myAspect.log();
return obj;
}
}
JdkProxy 类实现了InvocationHandler 接口,并实现了接口中的invoke()方法,所有动态代理类所调用的方法都会交由该方法处理。在创建的代理方法createProxy()中,使用了Proxy 类的newProxylnstance()方法来创建代理对象。newProxylnstance()方法中包含3个参数,其中第1 个参数是当前类的类加载器,第2 个参数表示的是被代理对象实现的所有接口,第3 个参数this 代表的就是代理类JdkProxy 本身。在invoke()方法中,目标类方法执行的前后,会分别执行切面类中的check_Permissions()方法和log()方法。
(6 )在com.itheima.jdk 包中,创建测试类JdkTest。在该类中的main()方法中创建代理对象和目标对象,然后从代理对象中获得对目标对象userDao 增强后的对象,最后调用该对象中的添加和删除方法。
public class JdkTest{
public static void main(String[] args) {
// 创建代理对象
JdkProxy jdkProxy = new JdkProxy();
// 创建目标对象
UserDao userDao= new UserDaoImpl();
// 从代理对象中获取增强后的目标对象
UserDao userDao1 = (UserDao) jdkProxy.createProxy(userDao);
// 执行方法
userDao1.addUser();
userDao1.deleteUser();
}
}
执行程序后,控制台的输出结果

userDao 实例中的添加用户和删除用户的方法已被成功调用,并且在调用前后分别增加了检查权限和记录日志的功能。这种实现了接口的代理方式,就是Spring 中的JDK 动态代理。

