一、理论基础:

AOP(Aspectoriented programming)面向切片/服务的编程,在Spring中使用最多的是对事物的处理。而AOP这种思想在程序中很多地方可以使用的,比如说,对某些规则的验证,可以抽象封装到一个模块中,并在该模块中定义一定的使用规则,然后植入到原有的程序中,其实这就是面向切片。这个模块叫做Aspect,定义的规则是pointcut,具体的验证的实现是advice,植入的目标叫TargetObject,切入到目标中的方法叫做joinpint,植入的过程叫weave。

Advice在模块中关注点的具体位置有:前、后或者抛出   Before、After、Throw 三种:

Before Advice

代用之前的处理

After Advice

调用之后的处理

Throw Advice

调用的时候抛出的异常处理

二、实践:

有了理论基础后,重点还是在程序中使用。下面的实例使用注解和配置两种方式,实现对添加用户的方法的验证服务的切入。

1、Anotation注解的方式:

1.首先定义业务逻辑接口UserManager类的方法:

public interface UserManager {

	public void addUser(String username, String password);

	public void delUser(int userId);

	public String findUserById(int userId);

	public void modifyUser(int userId, String username, String password);
}

2.UserManagerImpl类定义接口的实现:

public class UserManagerImpl implements UserManager {

	public void addUser(String username, String password) {
System.out.println("---------UserManagerImpl.add()--------");
} public void delUser(int userId) {
System.out.println("---------UserManagerImpl.delUser()--------");
} public String findUserById(int userId) {
System.out.println("---------UserManagerImpl.findUserById()--------");
return "张三";
} public void modifyUser(int userId, String username, String password) { System.out.println("---------UserManagerImpl.modifyUser()--------");
}
}

3.SecurityHandler定义要切入的方法:

@Aspect
public class SecurityHandler { /**
* 定义Pointcut,Pointcut的名称为addAddMethod(),此方法没有返回值和参数
* 该方法就是一个标识,不进行调用
*/
@Pointcut("execution(* add*(..))")
private void addAddMethod(){}; /**
* 定义Advice,表示我们的Advice应用到哪些Pointcut订阅的Joinpoint上
*/
@Before("addAddMethod()")
//@After("addAddMethod()")
private void checkSecurity() {
System.out.println("-------checkSecurity-------");
}
}

Anotation注解规则:

@Aspect表示注解的类是抽象的服务方法模块;

@Pointcut定义服务的使用规则,也就是服务方法要应用到目标类中哪些方法上。

@Pointcut("execution(*add*(..))") 第一个*表示不论是否有返回值,所有以add开头的方法,不管是否有参数。

private voidaddAddMethod(){};该方法只是注解模式中一个空的方法体,是一个模式化的方法定义结构,该方法不能有返回值,不能有任何参数,也不能对其进行实现。在Advice中要使用该方法名的标识。

Advice注解checkSecurity()方法,表示该方法是具体的服务方法,使用注解的方式,Advice不出现,而是使用上面理论部分提到的使用@After、@Befor、@Throw来表示,同时要在Advice中关联pointcut中定义的规则。

4.Client客户端调用:

public class Client {

	public static void main(String[] args) {
BeanFactory factory = new ClassPathXmlApplicationContext("applicationContext.xml");
UserManager userManager = (UserManager)factory.getBean("userManager");
userManager.addUser("张三", "123");
} }

AspecJ是通过applicationContext.xml文件中的配置,依据SecurityHandler类中的Anotation规则,将服务方法切入到UserManagerImpl类中的方法中的,所以,在客户端使用的时候要通过工厂取得UserManager对象。

5.applicationContext.xml文件中注入AspectJ的AOP服务:

  <!-- 启用AspectJ对Annotation的支持 -->
<aop:aspectj-autoproxy/> <bean id="userManager" class="com.xxx.spring.UserManagerImpl"/> <bean id="securityHandler" class="com.xxx.spring.SecurityHandler"/>


2、配置文件的方式:

注解的方式灵活性不够,不便于业务的修改,下面是使用配置文件的方式:

接口和实现接口类的方法以及在客户端的调用都是一样的,这里略。

1.SecurityHandler类像其他普通类一样,定义服务的方法即可:

public class SecurityHandler {

	private void checkSecurity() {
System.out.println("-------checkSecurity-------");
}
}

