Spring之Advice通知
Spring原生的经典模式 实现AOP
advice :通知
前置通知:在目标方法执行之前执行!不能改变方法的执行流程和结果!
实现MethodBeforeAdvice接口!
后置通知:在目标方法执行之后执行!不能改变方法的执行流程和结果!
实现AfterReturningAdvice接口!
环绕通知:方法的拦截器!进入了方法体,能改变方法的执行流程和结果!
实现MethodInterceptor接口!
异常通知: 当我们的目标方法发生异常才会被执行!
实现ThrowsAdvice接口!
如果想在项目中使用各种通知,必须引入需要的pom文件节点!
01.aop联盟
<dependency>
<groupId>aopalliance</groupId>
<artifactId>aopalliance</artifactId>
<version>1.0</version>
</dependency>
02. spring整合aop的jar
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>${spring.version}</version>
</dependency>
1. 创建需要的dao接口
2. 创建需要的daoImpl实现类
3. 创建前置增强类
4. 创建后置增强类
5. 创建环绕增强类
6. 创建异常增强类和自定义异常类
7. 配置spring.xml文件
<!--01. 配置目标对象 实际肯定是配置UserServiceImpl--> <bean id="userDaoImpl" class="com.xdf.dao.UserDaoImpl"/> <!--02.配置各种通知--> <!--001.前置通知--> <bean id="beforeAdvice" class="com.xdf.advice.BeforeAdvice"/> <!--002.后置通知--> <bean id="afterAdvice" class="com.xdf.advice.AfterAdvice"/> <!--003.环绕通知--> <bean id="aroundAdvice" class="com.xdf.advice.AroundAdvice"/> <!--004.异常通知--> <bean id="exceptionAdvice" class="com.xdf.advice.ExceptionAdvice"/> <!--03.需要配置代理工厂bean,来创建代理类! 从而把各种通知织入到目标对象的目标方法中--> <bean id="userProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!--注册了目标对象--> <property name="targetName" value="userDaoImpl"/> <!--注册通知--> <property name="interceptorNames"> <array> <value>beforeAdvice</value><!--前置通知--> <value>afterAdvice</value><!--后置通知--> <value>aroundAdvice</value><!--环绕通知--> </array> </property> </bean> <!--再次创建一个工厂 来为异常对象服务 因为一个工厂无法为多个对象服务--> <bean id="exceptionProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!--注册了目标对象--> <property name="targetName" value="userDaoImpl"/> <!--注册通知--> <property name="interceptorNames"> <array> <value>exceptionAdvice</value><!--异常通知--> </array> </property> </bean> </beans>
8. 创建测试类
@Test //前置通知 public void testBefore(){ ApplicationContext context=new ClassPathXmlApplicationContext("spring.xml"); /** * 获取的是代理工厂 返回的是目标对象 */ UserDao userDao= context.getBean("userProxy",UserDao.class); userDao.eat(); System.out.println("***********************"); userDao.sleep(); } @Test //后置通知 public void testAfter(){ ApplicationContext context=new ClassPathXmlApplicationContext("spring.xml"); UserDao userDao= context.getBean("userProxy",UserDao.class); userDao.eat(); } @Test //环绕通知 public void testAround(){ ApplicationContext context=new ClassPathXmlApplicationContext("spring.xml"); UserDao userDao= context.getBean("userProxy",UserDao.class); String result=userDao.eat(); System.out.println(result); } @Test //异常通知 public void testException(){ ApplicationContext context=new ClassPathXmlApplicationContext("spring.xml"); UserDao userDao= context.getBean("exceptionProxy",UserDao.class); try { userDao.login("aa","admin"); }catch (MyException ex){ ex.printStackTrace(); } }
面对以上案例 发现几个问题:
1.每个代理工厂只能给一个目标对象服务!实际开发过程中,不止一个目标对象!
2.不能实现给 指定的 主业务增强!
3.测试类getBean的时候还是获取的代理!应该是service层对象才对!
以上问题怎么解决呢?想知道的话,下次再来吧!哈哈~~(先让我们的大脑旋转起来吧!)