Spring入门4.AOP配置深入

Spring入门4.AOP配置深入

代码下载 链接: http://pan.baidu.com/s/11mYEO 密码: x7wa

前言:

之前学习AOP中的一些概念,包括连接点、切入点(pointcut),切面(Aspect),Advice,Proxy代理等等,同时学历了使用XML配置AOP的方式,这一个过程中遇到很多的错误,比如配置文件中的<aop:aspect-autoproxy/>,如果在配置中没有添加这一行配置,就会编译出错。还有学习Advice的5种方式以及配置原理。

这一次我们深入的学习XML配置的内容,然后是Annotation的配置方式。

一、XML配置方式深入

1.配置切入点

配置切入点的时候需要给出属性pointcut的属性值,指定切入点表达式。同时还可以使用<aop:pointcut/>元素在<aop:config>中单独的定义切入点,实际上就是为切入点起一个名字,在后面多个Advice中使用。

<aop:config proxy-target-class=”true”>

<aop:pointcut id = “mypointcut1” expression=”execution(* com.yang.service.*(..))”/>

<aop:aspect id=”adviceAspect” ref=”aspectBean”>

<aop:before methos=”methodName” pointcut-ref=”mypointcut1”/>

</aop:aspect>

</aop:config>

这样就可以重复使用了,对于精简代码十分有帮助。

2.切入点指示符

SpringAOP提供了多种切入点指示符来定义切入点的规则。切入点知识符用于标明一个连接点在什么条件下关联到切入点。切入点表达式就是通过切入点指示符进行定义的,如execution就是一种切入点指示符。

在Spring3中的切入点指示符有:

execution:用于匹配执行方法的连接点,他是SpringAOP中最主要的切入点指示符。

within:限定匹配特定类型的连接点,当使用Spring AOP的时候,只能匹配方法执行的连接点;只可以使用包名进行匹配,比如within(com.yang.service..*)

this:用于限定AOP代理必须是指定类型的实例,用于匹配该对象的所有连接点。当使用SpringAOP的时候,只能匹配方法执行的连接点.比如this(com.yang.service.UserService)

target:用于限定目标对象必须是指定类型的实例,用于匹配该对象的所有连接点。当使用SpringAOP的时候,只能匹配方法执行的连接点。

args:用于对连接点的参数类型进行限制,要求参数类型是指定类型的实例。Spring AOP值能够匹配方法执行的连接点。

execution表达式的语法格式如下:

execution(

modifiers-pattern? 指定方法的修饰符,支持通配符,可以省略             Return-typepatterns  指定方法的返回值类型,支持通配符, 可以使用* 通配符匹配所有类型的返回值类型

declaring-type-pattern?  指定方法所属的类,支持通配符,可以省略

Name-pattern(param-pattern) 指定匹配的方法名,支持通配符,可以使用*匹配所有的方法,其中的param-pattern指的是参数类型,支持两种通配符,一种是*,表示一个任意类型的参数;另一个是..,表示的是0个或者是多个任意类型的参数

throws-pattern?  表示指定的方法声明抛出异常,支持通配符,可以省略)

通过方法签名定义切入点:

execution(public       *       *(..)) :匹配目标类所有的public方法

execution(* find*(..))

通过类定义切入点,通过包定义切入点,通过方法形参定义切入点

execution(* com.yang.service.UserService+.*(..)) +表示其接口还有其实现子类中的方法

execution(* com..*.yang.service.*(..))表示包括子包

同时可以使用组合表达式&& || !将多个切入点表达式组合起来。

二、基于Annotation配置的AOP

其实掌握了XML方式配置的AOP之后,转换成为Annotation方式是十分容易的。AspectJ是一个基于Java语言的AOP框架,他扩展了标准的Java,从语言层面上提供了更加强大的AOP功能。AspectJ是AOP的实现之一,对整套的AOP机制都有着很好的实现机制,所以不使用Spring也可以直接使用AspectJ进行切面编程。从Spring2开始,就已经将AspectJ集成在Spring中。AspectJ允许使用Annotation来定义切面(Aspect)、切入点(pointcut)、增强(Advice)。Spring框架根据这些Annotation来生成对应的AOP代理。

1.配置切面在Spring配置文件中需要添加

<aop:aspectj-autoproxy/>

增加上述配置之后,Spring会根据注解判断一个Bean是否使用了一个或者多个切面,然后自动生成相对应的AOP代理以及拦截器方法调用,并且确认Advice是否如期进行。这里使用注解的方式,需要增加两个AspectJ的类库aspectjweaver.jar&& aspectjrt.jar

2.声明切面

在创建的AspectBean类前面声明是切面,使用如下方式:

@Aspect

public class AspectBean{}

3.配置Advice

AspectJ有一下几种配置方法,这些注解的存留期是RetentionPolicy.RUNTIME,注解目标都是ElementType.METHOD

AspectJ提供的几种注解方式:

@Before 成员有value指定一个切入点表达式,也可以指定一个已经存在的切入点

@AfterReturning value、 returning 表示目标对象返回值绑定给增强的方法, pointcut会覆盖

@Around value切入点表达式

@AfterThrowing value 切入点表达式 pointcut覆盖掉value, throwing将抛出的异常绑定到增强的方法

@After value

4配置切入点

使用@Pointcut进行生命,一个切入点的声明有两个部分组成,包含名字和任意参数的签名和切入点表达式。

@Aspect

public class AspectBean {

//@Before(value="execution(* com.yang.service.UserService.*(..))")

@Pointcut("execution(* com.yang.service.UserService.*(..))")

private void crud(){}

@Before(value="crud")

public void checkAuth(){

System.out.println("AspectBean.checkAuth()");

}

}