2.重点是applicationContext.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-2.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
<bean id="userManager" class="com.bjpowernode.spring.UserManagerImpl"/>
<bean id="securityHandler" class="com.bjpowernode.spring.SecurityHandler"/> <aop:config>
<aop:aspect id="securityAspect" ref="securityHandler">
<!--
以add开头的方法
<aop:pointcut id="addAddMethod" expression="execution(* add*(..))"/>
-->
<!--
com.bjpowernode.spring包下所有的类所有的方法
<aop:pointcut id="addAddMethod" expression="execution(* com.bjpowernode.spring.*.*(..))"/>
-->
<aop:pointcut id="addAddMethod" expression="execution(* com.bjpowernode.spring.*.add*(..)) || execution(* com.bjpowernode.spring.*.del*(..))"/>
<aop:before method="checkSecurity" pointcut-ref="addAddMethod"/>
</aop:aspect>
</aop:config>
</beans>

注意到:

1.      Pointcut的匹配规则很灵活,可以使用|| 也可以指定包下的类。

2.      采用配置文件的方法是将服务模块Aspect通过DI注入的方式

3.      配置文件的方式不同于注解的方式,不需要在配置文件中显式的启用AspectJ对Annotation的支持。

3、参数的传递:

在Advice的方法中使用JoinPoint对象可以取得客户端传入的参数值和调用的方法名:

public class SecurityHandler {

	private void checkSecurity(JoinPoint joinPoint) {
for (int i=0; i<joinPoint.getArgs().length; i++) {
System.out.println(joinPoint.getArgs()[i]);
} System.out.println(joinPoint.getSignature().getName()); System.out.println("-------checkSecurity-------");
}
}

4、使用CGLIB的方式:

如果UserManagerImpl不是实现UserManager接口,那么就需要使用CGLIB的方式,这时候工厂返回的不是UserManager接口对象。

1.UserManagerImpl类:

public class UserManagerImpl {

	public void addUser(String username, String password) {

		System.out.println("---------UserManagerImpl.add()--------");
} public void delUser(int userId) { System.out.println("---------UserManagerImpl.delUser()--------");
} public String findUserById(int userId) { System.out.println("---------UserManagerImpl.findUserById()--------");
return "张三";
} public void modifyUser(int userId, String username, String password) { System.out.println("---------UserManagerImpl.modifyUser()--------");
} }

2.客户端Client:修改返回值

public class Client {

	public static void main(String[] args) {
BeanFactory factory = new ClassPathXmlApplicationContext("applicationContext.xml"); //UserManager userManager = (UserManager)factory.getBean("userManager");
UserManagerImpl userManager = (UserManagerImpl)factory.getBean("userManager"); userManager.addUser("张三", "123"); } }


这里需要注意的是:

如果UserManagerImpl实现了UserManager接口,那么默认情况下,AspectJ采用jdk的动态代理,返回的是UserManager接口的动态代理对象;

如果UserManagerImpl不是实现了UserManager接口这种方式,那么,AspectJ就会使用CGLIB的方式返回UserManagerImpl的一个CGLIB对象。

Jdk的动态代理和CGLIB实现机制的区别:

Jdk基于接口实现:JDK动态代理对实现了接口的类进行代理

CGLIB基于继承:CGLIB代理可以对类代理,主要对指定的类生成一个子类,因为是继承,所以,目标类最好不要使用final声明。

通常情况下,鼓励使用jdk代理,因为业务一般都会抽象出一个接口,而且不用引入新的东西。

如果是遗留的系统,以前没有实现接口,那么只能使用CGLIB。

三、对比动态代理的实现服务切入:

动态代理的实现:接口和实现的方法不变,这里代码略。

1.动态代理类SecurityHandler:

public class SecurityHandler implements InvocationHandler {

	private Object targetObject;

	public Object createProxyInstance(Object targetObject) {
this.targetObject = targetObject;
return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
targetObject.getClass().getInterfaces(),
this);
} public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
checkSecurity(); //调用目标方法
Object ret = method.invoke(targetObject, args); return ret;
} private void checkSecurity() {
System.out.println("-------checkSecurity-------");
}
}

2.客户端Client:

public class Client {

	public static void main(String[] args) {
SecurityHandler hander = new SecurityHandler();
UserManager useraManager = (UserManager)hander.createProxyInstance(new UserManagerImpl());
useraManager.addUser("张三", "123");
} }

对比Spring的IOC和AOP,动态代理有几点是不同的:

1.动态代理使用new的方式创建接口的代理服务类。

2.代理类中将需要添加将服务方法植入到目标类方法中的代码。

3.需要在客户端维护接口类的实现,不够灵活。

四、结尾感触:

Aop面向切片(服务)的编程时一种思想,在学习框架使用的时候,更应该学习这种设计思路,这是架构的基础,也是灵活架构的灵魂所在。

