首先,为什么学Spring,学到代理模式来了?不是学Spring嘛?
为什么要学习代理模式?
因为这就是SpringAOP的底层!!SpringAOP的底层就是用代理实现的!!
所以要理解、懂代理模式,才能学懂SpringAOP!
代理模式,也是java,23中设计模式之一!
代理模式的分类:
1、静态代理
2、动态代理
举一个例子,我们在城市中租房子,往往找的都不是房东,而是中介,那么这个中介就可以说是一个代理!房东不想跟租客打交道,他只需要把房子给中介代理就可以了,这样租客就只能和中介代理谈价钱、看房、签合同。那么这就有共同点,房东跟中介代理都要租房,都想把房子租出去!如图:
角色分析:
抽象角色:一般会使用接口或者抽象类来解决
真实角色:被代理的角色
代理角色:代理真实角色,代理真实角色后,我们一般会做一些附属操作
客户: 访问代理对象的人!
--------------------------------------------------------------------------------------------------------------------
代码步骤:
1、(抽象角色)接口
//租房
2、真实角色
//房东
3、代理角色
//中介
4、客户端访问代理角色
--------------------------------------------------------------------------------------------------------------------
接着,我们就来写一下代码演示一下静态代理。
1、(抽象角色)接口
//租房
创建一个接口Rent
代理模式的好处:
可以使真实角色的操作更加纯粹,不用去关注一些公共的业务
公共也就就交给代理角色!实现了业务的分工!
公共业务发生扩展的时候,方便集中管理!
缺点:
一个真实角色就会产生一个代理角色,代码量会翻倍开发效率会变低~
接着上面的,我们再举一个例子加深理解这两句话:
公共也就就交给代理角色!实现了业务的分工!
公共业务发生扩展的时候,方便集中管理!
--------------------------------------------------------------------------------------------------------------------
我们新建一个包,创建一个UserService接口。
那么现在,我们的需求来了,我们要给它的每一个方法都添加一个日志功能。这怎么办呢?
有两种方法:
第一种 :在原有的代码上改
第二种: 使用代理来做,在不改变原有业务情况下,实现功能。
我们肯定是选择第二种,因为第一种方法在公司中是大忌,不能直接修改原有的业务代码,万一改崩了呢?那之前的是不是就崩掉了啊?所以选择第二种。
创建一个代理类
测试,没有问题
我们在上面的静态代理也说了,静态代理有一个缺点,就是这:
一个真实角色就会产生一个代理角色,代码量会翻倍开发效率会变低~
那么我们的动态代理就是解决这个问题的,我们不写代理类(代理角色)
--------------------------------------------------------------------------------------------------------------------
动态代理和静态代理角色一样
动态代理的代理类是动态生成的,不是我们直接写好的!
动态代理分为两大类:基于接口的动态代理,基于类的动态代理
基于接口——JDK动态代理(在这里,我们使用的是这个!)
基于类:cglib
java字节码实现:javasist
需要了解两个类:
Proxy:代理
InvocationHandler:调用处理程序
我们就开始写代码了。
--------------------------------------------------------------------------------------------------------------------
我们新建一个包agencyTwo,把我们之前的两个类复制过来。
接着,我们写一个ProxyInvocationHandler类,这个类的作用是自动生成代理类。
接着,我们测试一下,没有问题
通过上面的例子。如果你看不懂,还没有理解,那么我们继续第二个,我们写一个万能的,我们继续改造。
我们去代理一下,我们之前agencyone包里面的UserServiceImpl类,他里面有四个方法。
我们创建一个包agencyThree,把之前的ProxyInvocationHandler类拿过来进行改造。
如下:
这样,它是不是就通用了啊?想比之前,我们没有写死,你传给我什么,我就
用它去创建代理类,这样它是不是就成工具类了啊?这样就通用了。
--------------------------------------------------------------------------------------------------------------------
接着,我们来试一下。
创建一个Client类。运行,测试,没有问题。
那么现在问题来了,我们还是想给它加一个日志功能,这个怎么做呢?
我们来写一下:
那么我们怎么办呢?我们可以用它的method。就反射就完了。
需要了解的是,别人问你动态代理,你要知道Proxy跟InvocationHandler这两个类,然后Proxy是干嘛的?是生成动态代理这个实例的,InvocationHandler是调用处理程序并返回一个结果的。就白了就两件事情。