源代码:

UserService.java

package com.yang.service;

public interface UserService {

public void del();

public void add();

public void update();

public Object select();

}

UserServiceImpl.java

package com.yang.service.impl;

import com.yang.service.UserService;

public class UserServiceImpl implements UserService {

@Override

public void del() {

System.out.println("UserServiceImpl.del() function ");

}

@Override

public void add() {

System.out.println("UserServiceImpl.add() function ");

}

@Override

public void update() {

System.out.println("UserServiceImpl.update() function ");

}

@Override

public Object select() {

System.out.println("UserServiceImpl.select() function ");

return null;

}

}

AspectBean.java

import org.aspectj.lang.ProceedingJoinPoint;

//@Aspect

public class AspectBean {

//@Before(value="execution(* com.yang.service.UserService.*(..))")

//@Pointcut("execution(* com.yang.service.UserService.*(..))")

private void crud(){}

//@Before(value="crud")

public void checkAuth(){

System.out.println("AspectBean.checkAuth()");

}

public void release(){

System.out.println("AspectBean.release()");

}

public void log(Object result){

System.out.println("AspectBean.log()");

if(result==null){

System.out.println("database have result");

}else{

System.out.println("database has no result");

}

}

public void processException(Throwable ex){

System.out.println("Exception information:" + ex.getMessage());

}

public void proceedInTrans(ProceedingJoinPoint joinpoint) throws Throwable{

System.out.println("start transtraction");

joinpoint.proceed();

System.out.println("end transtraction");

}

}

bean.xml

<?xml version="1.0" encoding="UTF-8"?>

<beans

xmlns="http://www.springframework.org/schema/beans"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:aop="http://www.springframework.org/schema/aop"

xmlns:tx="http://www.springframework.org/schema/tx"

xsi:schemaLocation="

        http://www.springframework.org/schema/beans

        http://www.springframework.org/schema/beans/spring-beans-3.0.xsd

        http://www.springframework.org/schema/aop

        http://www.springframework.org/schema/aop/spring-aop-3.0.xsd

        http://www.springframework.org/schema/tx

        http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"

default-lazy-init="true">

<aop:aspectj-autoproxy/>

<bean id="aspectBean" class=" com.yang.aspect.AspectBean"></bean>

<bean id="userService" class="com.yang.service.impl.UserServiceImpl"></bean>

<aop:config proxy-target-class="true">

<aop:aspect id="myadvice" ref="aspectBean">

<aop:before method="checkAuth" pointcut="execution(* com.yang.service.UserService.*(..))"/>

<aop:after-returning method="log" pointcut="execution(* com.yang.service.UserService.select(..))" returning="result"/>

<aop:after-throwing method="processException" pointcut="execution(* com.yang.service.UserService.*(..))" throwing="ex"/>

<aop:around method="proceedInTrans" pointcut="execution(* com.yang.service.UserService.*(..))"/>

<aop:after method="release" pointcut="execution(* com.yang.service.UserService.*(..))"/>

</aop:aspect>

</aop:config>

</beans>

TestMain.java

ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");

//System.out.println("before");

UserService userService = (UserService) context.getBean("userService");

System.out.println("=======================================");

userService.add();

System.out.println("=======================================");

userService.select();

System.out.println("=======================================");

YangTengfei

2013.11.26

补充知识:

记得自己在S2SH框架整合的时候,遇到的一个问题,其实是因为自己对于前面知识掌握的不怎么到位,以至于在S2SH整合的时候因为一个小小的知识点的错误,而导致了我找了4个小时的错误才依旧是没有找出来。

先说一下自己的错误:在S2SH框架整合的时候,最后一步关于实现S2SH框架中的事务机制,我们使用的是Hibernate的HibernateTransactionManager,然后配置了事务的切入点:具体配置如下:

<!-- S2SH配置的 Hibernate 的事物管理器 -->

<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">

<property name="sessionFactory" ref="sessionFactory"/>

</bean>

<!--  配置事务的Advice,指定事务管理器 -->

<tx:advice id="txAdvice" transaction-manager="txManager">

<!-- 配置事务的详细定义 -->

<tx:attributes>

<tx:method name="add*" rollback-for="Exception"/>

<tx:method name="*" />

</tx:attributes>

</tx:advice>

<aop:config>

<!-- 配置切入点信息,并且将它和切面整合 -->

<aop:advisor advice-ref="txAdvice" pointcut="execution(* com.yang.s2s.service.impl.*.*(..))"/>

</aop:config>

强敌出现:

在这个配置中,启动Tomcat服务器总是出错,然后就是各种错误说是dataSource出错,然后我们尝试各种方法总是找不到错误,悲了个剧,纠结了4个小时还是不可以。原因是我么配置的pointcut的表达式错误。

回顾配置这个的切入点的规范:

execution (

modifiers-pattern?

return-type-pattern

declaring-type-pattern?

Name-pattern(param-pattern)

throw-pattern?

)

其中的name-pattern * 表示包下的所有类,之后必须还有一个匹配方法,因为配置事务的基本单位是方法而不是类。所以需要修改成为 execution(* com.yang.s2s.service.impl.*Impl.*(..))表示的是任何返回值类型的 在com.yang.s2s.service.impl.包下的以Impl结尾的Class中的任意方法,其中参数也是任意的。

Add by YangTengfei

2013.12.02

上一篇:Django视图函数


下一篇:腾讯云分布式高可靠消息队列服务CMQ架构