Spring之AOP面向切片的更多相关文章

  1. spring总结————AOP面向切面总结

    spring总结————AOP面向切面 一.spring aop概念 spring aop面向切面编程,java是面向对象的语言. 真正的service层代码 业务逻辑层再处理业务之前和之后都要进行一 ...

  2. Spring:AOP面向切面编程

    AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果. AOP是软件开发思想阶段性的产物,我们比较熟悉面向过程O ...

  3. Spring框架 AOP面向切面编程(转)

    一.前言 在以前的项目中,很少去关注spring aop的具体实现与理论,只是简单了解了一下什么是aop具体怎么用,看到了一篇博文写得还不错,就转载来学习一下,博文地址:http://www.cnbl ...

  4. Spring 08: AOP面向切面编程 + 手写AOP框架

    核心解读 AOP:Aspect Oriented Programming,面向切面编程 核心1:将公共的,通用的,重复的代码单独开发,在需要时反织回去 核心2:面向接口编程,即设置接口类型的变量,传入 ...

  5. 【spring源码学习】spring的AOP面向切面编程的实现解析

    一:Advice(通知)(1)定义在连接点做什么,为切面增强提供织入接口.在spring aop中主要描述围绕方法调用而注入的切面行为.(2)spring定义了几个时刻织入增强行为的接口  => ...

  6. Spring的AOP面向切面编程

    什么是AOP? 1.AOP概念介绍 所谓AOP,即Aspect orientied program,就是面向方面(切面)的编程. 功能: 让关注点代码与业务代码分离! 关注点: 重复代码就叫做关注点: ...

  7. spring:AOP面向切面编程02

    参考: https://blog.csdn.net/jeffleo/article/details/54136904 一.AOP的核心概念AOP(Aspect Oriented Programming ...

  8. Spring注解 - AOP 面向切面编程

    基本概念: AOP:Aspect Oriented Programming,即面向切面编程 指在程序运行期间动态的将某段代码切入到指定方法指定位置进行运行的编程方式 前置通知(@Before):在目标 ...

  9. Spring框架——AOP面向切面编程

    简介 AOP练习 使用动态代理解决问题 Spring AOP 用AspectJ注解声明切面 前置后置通知 利用方法签名编写AspectJ切入点表达式 指定切面的优先级 基于XML的配置声明切面 Spr ...

随机推荐

  1. Ant学习-002-ant 执行 TestNG 测试用例时 [testng] java.lang.NoClassDefFoundError: com/beust/jcommander/ParameterException 解决方案

    上篇文章中概述了 Ant windows 环境的基本配置,此文讲述在初次使用的过程中遇到的问题. 今天通过 ant 执行 TestNG 测试用例时,执行报错,相应的错误信息如下所示: Buildfil ...

  2. Java学习-013-文本文件读取实例源代码(两种数据返回格式)

    此文源码主要为应用 Java 读取文本文件内容实例的源代码.若有不足之处,敬请大神指正,不胜感激! 1.读取的文本文件内容以一维数组[LinkedList<String>]的形式返回,源代 ...

  3. notepad++代码折叠对应的树形结构快捷键

    树形层次,从1开始计数 <!doctype html> <html lang="en" class="1"> <head clas ...

  4. docker-compose常用命令

    --verbose:输出详细信息-f 制定一个非docker-compose.yml命名的yaml文件-p 设置一个项目名称(默认是directory名)docker-compose的动作包括:bui ...

  5. ionic 白名单

    1.在本目录下执行 cordova plugin add cordova-plugin-whitelist 2.在config.xml里添加 <allow-navigation href=&qu ...

  6. JBOSS的安全配置 .

    JBoss版本:JBoss 4.2.2.GA jboss默认配置了以下服务:JMX ConsoleJBoss Web Console为了安全起见,需要用户通过授权进行访问. 一.Java 鉴别与授权服 ...

  7. mysql 操作日期函数【增加,减少时间】

    详情链接:http://www.runoob.com/sql/func-date-add.html 定义和用法 DATE_ADD() 函数向日期添加指定的时间间隔. DATE_ADD(date,INT ...

  8. javascript设计模式学习之九——命令模式

    一.命令模式使用场景及定义 命令模式常见的使用场景是:有时候需要向某些对象发送请求,但是并不知道请求的接受者是谁,也不知道请求的具体操作是什么.此时希望用一种松耦合的方式来设计程序,使得请求的发送者和 ...

  9. respondsToSelector: selector

    -(BOOL) respondsToSelector: selector 用来判断是否有以某个名字命名的方法(被封装在一个selector的对象里传递)

  10. 为benchmarksql的PostgreSQL java驱动进行升级

    为benchmarksql的PostgreSQL java驱动进行升级[root@minion1 benchmarksql-4.1.0]# wget https://jdbc.postgresql.o ...