Spring能够为容器中管理的对象创建代理对象。
以前我们创建动态代理对象要调用该方法:
proxy.newProxyInstance(xx,xx,xx)。
Spring中是使用动态代理和cglib代理混合使用,优先使用动态代理,如果不能使用动态代理,则使用cglib代理。
动态代理:被代理对象必须要实现接口,才能产生代理对象.如果没有接口将不能使用动态代理技术。被代理对象和代理对象只是实现了同一接口。
cglib代理:第三方代理技术(Spring中整合了该jar,所以不用导包),cglib代理.可以对任何类生成代理.代理的原理是对目标对象进行继承代理. 如果目标对象被final修饰.那么该类无法被cglib代理。被代理和代理对象是继承和被继承的关系。
public class UserServiceProxyFactory2 implements MethodInterceptor {
public UserService getUserServiceProxy(){
Enhancer en = new Enhancer();//帮我们生成代理对象
en.setSuperclass(UserServiceImpl.class);//设置对谁进行代理
en.setCallback(this);//代理要做什么
UserService us = (UserService) en.create();//创建代理对象
return us;
}
@Override
public Object intercept(Object prxoyobj, Method method, Object[] arg, MethodProxy methodProxy) throws Throwable {
//打开事务
System.out.println("打开事务!");
//调用原有方法
Object returnValue = methodProxy.invokeSuper(prxoyobj, arg);
//提交事务
System.out.println("提交事务!");
return returnValue;
}
}
JoinPoint(连接点):目标对象中,所有可以增强的方法。
Pintcut(切入点):目标对象,已经增强的方法。
Advice(通知/增强):增强的代。
Target(目标对象):被代理对象。
Weaving(织入):将通知应用到切入点的过程。
Proxy(代理):将通知织入到目标对象之后,形成代理对象。
aspect(切面):切入点+通知
使用Spring创建代理对象
4+2+2+2
Spring中的aop包
Spring需要的第三方包
public class Target{
public void add() {
System.out.println("add");
}
public void remove() {
System.out.println("remove");
}
public void update() {
System.out.println("update");
}
public void query() {
System.out.println("query");
}
}
该对象可以实现接口,也可以不用实现,实现了接口则Spring会优先使用动态代理。如果没有实现接口,Spring会使用cglib代理。
注意,如果使用接口,通过getBean获得对象时,记得强制转换对象时,使用接口,而不是目标类。否则会出现ClassCaseException
public class AdviceClass {
public void before() {
System.out.println("这是前置通知");
}
public void afterReturning() {
System.out.println("这是后置通知(出现异常将不会调用)");
}
public Object around(ProceedingJoinPoint pj) throws Throwable {
System.out.println("这是环绕通知之前的部分");
Object proceed = pj.proceed();//调用目标方法
System.out.println("这是环绕通知之后的部分");
return proceed;
}
public void afterException() {
System.out.println("出现异常后调用");
}
public void after() {
System.out.println("这是后置通知(无论会不会出现异常都会调用)");
}
}
public class AopTest {
@Test
public void test() {
ApplicationContext ac = new ClassPathXmlApplicationContext("club/ityuchao/test/applicationContext.xml");
Target target = (Target) ac.getBean("target");
target.add();
}
}
测试结果:
出现异常:
出现异常后,环绕通知的后执行将不会执行,afterRetruning也不会执行
(1)applicationContext.xml中配置
(2)advice中使用注解
//表示该类是一个通知类
@Aspect
public class AdviceClass {
//使用该方法可以不用下面每一个方法都写expression
@Pointcut("execution(* club.ityuchao.test.Target.*(..))")
public void pc() {}
@Before("execution(* club.ityuchao.test.Target.*(..))")
public void before() {
System.out.println("这是前置通知");
}
@AfterReturning("AdviceClass.pc()")
public void afterReturning() {
System.out.println("这是后置通知(出现异常将不会调用)");
}
@Around("AdviceClass.pc()")
public Object around(ProceedingJoinPoint pj) throws Throwable {
System.out.println("这是环绕通知之前的部分");
Object proceed = pj.proceed();//调用目标方法
System.out.println("这是环绕通知之后的部分");
return proceed;
}
@AfterThrowing("AdviceClass.pc()")
public void afterException() {
System.out.println("出现异常后调用");
}
@After("AdviceClass.pc()")
public void after() {
System.out.println("这是后置通知(无论会不会出现异常都会调用)");
}
}