`
IT阿狸
  • 浏览: 67915 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

Spring-AOP(前置/后置/环绕)通知的例子

阅读更多

一、包结构

 

二、biz接口

package org.aop.biz;

/**
 * 在线图书销售系统业务逻辑接口
 * 
 * @author miao
 */
public interface BookBiz {

	/**
	 * 买书的业务逻辑
	 * 
	 * @param userName
	 * @param bookName
	 * @param price
	 * @return
	 */
	public boolean buy(String userName, String bookName, double price);

	/**
	 * 发表书评的业务逻辑
	 * 
	 * @param userName
	 * @param comments
	 */
	public void comment(String userName, String comments);

}

 

 

三、bizImpl实现类

package org.aop.biz.impl;

import org.aop.biz.BookBiz;

/**
 * 在线图书销售系统业务逻辑接口实现类
 * 
 * @author miao
 * 
 */
public class BookBizImpl implements BookBiz {

	/**
	 * 购买图书
	 */
	@Override
	public boolean buy(String userName, String bookName, double price) {
		System.out.println("------------------------");
		System.out.println("·" + userName + "购买图书:" + bookName);
		System.out.println("·" + userName + "增加积分:" + (int) (price / 10));
		System.out.println("------------------------");
		return true;
	}

	/**
	 * 发表书评
	 */
	@Override
	public void comment(String userName, String comments) {
		System.out.println("------------------------");
		System.out.println("·" + userName + "发表书评:" + comments);
		System.out.println("------------------------");

	}

}

 

 

四、前置/后置/环绕通知

前置:

package org.aop.log;

import java.lang.reflect.Method;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Date;
import org.springframework.aop.MethodBeforeAdvice;

/**
 * 前置通知 输出每个方法的参数,调用的时间
 * 
 * @author miao
 * 
 */
public class LogAdvice implements MethodBeforeAdvice {

	private DateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 hh时mm分ss秒sss毫秒");

	/**
	 * @param m 方法本身
	 * @param args 方法的参数
	 * @param target 调用此方法的对象
	 */
	public void before(Method m, Object[] args, Object target) throws Throwable {
		System.out.println("\n[系统日志][" + sdf.format(new Date()) + "[调用方法:" + m.getName() + "(参数是:"
				+ Arrays.toString(args) + ")调用的对象是:" + target);
	}

}

 

 

后置:

package org.aop.log;

import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.springframework.aop.AfterReturningAdvice;

/**
 * 后置通知
 * 
 * @author miao
 * 
 */
public class RakeOffAdvice implements AfterReturningAdvice {

	/**
	 * @param returnValue 返回值
	 * @param method 方法本身
	 * @param args 参数数组
	 * @param target 对象本身
	 */
	public void afterReturning(Object returnValue, Method method, Object[] args, Object target)
			throws Throwable {
		// 只有buy方法才有返利
		if (method.getName().equals("buy")) {
			System.out.println("[销售返利]["
					+ new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()) + "]"
					+ args[0] + ":返利5元哦亲!");
		}
	}

}

 

 

环绕:

package org.aop.log;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

/**
 * 环绕通知实例
 * 
 * @author miao
 * 
 */
public class RoundAdvice implements MethodInterceptor {

	/**
	 * MethodInterceptor不但封装目标方法及其入参数组,还封装了目标方法所在的实例对象
	 * 通过getArguments()可以获取目标犯法的入参数组,通过proceed()反射调用目标实例相应的方法
	 */
	public Object invoke(MethodInvocation invocation) throws Throwable {
		if ("buy".equals(invocation.getMethod().getName())) {
			Object[] args = invocation.getArguments();// 目标方法入参
			String userName = (String) args[0];
			System.out.println("你好,欢迎光临!" + userName);// 运行方法前调用
			boolean obj = (Boolean) invocation.proceed();// 调用方法,obj方法的返回值
			if (obj) {
				System.out.println("谢谢惠顾!");// 目标方法之后运行
			} else {
				System.out.println("购买失败!");// 目标方法之后运行
			}
			return obj;
		}
		return null;
	}

}

 

 

五、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:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

	<!-- 真正的业务实现类 -->
	<bean id="bookBizTarget" class="org.aop.biz.impl.BookBizImpl"/>
	
	<!-- 前置通知类 -->
	<bean id="logAdvice" class="org.aop.log.LogAdvice"/>
	
	<!-- 代理类,开始织入 -->
	<bean id="bookBiz" class="org.springframework.aop.framework.ProxyFactoryBean">
		<!-- 相应的业务接口 -->
		<property name="proxyInterfaces">
			<list>
				<value>org.aop.biz.BookBiz</value>
			</list>
		</property>
		<!-- 切面类 -->
		<property name="interceptorNames">
			<list>
				<value>logAdvice</value>
			</list>
		</property>
		<!-- 代理的实现类 -->
		<property name="targetName">
			<value>bookBizTarget</value>
		</property>
	</bean>
	

</beans>

 

 

后置:

<?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:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

	<!-- 后置通知类 -->
	<bean id="rakeOffAdvice" class="org.aop.log.RakeOffAdvice" />
	
	<!-- 书籍的业务逻辑实现类 -->
	<bean id="bookBiz" class="org.aop.biz.impl.BookBizImpl"/>

	<!-- 通过ID自动代理工厂类 -->
	<bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
		<property name="beanNames">
			<value>*Biz</value>
		</property>
		<property name="interceptorNames">
			<list>
				<value>rakeOffAdvice</value>
			</list>
		</property>
	</bean>
	
</beans>

 

 

环绕:

<?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:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

	<!-- 真正的业务实现类 -->
	<bean id="bookBizTarget" class="org.aop.biz.impl.BookBizImpl" />

	<!-- 前置通知类 -->
	<bean id="roundAdvice" class="org.aop.log.RoundAdvice" />

	<!-- 代理类,开始织入 -->
	<bean id="bookBiz" class="org.springframework.aop.framework.ProxyFactoryBean">
		<!-- 相应的业务接口 -->
		<property name="proxyInterfaces">
			<list>
				<value>org.aop.biz.BookBiz</value>
			</list>
		</property>
		<!-- 切面类 -->
		<property name="interceptorNames">
			<list>
				<value>roundAdvice</value>
			</list>
		</property>
		<!-- 代理的实现类 -->
		<property name="target" ref="bookBizTarget" />
	</bean>

</beans>

 

 

七、测试类:

package org.aop.test;

import org.aop.biz.BookBiz;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * 测试类
 * 
 * @author miao
 * 
 */
public class AopTest {

	/**
	 * 测试类,调用代理的工厂类
	 * 
	 * @param args
	 */
	public static void main(String[] args) {
		ApplicationContext context = new ClassPathXmlApplicationContext("aopBefore.xml");
		//ApplicationContext context = new ClassPathXmlApplicationContext("aopRound.xml");
		//ApplicationContext context = new ClassPathXmlApplicationContext("aopAfter.xml");
		// 将代理工厂作为业务接口的子对象
		BookBiz bookBiz = (BookBiz) context.getBean("bookBiz");
		// 直接调用接口方法
		bookBiz.buy("小伙伴", "暴走漫画", 100);
		bookBiz.comment("王蜜桃", "不要在意这些细节");
	}
}

 

 

八、demo

 Spring-AOP.zip

  • 大小: 13.2 KB
分享到:
评论

相关推荐

    spring-aop.xsd

    2. 通知(Advice):在特定连接点上执行的代码,如前置通知、后置通知、环绕通知等。 3. 连接点(Join Point):程序执行过程中能够插入通知的一个点,通常是方法的执行。 4. 切入点(Pointcut):定义一组连接点的...

    spring的Aop中的前置通知,后置通知以及环绕通知简单代码

    在Spring AOP中,有三种主要的通知类型:前置通知、后置通知和环绕通知。下面将详细解释这三种通知,并通过简单的代码示例进行演示。 **1. 前置通知(Before Advice)** 前置通知在目标方法被调用之前执行,但无法...

    spring-aop.jar

    3. **通知(Advice)**:通知是切面中定义的行为,包括前置通知、后置通知、异常通知、环绕通知等。Spring AOP提供了多种通知实现,如`org.springframework.aop.interceptor.AbstractAsyncTimingInterceptor`用于...

    spring-aop-4.0.0.RELEASE

    - **通知(Advice)**:在特定的连接点(join point)执行的代码,例如:前置通知(before)、后置通知(after)、环绕通知(around)等。 - **切入点(Pointcut)**:定义何时执行通知的表达式,标识程序执行的...

    spring-aop实例demo

    Spring支持五种不同类型的的通知:前置通知(Before)、后置通知(After)、返回后通知(After Returning)、异常后通知(After Throwing)和环绕通知(Around)。 在Spring 3.2.8中实现AOP,我们需要以下步骤: 1...

    spring-aop

    Spring支持五种不同类型的通知:前置通知(Before)、后置通知(After)、返回后通知(After Returning)、异常后通知(After Throwing)和环绕通知(Around)。 3. 连接点(Join Point):连接点是在程序执行过程...

    spring aop注解方式、xml方式示例

    通知是在特定连接点(join point)执行的代码,可以是前置通知、后置通知、异常通知、最终通知或环绕通知。使用`@Before`、`@After`、`@AfterReturning`、`@AfterThrowing`和`@Around`注解定义不同的通知,例如: `...

    spring-boot示例项目

    aop|[aop,正则,前置通知,后置通知,环绕通知](https://github.com/smltq/spring-boot-demo/blob/master/aop/HELP.md) data-redis|[lettuce,redis,session redis,YAML配置,连接池,对象存储]...

    Spring-AOP demo

    1. **注解驱动的AOP**:Spring支持使用注解来定义切面,如`@Aspect`用于声明一个类为切面,`@Before`、`@After`、`@Around`、`@AfterReturning`和`@AfterThrowing`分别定义前置通知、后置通知、环绕通知、返回后通知...

    spring-aop demo及junit测试

    1. **基于XML的配置**:这是Spring AOP的传统配置方式,需要在Spring的配置文件中声明切入点表达式、定义通知类型(前置、后置、环绕等)以及组装切点和通知。以下是一个简单的示例: ```xml &lt;aop:config&gt; &lt;!-- ...

    Spring AOP运用Spring AOP技术,要求包含前置通知、后置通知、环绕通知、返回通知、异常返回通知。

    1、编写切面类,包含权限审核方法和日志记录方法,这两个方法将来会织入到...5、编写案例,运用Spring AOP技术,要求包含前置通知、后置通知、环绕通知、返回通知、异常返回通知。请掌握这五种通知的特点,及应用场景

    spring-aop-aspectj(Schema)-case

    4. `&lt;aop:before&gt;`、`&lt;aop:after&gt;`、`&lt;aop:around&gt;`等:定义不同类型的通知,如前置通知、后置通知和环绕通知,它们定义了在特定方法执行前后执行的代码。 学习这个案例,你将了解到如何使用XML Schema配置实现...

    spring-aop实例

    2. **通知(Advice)**:通知是在特定的连接点(join point)执行的代码,它可以是前置通知(before advice)、后置通知(after advice)、返回通知(return advice)、异常通知(throw advice)以及环绕通知...

    spring-aop源码

    Spring支持五种不同类型的通知:前置通知(Before)、后置通知(After)、返回后通知(After Returning)、异常后通知(After Throwing)和环绕通知(Around)。这些通知通过`org.springframework.aop....

    spring对AOP的支持(使用Spring的配置文件来演示)

    3. **通知(Advice)**:在特定连接点执行的动作,例如环绕通知、前置通知、后置通知、异常通知等。 4. **切入点(Pointcut)**:定义切面将在哪些连接点处应用的通知。通常使用表达式来匹配连接点。 5. **织入...

Global site tag (gtag.js) - Google Analytics