代理模式详解
` 温馨提示:此学习笔记参考狂神B站视频学习,可以结合视频进行学习
1、代理模式
为什么要学习代理模式?
因为这就是Spring AOP的底层!【Spring AOP和Spring MVC】
代理模式的分类:
- 静态代理
- 动态代理
】
角色分析:
- 抽象角色:一般会使用接口或者抽象类来 解决
- 真是角色:被代理的角色
- 代理角色:代理真是角色,代理真是角色后,我们一般会做一些附属操作
- 客户:访问代理对象的人
2、 静态代理
代码步骤:
-
接口
//租房 public interface Rent { public void rent(); }
-
真是角色
//房东 public class Host implements Rent{ public void rent() { System.out.println("房东要出租房子!"); } }
-
代理角色
public class Proxy implements Rent{ private Host host; public Proxy() { } public Proxy(Host host) { this.host = host; } public void rent() { seehouse(); host.rent(); signContract(); fee(); } //看房 public void seehouse(){ System.out.println("中介带你看房"); } //签合同 public void signContract(){ System.out.println("签合同"); } //收中介费 public void fee(){ System.out.println("收中介费"); } }
-
客户端访问代理角色
public static void main(String[] args) { //房东要租房子 Host host = new Host(); //代理,中介帮房东租房子,代理角色一般会有一些附属操作 Proxy proxy = new Proxy(host); //你不用面对房东,直接找中介租房即可。 proxy.rent(); }
代理模式的优点:
- 可以使真是角色的操作更纯粹,不用去关注一些公共业务
- 公共业务就交给代理角色,实现了业务的分工
- 公共业务发生扩展的时候,方便集中管理
缺点:
- 一个真是角色就会产生一个代理角色,代码量会翻倍,开发效率变低
3、静态代理加深理解
代码步骤:
-
接口
public interface UserService { public void add(); public void delete(); public void update(); public void query(); }
-
真是角色
//真实对象 public class UserServiceImpl implements UserService{ public UserServiceImpl() { super(); } public void add() { System.out.println("增加了一个用户"); } public void delete() { System.out.println("删除了一个用户"); } public void update() { System.out.println("修改了一个用户"); } public void query() { System.out.println("查询了一个用户"); } }
-
、代理角色
public class UserServiceProxy implements UserService{ private UserServiceImpl userService; // public UserServiceProxy(UserServiceImpl userService) { // } public void setUserService(UserServiceImpl userService) { this.userService = userService; } public void add() { log("add"); userService.add(); } public void delete() { log("delete"); userService.delete(); } public void update() { log("update"); userService.update(); } public void query() { log("query"); userService.query(); } //日志方法 public void log(String msg){ System.out.println("[Debug] 使用了"+ msg +"方法"); } }
-
客户端访问代理角色
public static void main(String[] args) { UserServiceImpl userService = new UserServiceImpl(); UserServiceProxy proxy = new UserServiceProxy(); proxy.setUserService(userService); proxy.add(); }
聊聊AOP
4、动态代理
- 动态代理和静态代理角色一样
- 动态代理的代理类是动态生成的,不是我们直接写好的
- 动态代理类分为两大类:基于接口的动态代理,基于类的动态代理
- 基于接口—>JDK 动态代理
- 基于类:cglib
- java字节码实现:javasist
动态代理类 (以下简称为代理类 ):是一个实现在类创建时在运行时指定的接口列表的类,具有如下所述的行为。 代理接口是由代理类实现的接口。 代理实例是代理类的一个实例。 每个代理实例都有一个关联的调用处理程序对象,它实现了接口InvocationHandler。 通过其代理接口之一的代理实例上的方法调用将被分派到实例调用处理程序的[invoke方法,传递代理实例, java.lang.reflect.Method被调用方法的java.lang.reflect.Method对象以及包含参数的类型Object Object的数组。 调用处理程序适当地处理编码方法调用,并且返回的结果将作为方法在代理实例上调用的结果返回。
需要了解两个类:Proxy:代理 ; InvocationHandler:调用处理程序。
Proxy:提供了创建动态代理类和实例的静态方法,它也是由这些方法创建的所有动态代理类的超类。
InvocationHandler:是由代理实例的调用处理程序实现的接口 。
动态代理的优点:
- 可以使真是角色的操作更纯粹,不用去关注一些公共业务
- 公共业务就交给代理角色,实现了业务的分工
- 公共业务发生扩展的时候,方便集中管理
- 一个动态代理类代理的是一个接口,一般就是对应的一类业务
- 一个动态代理类可以代理多个类,只要实现了同一个接口即可
代码实现:
-
接口
public interface Rent { public void rent(); }
-
真实角色
public class Host implements Rent{ public void rent() { System.out.println("房东要租房子!"); } }
-
代理角色
//我们会用这个类,自动生成代理类 public class ProxyInvocationHandler implements InvocationHandler { //被代理的接口 private Rent rent; public void setRent(Rent rent) { this.rent = rent; } //生成得到代理类 public Object getProxy(){ return Proxy.newProxyInstance(this .getClass().getClassLoader(),rent.getClass().getInterfaces(),this); }
-
客户端
public class Client { public static void main(String[] args) { //真实角色 Host host = new Host(); //代理角色,现在没有 ProxyInvocationHandler pih = new ProxyInvocationHandler(); //通过调用程序处理角色来处理我们要调用的接口对象! pih.setRent(host); Rent proxy = (Rent) pih.getProxy();//这里的proxy是动态生成的,我们并没有写 proxy.rent(); } }