一 ,什么是代理?
代理 : 本来是自己应该做的事, 却请了别人来做, 被请的人就是代理对象
举例 : 春节回家买票找人代买, 黄牛就是代理对象
二 ,什么是动态代理?
代理的对象是变动的, 在程序运行过程中产生的 . 而在程序运行过程中产生对象, 这个对象是不固定的, 那么可以通过反射来实现, 所以动态代理是基于反射实现的.
三 ,动态代理常见有JDK提供的动态代理,和Cglib提供的代理 , 这次介绍JDK提供的代理
1, 在Java中java.lang.reflect包下提供了一个Proxy类和InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象. JDK提供的代理只能针对接口做代理. Cglib提供的代理会更强大一些.
2, Proxy类中的方法创建动态代理类对象,该方法会返回代理类
Public static Object newProxyInstance(ClassLoader loader, Class<?>[] intergaces,InvocationHandler h )
最终会调用InvocationHandler的方法
3, InvocationHandler接口 , 定义动态代理要实现的具体业务逻辑
Object invoke(Objectproxy, Method method , Object[] args );
自定义类实现IncovationHandler接口, 复写invoke()方法, 具体代理的逻辑写在该方法中. invoke()方法执行后的返回值为代理对象 .
代码举例:
1,创建要代理的接口和实现类
public interface UserDao {
void add();
void delete();
}
public class UserDaoImp implements UserDao{
public void add() {
System.out.println("执行了添加的方法");
}
public void delete() {
System.out.println("执行了删除的方法");
}
}
public interface WorkDao {
void work();
void study();
}
public class WorkDaoImp implements WorkDao {
public void work() {
System.out.println("工作");
}
public void study() {
System.out.println("学习");
}
}
2,创建InvocationHandler接口的实现类,具体代理的业务逻辑在Invoke方法
/**
* 自定义类实现IncovationHandler接口, 复写invoke()方法
*/
public class MyInvocationHandler implements InvocationHandler {
private Object target; //目标对象
public MyInvocationHandler(Object target){
this.target = target;
}
/**
* 在代理实例上处理方法调用并返回结果
* proxy - 在其上调用方法的代理实例
* method - 对应于在代理实例上调用的接口方法的 Method 实例
* args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("权限校验");
//invoke 方法执行后的返回值为代理类对象
Object result = method.invoke(target, args);
System.out.println("日志分析");
return result; //返回代理类对象
}
}
3,测试类,Proxy类中的方法创建动态代理类对象,该方法会返回代理类
public class ProxyTest {
/**
* 使用JDK动态代理,在原来的方法执行前后执行授权,日志记录
* @param args
*/
public static void main(String[] args) {
UserDao userDao = new UserDaoImp();
//不使用代理的情况下调用方法
userDao.add();
userDao.delete();
//代理用户
MyInvocationHandler myInvocationHandler = new MyInvocationHandler(userDao);
//返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序
//返回对象只能是接口UserDao,不能是接口的实现类UserDaoImp,所以说JDK提供的代理只能针对接口做代理
UserDao newProxyInstance = (UserDao) Proxy.newProxyInstance(userDao.getClass().getClassLoader(),
userDao.getClass().getInterfaces(),myInvocationHandler);
newProxyInstance.add();
newProxyInstance.delete();
System.out.println("--------------------------------");
WorkDao workDao = new WorkDaoImp();
//不使用代理的情况下调用方法
workDao.work();
workDao.study();
//代理工人
MyInvocationHandler myInvocationHandler2 = new MyInvocationHandler(workDao);
WorkDao newProxyInstance2 = (WorkDao) Proxy.newProxyInstance(workDao.getClass().getClassLoader(),
workDao.getClass().getInterfaces(), myInvocationHandler2);
newProxyInstance2.work();
newProxyInstance2.study();
}
}
4, 控制台输出 , 在方法执行前后执行了授权和添加日志的方法,具体方法中业务逻辑可以根据实际情况定义
执行了添加的方法
执行了删除的方法
权限校验
执行了添加的方法
日志分析
权限校验
执行了删除的方法
日志分析
--------------------------------
工作
学习
权限校验
工作
日志分析
权限校验
学习
日志分析