Java程序员从笨鸟到菜鸟之(七十四)细谈Spring(六)spring之AOP基本概念和配置详解

分享一下我老师大神的人工智能教程!零基础,通俗易懂!http://blog.csdn.net/jiangjunshow

也欢迎大家转载本篇文章。分享知识,造福人民,实现我们中华民族伟大复兴!

               

       首先我们来看一下官方文档所给我们的关于AOP的一些概念性词语的解释:

切面(Aspect):一个关注点的模块化,这个关注点可能会横切多个对象。事务管理是J2EE应用中一个关于横切关注点的很好的例子。在Spring AOP中,切面可以使用基于模式)或者基于Aspect注解方式来实现。通俗点说就是我们加入的切面类(比如log类),可以这么理解。

连接点(Joinpoint:在程序执行过程中某个特定的点,比如某方法调用的时候或者处理异常的时候。在Spring AOP中,一个连接点总是表示一个方法的执行。通俗的说就是加入切点的那个点

通知(Advice:在切面的某个特定的连接点上执行的动作。其中包括了“around”“before”“after”等不同类型的通知(通知的类型将在后面部分进行讨论)。许多AOP框架(包括Spring)都是以拦截器做通知模型,并维护一个以连接点为中心的拦截器链。

切入点(Pointcut:匹配连接点的断言。通知和一个切入点表达式关联,并在满足这个切入点的连接点上运行(例如,当执行某个特定名称的方法时)。切入点表达式如何和连接点匹配是AOP的核心:Spring缺省使用AspectJ切入点语法。

引入(Introduction:用来给一个类型声明额外的方法或属性(也被称为连接类型声明(inter-type declaration))。Spring允许引入新的接口(以及一个对应的实现)到任何被代理的对象。例如,你可以使用引入来使一个bean实现IsModified接口,以便简化缓存机制。

目标对象(Target Object: 被一个或者多个切面所通知的对象。也被称做被通知(advised)对象。 既然Spring AOP是通过运行时代理实现的,这个对象永远是一个被代理(proxied)对象。

AOP代理(AOP ProxyAOP框架创建的对象,用来实现切面契约(例如通知方法执行等等)。在Spring中,AOP代理可以是JDK动态代理或者CGLIB代理。

织入(Weaving:把切面连接到其它的应用程序类型或者对象上,并创建一个被通知的对象。这些可以在编译时(例如使用AspectJ编译器),类加载时和运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。



通知类型:

前置通知(Before advice:在某连接点之前执行的通知,但这个通知不能阻止连接点之前的执行流程(除非它抛出一个异常)。

后置通知(After returning advice:在某连接点正常完成后执行的通知:例如,一个方法没有抛出任何异常,正常返回。

异常通知(After throwing advice:在方法抛出异常退出时执行的通知。

最终通知(After (finally) advice:当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。

环绕通知(Around Advice):包围一个连接点的通知,如方法调用。这是最强大的一种通知类型。环绕通知可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它自己的返回值或抛出异常来结束执行。

环绕通知是最常用的通知类型。和AspectJ一样,Spring提供所有类型的通知,我们推荐你使用尽可能简单的通知类型来实现需要的功能。例如,如果你只是需要一个方法的返回值来更新缓存,最好使用后置通知而不是环绕通知,尽管环绕通知也能完成同样的事情。用最合适的通知类型可以使得编程模型变得简单,并且能够避免很多潜在的错误。比如,你不需要在JoinPoint上调用用于环绕通知的proceed()方法,就不会有调用的问题。



spring AOP的实现

      在spring2.5中,常用的AOP实现方式有两种。第一种是基于xml配置文件方式的实现,第二种是基于注解方式的实现。接下来,以具体的示例来讲解这两种方式的使用。下面我们要用到的实例是一个注册,就有用户名和密码,我们利用AOP来实现在用户注册的时候实现在保存数据之前和之后或者是抛出异常时,在这些情况下都给他加上日志。在这里我们只讲解AOP,所以我只把关键代码贴出来,不相干的就不贴了。

首先我们来看一下业务逻辑service层:

/** * RegisterService的实现类 * @author 曹胜欢 */public class RegisterServiceImpl implements RegisterService {    private  RegisterDao registerDao;    public RegisterServiceImpl() {}    /** 带参数的构造方法 */    public RegisterServiceImpl(RegisterDao  registerDao){        this.registerDao =registerDao;    }    public void save(String loginname, String password) {        registerDao.save(loginname, password);        throw new RuntimeException("故意抛出一个异常。。。。");    }      /** set方法 */    public void setRegisterDao(RegisterDao registerDao) {        this.registerDao = registerDao;}}


对于业务系统来说,RegisterServiceImpl类就是目标实现类,它的业务方法,如save()方法的前后或代码会出现异常的地方都是AOP的连接点。

 

下面是日志服务类的代码:


/** * 日志切面类 * @author 曹胜欢 */public class LogAspect {    //任何通知方法都可以将第一个参数定义为 org.aspectj.lang.JoinPoint类型     public void before(JoinPoint call) {        //获取目标对象对应的类名        String className = call.getTarget().getClass().getName();        //获取目标对象上正在执行的方法名        String methodName = call.getSignature().getName();        System.out.println("前置通知:" + className + "类的" + methodName + "方法开始了");    }    public void afterReturn() {        System.out.println("后置通知:方法正常结束了");    }    public void after(){        System.out.println("最终通知:不管方法有没有正常执行完成,一定会返回的");    }    public void afterThrowing() {        System.out.println("异常抛出后通知:方法执行时出异常了");    }    //用来做环绕通知的方法可以第一个参数定义为org.aspectj.lang.ProceedingJoinPoint类型    public Object doAround(ProceedingJoinPoint call) throws Throwable {        Object result = null;        this.before(call);//相当于前置通知        try {            result = call.proceed();            this.afterReturn(); //相当于后置通知        } catch (Throwable e) {            this.afterThrowing();  //相当于异常抛出后通知            throw e;        }finally{            this.after();  //相当于最终通知        }        return result;    }}


     这个类属于业务服务类,如果用AOP的术语来说,它就是一个切面类,它定义了许多通知。Before()afterReturn()after()afterThrowing()这些方法都是通知。

 

下面我们就来看具体配置,首先来看一下:

<1>.基于xml配置文件的AOP实现:这种方式在实现AOP时,有4个步骤。

 

<?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"        xsi:schemaLocation="            http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd>    <bean id="registerDaoImpl" class="com.zxf.dao.RegisterDaoImpl"/>    <bean id="registerService" class="com.zxf.service.RegisterServiceImpl">        <property name=" registerDaoImpl " ref=" RegisterDaoImpl "/>    </bean>    <!-- 日志切面类 -->    <bean id="logAspectBean" class="com.zxf.aspect.LogAspect"/>    <!-- 第1步: AOP的配置 -->    <aop:config>        <!-- 第2步:配置一个切面 -->        <aop:aspect id="logAspect" ref="logAspectBean">            <!-- 第3步:定义切入点,指定切入点表达式 -->            <aop:pointcut id="allMethod"                 expression="execution(* com.zxf.service.*.*(..))"/>             <!-- 第4步:应用前置通知 -->            <aop:before method="before" pointcut-ref="allMethod" />            <!-- 第4步:应用后置通知 -->            <aop:after-returning method="afterReturn" pointcut-ref="allMethod"/>            <!-- 第4步:应用最终通知 -->            <aop:after method="after" pointcut-ref="allMethod"/>            <!-- 第4步:应用抛出异常后通知 -->            <aop:after-throwing method="afterThrowing" pointcut-ref="allMethod"/>            <!-- 第4步:应用环绕通知 -->            <!--             <aop:around method="doAround" pointcut-ref="allMethod" />             -->        </aop:aspect>    </aop:config></beans>

    上述配置针对切入点应用了前置、后置、最终,以及抛出异常后通知。这样在测试执行RegisterServiceImpl类的save()方法时,控制台会有如下结果输出:

 

前置通知:com.zxf.service.RegisterServiceImpl类的save方法开始了。

针对MySQL的RegisterDao实现中的save()方法。

后置通知:方法正常结束了。

最终通知:不管方法有没有正常执行完成,一定会返回的。

下面我们在来看一下第二种配置方式:

<2>基于注解的AOP的实现

 

     首先创建一个用来作为切面的类LogAnnotationAspect,同时把这个类配置在spring的配置文件中。

        在spring2.0以后引入了JDK5.0的注解Annotation的支持,提供了对AspectJ基于注解的切面的支持,从而 更进一步地简化AOP的配置。具体的步骤有两步。

 

Spring的配置文件是如下的配置:

 

<?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"        xsi:schemaLocation="            http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd>    <bean id="registerDao" class="com.zxf.dao.RegisterDaoImpl"/>    <bean id="registerService" class="com.zxf.service.RegisterServiceImpl">        <property name="registerDao" ref="registerDao"/>    </bean>    <!-- 把切面类交由Spring容器来管理 -->    <bean id="logAspectBean" class="com.zxf.aspect.LogAnnotationAspect"/>    <!-- 启用spring对AspectJ注解的支持 -->    <aop:aspectj-autoproxy/></beans>

 

这是那个切面的类LogAnnotationAspect

/** * 日志切面类 */@Aspect  //定义切面类public class LogAnnotationAspect {    @SuppressWarnings("unused")    //定义切入点,提供一个方法,这个方法的名字就是改切入点的id    @Pointcut("execution(* com.zxf.service.*.*(..))")    private void allMethod(){}    //针对指定的切入点表达式选择的切入点应用前置通知    @Before("execution(* com. zxf.service.*.*(..))")    public void before(JoinPoint call) {        String className = call.getTarget().getClass().getName();        String methodName = call.getSignature().getName();        System.out.println("【注解-前置通知】:" + className + "类的"                 + methodName + "方法开始了");    }    //访问命名切入点来应用后置通知    @AfterReturning("allMethod()")    public void afterReturn() {        System.out.println("【注解-后置通知】:方法正常结束了");    }    //应用最终通知    @After("allMethod()")    public void after(){        System.out.println("【注解-最终通知】:不管方法有没有正常执行完成,"                 + "一定会返回的");    }    //应用异常抛出后通知    @AfterThrowing("allMethod()")    public void afterThrowing() {        System.out.println("【注解-异常抛出后通知】:方法执行时出异常了");    }    //应用周围通知    //@Around("allMethod()")    public Object doAround(ProceedingJoinPoint call) throws Throwable{        Object result = null;        this.before(call);//相当于前置通知        try {            result = call.proceed();            this.afterReturn(); //相当于后置通知        } catch (Throwable e) {            this.afterThrowing();  //相当于异常抛出后通知            throw e;        }finally{            this.after();  //相当于最终通知        }        return result;    }}

  备注:输出结果和前面的一样。


           

给我老师的人工智能教程打call!http://blog.csdn.net/jiangjunshow

这里写图片描述
你好! 这是你第一次使用 **Markdown编辑器** 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章,了解一下Markdown的基本语法知识。

新的改变

我们对Markdown编辑器进行了一些功能拓展与语法支持,除了标准的Markdown编辑器功能,我们增加了如下几点新功能,帮助你用它写博客:

  1. 全新的界面设计 ,将会带来全新的写作体验;
  2. 在创作中心设置你喜爱的代码高亮样式,Markdown 将代码片显示选择的高亮样式 进行展示;
  3. 增加了 图片拖拽 功能,你可以将本地的图片直接拖拽到编辑区域直接展示;
  4. 全新的 KaTeX数学公式 语法;
  5. 增加了支持甘特图的mermaid语法1 功能;
  6. 增加了 多屏幕编辑 Markdown文章功能;
  7. 增加了 焦点写作模式、预览模式、简洁写作模式、左右区域同步滚轮设置 等功能,功能按钮位于编辑区域与预览区域中间;
  8. 增加了 检查列表 功能。

功能快捷键

撤销:Ctrl/Command + Z
重做:Ctrl/Command + Y
加粗:Ctrl/Command + B
斜体:Ctrl/Command + I
标题:Ctrl/Command + Shift + H
无序列表:Ctrl/Command + Shift + U
有序列表:Ctrl/Command + Shift + O
检查列表:Ctrl/Command + Shift + C
插入代码:Ctrl/Command + Shift + K
插入链接:Ctrl/Command + Shift + L
插入图片:Ctrl/Command + Shift + G

合理的创建标题,有助于目录的生成

直接输入1次#,并按下space后,将生成1级标题。
输入2次#,并按下space后,将生成2级标题。
以此类推,我们支持6级标题。有助于使用TOC语法后生成一个完美的目录。

如何改变文本的样式

强调文本 强调文本

加粗文本 加粗文本

标记文本

删除文本

引用文本

H2O is是液体。

210 运算结果是 1024.

插入链接与图片

链接: link.

图片: Alt

带尺寸的图片: Alt

当然,我们为了让用户更加便捷,我们增加了图片拖拽功能。

如何插入一段漂亮的代码片

博客设置页面,选择一款你喜欢的代码片高亮样式,下面展示同样高亮的 代码片.

// An highlighted block var foo = 'bar'; 

生成一个适合你的列表

  • 项目
    • 项目
      • 项目
  1. 项目1
  2. 项目2
  3. 项目3
  • 计划任务
  • 完成任务

创建一个表格

一个简单的表格是这么创建的:

项目 Value
电脑 $1600
手机 $12
导管 $1

设定内容居中、居左、居右

使用:---------:居中
使用:----------居左
使用----------:居右

第一列 第二列 第三列
第一列文本居中 第二列文本居右 第三列文本居左

SmartyPants

SmartyPants将ASCII标点字符转换为“智能”印刷标点HTML实体。例如:

TYPE ASCII HTML
Single backticks 'Isn't this fun?' ‘Isn’t this fun?’
Quotes "Isn't this fun?" “Isn’t this fun?”
Dashes -- is en-dash, --- is em-dash – is en-dash, — is em-dash

创建一个自定义列表

Markdown
Text-to- HTML conversion tool
Authors
John
Luke

如何创建一个注脚

一个具有注脚的文本。2

注释也是必不可少的

Markdown将文本转换为 HTML

KaTeX数学公式

您可以使用渲染LaTeX数学表达式 KaTeX:

Gamma公式展示 Γ ( n ) = ( n 1 ) ! n N \Gamma(n) = (n-1)!\quad\forall n\in\mathbb N 是通过欧拉积分

Γ ( z ) = 0 t z 1 e t d t &ThinSpace; . \Gamma(z) = \int_0^\infty t^{z-1}e^{-t}dt\,.

你可以找到更多关于的信息 LaTeX 数学表达式here.

新的甘特图功能,丰富你的文章

gantt
        dateFormat  YYYY-MM-DD
        title Adding GANTT diagram functionality to mermaid
        section 现有任务
        已完成               :done,    des1, 2014-01-06,2014-01-08
        进行中               :active,  des2, 2014-01-09, 3d
        计划一               :         des3, after des2, 5d
        计划二               :         des4, after des3, 5d
  • 关于 甘特图 语法,参考 这儿,

UML 图表

可以使用UML图表进行渲染。 Mermaid. 例如下面产生的一个序列图::

张三 李四 王五 你好!李四, 最近怎么样? 你最近怎么样,王五? 我很好,谢谢! 我很好,谢谢! 李四想了很长时间, 文字太长了 不适合放在一行. 打量着王五... 很好... 王五, 你怎么样? 张三 李四 王五

这将产生一个流程图。:

链接
长方形
圆角长方形
菱形
  • 关于 Mermaid 语法,参考 这儿,

FLowchart流程图

我们依旧会支持flowchart的流程图:

  • 关于 Flowchart流程图 语法,参考 这儿.

导出与导入

导出

如果你想尝试使用此编辑器, 你可以在此篇文章任意编辑。当你完成了一篇文章的写作, 在上方工具栏找到 文章导出 ,生成一个.md文件或者.html文件进行本地保存。

导入

如果你想加载一篇你写过的.md文件或者.html文件,在上方工具栏可以选择导入功能进行对应扩展名的文件导入,
继续你的创作。


  1. mermaid语法说明 ↩︎

  2. 注脚的解释 ↩︎

猜你喜欢

转载自blog.csdn.net/uffvhj/article/details/83892994