一、概念
一个类代表另一类的功能。功能主要是起到增强方法和权限拦截的作用。
二、分类
1、静态代理
2、动态代理
三、静态代理
场景:学生需要交班费购买桶装水,班费由班长代为收取。
1个公共接口
public interface PersonDao {
/**
* 交班费
*/
public void giveMoney();
}
1个实现类
public class StudentImpl implements PersonDao {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public StudentImpl(String name) {
this.name = name;
}
@Override
public void giveMoney() {
System.out.println( this.name + "上交班费");
}
}
1个代理类
public class MonitorImpl implements PersonDao {
/**
* 为其他对象提供一种代理以控制对这个对象的访问。
* 在某些情况下,一个对象不适合或者不能直接引用另一个对象,
* 而代理对象可以在客户端和目标对象之间起到中介的作用。
*
* 代理模式 适配器模式
* 原类和代理类继承同一父类 只有适配继承目标接口
* 原类对象与代理对象接口相同 原类要通过适配来连接父类
* 起到了隐藏原类的作用
*/
private StudentImpl student;
public MonitorImpl(PersonDao student) {
this.student = (StudentImpl) student;
}
@Override
public void giveMoney() {
student.giveMoney();
}
}
客户端
public class ProxyDemo {
public static void main(String[] args) {
/**
* 代理类:1个公共接口 + 1个具体类 + 1个代理类
* 静态代理类
*/
PersonDao student = new StudentImpl("张三");
MonitorImpl monitor = new MonitorImpl(student);
System.out.println("静态代理类-------->");
monitor.giveMoney();
}
静态代理模式的缺点:
1️⃣MonitorImpl需要与StudentImpl实现一样的PersonDao的接口。
2️⃣当StudentImpl增加方法,MonitorImpl对应要做出修改。
四、动态代理
动态代理和静态代理的区别,动态代理不需要我们手动编写代理类,在运行时,动态在内存中生产代理类。
1、动态代理的API
基于jdk
java.lang.reflect.Proxy
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
返回值Object就是代理对象
参数:
loader:代表与目标对象相同的类加载器,目标对象.getClass.getClassLoder();
interfaces:代表与目标对象实现的所有的接口字节码对象数组,数组因为目标类可以有多个接口;
h:具体的代理的操作,InvocationHandler接口
:JDK的Proxy方式实现的动态代理 目标对象必须有接口 没有接口不能实现jdk版动态代理!
2、代码实例
代理类工厂
public class ProxyFactory {
public static <T> Object getProxy(T t){
//返回一个代理对象
Object object = Proxy.newProxyInstance(t.getClass().getClassLoader(), t.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("执行方法前");
Object invoke = method.invoke(t, args);
System.out.println("执行方法后");
return invoke;
}
});
return object;
}
}
客户端
public class ProxyDemo {
public static void main(String[] args) {
/**
* 动态代理类
*/
PersonDao student = new StudentImpl("张三");
PersonDao proxy = (PersonDao)ProxyFactory.getProxy(student);
System.out.println("动态代理类-------->");
proxy.giveMoney();
}
}
四、Spring AOP中的动态代理
1、JDK的Proxy方式
底层是通过反射机制实现的。
2、Cglib代理
代理的原理是可以对目标对象接口实现代理,也可以进行继承代理(不能对final修饰的类进行继承代理)。
Cglib动态代理底层借助ASM(一个短小精悍的字节码操作框架)来操作字节码生成新的类。