一、代理模式
1. 代理模式简介
客户端不直接调用实际对象,而是通过代理间接调用实际的对象
SpringAOP的底层就是代理模式
代理模式分为静态代理模式和动态代理模式
2. 静态代理
2.1 角色
1. 抽象角色:
一般会用接口或抽象类来实现(租房)
2. 真实角色:
被代理的角色(房东)
3. 代理角色:
代理真实角色(中介)
4. 客户:
访问代理对象的人(租客)
2.2. 代码实现
2.2.1 原本
1. 租房接口
//租房接口
public interface Rent
{
public void rent();
}
2. 房东
//房东
public class Host implements Rent
{
@Override
public void rent()
{
System.out.println("房东要出租房子了!");
}
}
3. 客户端
//客户端(租客)
public class Client
{
public static void main(String[] args) {
Host host=new Host();
host.rent();
}
}
2.2.2 静态代理
1. 代理类
public class MyProxy implements Rent
{
Host host=new Host();
@Override
public void rent() {
host.rent();
}
}
2. 客户端
原本调用的是Host的方法,现在是代理类的方法
//客户端(租客)
public class Client
{
public static void main(String[] args) {
MyProxy myProxy=new MyProxy();
myProxy.rent();
}
}
代理一般会有一些附属操作,如带租客去看房子,签合同,这些是房东不用做的,房东只负责出租房子
因此在中介的rent方法里,可以加入一些附属操作
public class MyProxy implements Rent
{
Host host=new Host();
@Override
public void rent() {
seeHouse();
sign();
host.rent();
}
public void seeHouse()
{
System.out.println("中介带租客去看房子");
}
public void sign()
{
System.out.println("中介与租客签合同");
}
}
这就像我们的业务类
此时若需要在已经写好的业务类里面,添加新的方法
为了不改变原有代码,我们用一个代理类,和业务类实现同样的接口,实现接口的方法
然后在代理类里,编写其他要添加的附属方法即可。
2.3. 优点
- 让真实角色更加纯粹,不用去关注一些公共的业务
- 公共的方法交给代理角色,实现角色的分工
- 公共业务发生扩展时,方便集中管理
2.4. 缺点
一个真实角色产生一个代理角色,代码量翻倍,开发效率低------>动态代理
3. 动态代理
3.1 动态代理简介
动态代理的代理类是动态生成的,不是我们直接写好的,也就是我们不需要去写代理类了
3.2 动态代理分类
- 基于接口的动态代理(jdk)
- 基于类的动态代理(cglib)
3.3 动态代理实现
3.3.1 两个类
1. InvocationHandler
简介:
InvocationHandler是由代理实例的调用处理程序实现的接口。也就是有一个代理实例的调用处理程序的类,实现了该接口,我们需要去写一个代理实例的调用处理程序,去实现该接口。
每个代理实例都有一个关联的调用处理程序
当在代理实例上调用方法时,方法调用将被编码并分派到其调用处理程序的invoke方法
invoke方法:
Object invoke(Object proxy,方法 method,Object[] args) throws Throwable
处理代理实例上的方法调用并返回结果
当在与之关联的代理实例上调用方法时,将在调用处理程序中调用该方法
参数:
1. proxy:调用该方法的代理实例
2. method:所述方法对应于调用代理实例上的接口方法的实例(也就是原方法)
方法对象的声明类将是该方法声明的接口
3. args:包含的方法调用传递代理实例的参数值的对象的阵列
2. Proxy
public class Proxy extends Object implements Serializable
提供了创建动态代理类和实例的静态方法
也就是说可以用该类去调用该类内部的一个方法,来达到创建动态代理类的功能
Proxy本身也是由这些方法创建的所有动态代理类的超类
动态代理类:
是一个实现在类创建时,在运行时指定的接口列表的类
代理接口是由代理类实现的接口
代理实例是代理类的一个实例
每个代理实例都有一个关联的调用处理程序对象,它实现了接口InvocationHandler
通过其代理接口之一的代理实例上的方法调用,将被分派到实例调用处理程序的invoke方法,传递代理实例,方法和参数
调用处理程序适当地处理编码方法调用,并且返回的结果将作为 方法在代理实例上调用的结果 返回
3.3.2 代码实现
调用处理程序处理角色
//等会会用该类来自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler
{
private Rent rent;
public void setRent(Rent rent)
{
this.rent=rent;
}
public Object getProxy()
{
//Proxy.newProxyInstance(类加载器,接口,InvocationHandler)创建代理实例
return Proxy.newProxyInstance(this.getClass().getClassLoader(),rent.getClass().getInterfaces(),this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//反射
Object invoke = method.invoke(rent, args);
return invoke;
}
}
客户端
public class Client
{
public static void main(String[] args) {
//真实角色
Host host=new Host();
//代理角色,现在没有
ProxyInvocationHandler proxyInvocationHandler = new ProxyInvocationHandler();
//通过调用程序处理角色来处理我们要调用的接口对象
proxyInvocationHandler.setRent(host);
//获得代理类
Rent proxy = (Rent) proxyInvocationHandler.getProxy();
//用代理类去调用方法
proxy.rent();
}
}
在调用处理程序的类里面,添加附属方法
//等会会用该类来自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler
{
private Rent rent;
public void setRent(Rent rent)
{
this.rent=rent;
}
public Object getProxy()
{
//Proxy.newProxyInstance(类加载器,接口,InvocationHandler)创建代理实例
return Proxy.newProxyInstance(this.getClass().getClassLoader(),rent.getClass().getInterfaces(),this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
seeHouse();
//反射
Object invoke = method.invoke(rent, args);
return invoke;
}
public void seeHouse()
{
System.out.println("中介带租客去看房子啦!");
}
}
3.4 通用实现
public class ProxyInvocationHandler implements InvocationHandler
{
//被代理的接口
private Object target;
public Object getTarget() {
return target;
}
public Object getProxy()
{
return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
log(method.getName());
Object invoke = method.invoke(target, args);
return invoke;
}
public void log(String msg)
{
System.out.println("执行了"+msg+"方法");
}
}
public class Client
{
public static void main(String[] args)
{
//真实角色
Service service = new ServiceImpl();
//代理角色,不存在
ProxyInvocationHandler proxyInvocationHandler = new ProxyInvocationHandler();
//设置要代理的对象
proxyInvocationHandler.setTarget(service);
//动态生成代理类
Service proxy = (Service) proxyInvocationHandler.getProxy();
//调用方法
proxy.add();
proxy.delete();
proxy.select();
proxy.update();
}
}
二、AOP
1. AOP简介
1.1 AOP在Spring中的作用
- 提供声明式事务
- 允许用户自定义切面
1.2 抽象概念
- 横切关注点:跨越应用程序多个模块的方法或功能。即与业务逻辑无关的,但需要关注的部分
- 切面(aspect):横切关注点被模块化的特殊对象,即将横切关注点整合成一个类
- 通知(advice):切面必须要完成的工作,即类中的一个方法
- 目标(target):被通知的对象,即被扩展方法的对象
- 代理(Proxy):向目标对象应用通知之后创建的对象
- 切入点(pointcut):切面通知执行的地点,即扩展的方法要在哪里实现
- 连接点(joinpoint):与切入点匹配的执行点
1.3 通知类型
- 前置通知
- 后置通知
- 环绕通知
- 异常抛出通知
- 引介通知:在类中增加新的方法属性
1.4 总结
即AOP在不改变原有代码的情况下,去增加新的功能
2. AOP实现
要使用AOP织入,首先要导入一个包
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
2.1 SpringAPI接口实现
前置日志
//实现接口
public class BeforeLog implements MethodBeforeAdvice
{
/**
*
* @param method 要执行的目标对象的方法
* @param args 参数
* @param target 目标对象
* @throws Throwable
*/
@Override
public void before(Method method, Object[] args, Object target) throws Throwable
{
System.out.println(target.getClass().getName()+"类的"+method.getName()+"方法被执行了");
}
}
后置日志
public class AfterLog implements AfterReturningAdvice
{
/**
*
* @param returnValue 返回值
* @param method
* @param args
* @param target
* @throws Throwable
*/
@Override
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println(target.getClass().getName()+"类的"+method.getName()+"方法被执行了,返回结果为:"+returnValue);
}
}
Spring配置
<bean id="userService" class="com.xk.aop1.service.UserServiceImpl"/>
<bean id="beforelog" class="com.xk.aop1.log.BeforeLog"/>
<bean id="afterlog" class="com.xk.aop1.log.AfterLog"/>
<!--方式一:使用原生的SpringAPI接口-->
<aop:config>
<!--切入点-->
<aop:pointcut id="pointcut" expression="execution(* com.xk.aop1.service.UserServiceImpl.*(..))"/>
<!--执行通知-->
<aop:advisor advice-ref="beforelog" pointcut-ref="pointcut"/>
<aop:advisor advice-ref="afterlog" pointcut-ref="pointcut"/>
</aop:config>
测试
public class Test
{
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserService userService = context.getBean("userService",UserService.class);
userService.add();
userService.delete();
userService.select();
userService.update();
}
}
结果
2.2 自定义实现(切面定义)
自定义日志类
public class MyLog
{
public void before()
{
System.out.println("这是前置通知");
}
public void after()
{
System.out.println("这是后置通知");
}
}
Spring配置
<!--方式二:自定义-->
<bean id="mylog" class="com.xk.aop2.log.MyLog"/>
<aop:config>
<!--切面-->
<aop:aspect ref="mylog">
<!--切入点-->
<aop:pointcut id="point" expression="execution(* com.xk.aop2.service.UserServiceImpl.*(..))"/>
<!--通知-->
<aop:before method="before" pointcut-ref="point"></aop:before>
<aop:after method="after" pointcut-ref="point"></aop:after>
</aop:aspect>
</aop:config>
结果
2.3 注解实现
日志
@Aspect //标注这是一个切面
public class AnnotationPointCut
{
@Before("execution(* com.xk.aop3.service.UserServiceImpl.*(..))")
public void before()
{
System.out.println("方法执行前");
}
}
Spring配置
<!--方式三:注解-->
<bean id="annotationPointCut" class="com.xk.aop3.log.AnnotationPointCut"/>
<!---开启注解支持-->
<aop:aspectj-autoproxy/>