Spring AOP应用实例demo

AOP(Aspect-Oriented Programming,面向方面编程),可以说是OOP(Object-OrientedPrograming,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。

OOP的问题,AOP的补充

当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。

Spring中对 AOP的支持

Spring中 AOP代理由Spring的
IoC容器负责生成、管理,其依赖关系也由 IoC容器负责管理。因此,AOP代理可以直接使用容器中的其他 Bean实例作为目标,这种关系可由
IoC容器的依赖注入提供。Spring默认使用 Java动态代理来创建AOP代理,这样就可以为任何接口实例创建代理了。当需要代理的类不是代理接口的时候,
Spring自动会切换为使用 CGLIB代理,也可强制使用 CGLIB。

程序员参与部分

AOP编程其实是很简单的事情。纵观 AOP编程,其中需要程序员参与的只有三个部分:

定义普通业务组件。

定义切入点,一个切入点可能横切多个业务组件。

定义增强处理,增强处理就是在AOP框架为普通业务组件织入的处理动作。

所以进行 AOP编程的关键就是定义切入点和定义增强处理。一旦定义了合适的切入点和增强处理,AOP框架将会自动生成AOP代理,即:代理对象的方法
=增强处理 +被代理对象的方法。

Spring中使用方式:

基于 Annotation的“零配置”方式。

(1)启动注解,配置文件applicationContext.xml

<!-- 启动对@AspectJ注解的支持 -->
<aop:aspectj-autoproxy/>

<bean id="user" class="com.tgb.spring.aop.IUserImpl"/>

<bean id="check" class="com.tgb.spring.aop.CheckUser"/>

(2)编写切面类

package com.tgb.spring.aop;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

@Aspect
public class CheckUser {

	@Pointcut("execution(* com.tgb.spring.aop.*.find*(..))")
	public void checkUser(){
		System.out.println("**************The System is Searching Information For You****************");
	}

	@Pointcut("execution(* com.tgb.spring.aop.*.add*(..))")
	public void checkAdd(){
		System.out.println("**************<< Add User >> Checking.....***************");
	}

	@Before("checkUser()")
	public void beforeCheck(){
		System.out.println(">>>>>>>> 准备搜查用户..........");
	}

	@After("checkUser()")
	public void afterCheck(){
		System.out.println(">>>>>>>> 搜查用户完毕..........");
	}

	@Before("checkAdd()")
	public void beforeAdd(){
		System.out.println(">>>>>>>> 增加用户--检查ing..........");
	}

	@After("checkAdd()")
	public void afterAdd(){
		System.out.println(">>>>>>>> 增加用户--检查完毕!未发现异常!..........");
	}

	 //声明环绕通知
    @Around("checkUser()")
    public Object doAround(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("进入方法---环绕通知");
        Object o = pjp.proceed();
        System.out.println("退出方法---环绕通知");
        return o;
    }
}

(3)定义接口

package com.tgb.spring.aop;

public interface IUser {

	public String findUser(String username);
	public void addUser(String username);
	public void findAll();
}

(4)定义实现

package com.tgb.spring.aop;

import java.util.HashMap;
import java.util.Map;

public class IUserImpl implements IUser {

	public static Map map = null;
	public static void init(){
		String[] list = {"Lucy", "Tom", "小明", "Smith", "Hello"};
		Map tmp = new HashMap();
		for(int i=0; i<list.length; i++){
			tmp.put(list[i], list[i]+"00");
		}
		map = tmp;
	}
	public void addUser(String username) {
		init();
		map.put(username, username+"11");
		System.out.println("--------------【addUser】: "+username+" --------------");
		System.out.println("【The new List:"+map+"】");
	}

	public void findAll() {
		init();
		System.out.println("---------------【findAll】: "+map+" ------------------");
	}

	public String findUser(String username) {
		init();
		String password = "没查到该用户";
		if(map.containsKey(username)){
			password = map.get(username).toString();
		}
		System.out.println("-----------------【findUser】-----------------");
		System.out.println("-----------------Username:"+username+"-----------------");
		System.out.println("-----------------【Result】:"+password+"------------------");
		return password;

	}

}

(5)测试

public class Test {

	public static void main(String as[]){
		BeanFactory factory = new ClassPathXmlApplicationContext("applicationContext.xml");
		IUser user = (IUser) factory.getBean("user");
		user.findAll();

		User u = new User();
//		u.setUsername("Tom");
//		user.findUser(u.getUsername());

		/*u.setUsername("haha");
		user.addUser(u.getUsername());*/
	}
}

执行结果:

进入方法---环绕通知

>>>>>>>>准备搜查用户..........

---------------【findAll】: {Smith=Smith00, Tom=Tom00, 小明=小明00, Lucy=Lucy00,Hello=Hello00} ------------------

退出方法---环绕通知

>>>>>>>> 搜查用户完毕..........

注:@Before是在所拦截方法执行之前执行一段逻辑。@After是在所拦截方法执行之后执行一段逻辑。@Around是可以同时在所拦截方法的前后执行一段逻辑。

以上是针对注解的方式来实现,那么配置文件也一样,只需要在applicationContext.xml中添加如下代码:

<!--  <aop:config>
		<aop:pointcut id="find" expression="execution(* com.tgb.spring.aop.*.find*(..))" />
		<aop:pointcut id="add" 	expression="execution(* com.tgb.spring.aop.*.add*(..))" />

		<aop:aspect id="checkUser" ref="check">
			<aop:before method="beforeCheck" pointcut-ref="find"/>
			<aop:after method="afterCheck" pointcut-ref="find"/>
		</aop:aspect>

		<aop:aspect id="checkAdd" ref="check">
			<aop:before method="beforeAdd" pointcut-ref="add"/>
			<aop:after method="afterAdd" pointcut-ref="add"/>
		</aop:aspect>

	</aop:config>-->

总结:

以上是简单介绍了一下如何使用Spring AOP,在使用的过程中也加深我们对AOP思想的理解,其实AOP就是要我们实现热插拔效果,下篇会继续介绍Spring
AOP的实现原理。

附:在使用过程中注意JDK版本与易用的aspectJrt的版本问题

时间: 2024-12-23 21:00:58

Spring AOP应用实例demo的相关文章

Spring aop 小实例demo

Hadoop从2.4.0版本开始支持hdfs的ACL,在CDH5.0当中也集成了该特性,下面对其进行一些测试: unnamed user (file owner) 文件的拥有者 unnamed group (file group) 文件的所属组 named user 除了文件的拥有者和拥有组之外,的其它用户 named group 除了文件的拥有者和拥有组之外,的其它用户 mask  权限掩码,用于过滤named user和named group的权限 一.启用ACL: <property>

SSH框架系列:Spring AOP应用记录日志Demo

分类: [java]2013-12-10 18:53 724人阅读 评论(0) 收藏 举报 1.简介 Spring 中的AOP为Aspect Oriented Programming的缩写,面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.以下是Spring AOP的小例子 源代码:https://github.com/nuptboyzhb/SpringAOPDemo 2.例子简介 2.1切面aspect:Logging.java [java] view plainc

Spring AOP通知实例 – Advice

Spring AOP(面向方面编程)框架,用于在模块化方面的横切关注点.简单得说,它只是一个拦截器拦截一些过程,例如,当一个方法执行,Spring AOP 可以劫持一个执行的方法,在方法执行之前或之后添加额外的功能. 在Spring AOP中,有 4 种类型通知(advices)的支持: 通知(Advice)之前 - 该方法执行前运行 通知(Advice)返回之后 – 运行后,该方法返回一个结果 通知(Advice)抛出之后 – 运行方法抛出异常后, 环绕通知 – 环绕方法执行运行,结合以上这三

Spring aop 小例子demo

由于最近的服务项目提供接口有一个需求,所有操作都必须检查操作的服务可用,所以感觉Aop特别适合实施.完成学习的小例子. 关于spring-Aop原理:http://m.oschina.net/blog/174838这篇文章写的非常好. 个人觉着可能上线的时候配置文件更方便一下.所以样例主要是配置文件方式 Demo文件下载地址: http://download.csdn.net/detail/ruishenh/7261121 Spring配置文件 /idle-service-impl/src/ma

spring aop 的一个demo(未完,待完善)

假设我们有这样的一个场景 : 对于一个类的众多方法,有些方法需要从缓存读取数据,有些则需要直接从数据库读取数据.怎样实现呢? 实现方案有多种.下面我说下常见的几种实现方案 : 1.直接采用spring xml.或者  annotation AOP完成.但个人认为这种方案似乎有点不是很完美. 原因 :  ①.如果只有针对这个类做切面拦截,这种方案是没有问题的,只需对需要走DB(or 缓存,两者择一)的方法配置切面. ②.那如果是多个类呢?统一做一个切面,对指定方法拦截,如selectXXX.但,还

[Spring] AOP, Aspect实例解析

最近要用到切面来统一处理日志记录,写了个小实例练了练手: 具体实现类: public interface PersonServer { public void save(String name); public void update(String name, Integer id); public String getPersonName(Integer id); } import org.springframework.stereotype.Component; @Component("pe

Spring Aop编程的demo

1: 新建一个普通的bean :Role 属性 Id,name,添加无参构造,setter getter方法 2:新建一个接口:RoleService,随便写一个方法printRole 3:新建一个类RoleServiceImpl,实现RoleService接口,重写printRole方法 注意 @Component 注解别忘了 4:定义切面类 RoleAspect 添加四个通知方法 注意: 添加@Aspect注解 execution中的参数一定要写对,例:"execution(* aop.se

Spring AOP编程实例

整个类包如下: 一.具体各个类 1.1前置通知类 package com.yuan.aop; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; public class MymethodBeforeAdvice implements MethodBeforeAdvice { @Override public void before(Method arg0, Object[] ar

spring AOP 环绕增强小Demo

前面写了一个前置增强,后置增强的小demo,前置增强即在方法调用前对方法增强:后置增强即在方法调用后对方法增强.环绕增强允许在目标类方法调用前后织入横切逻辑,它综合了前置.后置增强两者的功能. 还继续沿用之前的代码,这里介绍环绕增强的实现类和测试类. 环绕增强类 GreetingAroundAdvice.java package com.paic.zhangqi.spring.aop; import org.aopalliance.intercept.MethodInterceptor; imp