@annotation() | 限制匹配带有指定注解的连接点 |
各个类最终结构图如下:
1、Factory
package com.test.aspectj.expression;
/**
* 工厂接口
*/
public interface Factory {
// 制作产品
void make();
// 运输
void delivery(String address);
}
2、PhoneFactory
package com.test.aspectj.expression;
import com.test.aspectj.expression.annotation.Log;
import org.springframework.stereotype.Component;
/**
* 手机工厂
*/
@Component
public class PhoneFactory implements Factory {
// 制作产品的方法,注意有@Log注解
@Override
@Log
public void make() {
System.out.println("来自目标类PhoneFactory的消息:生产手机");
}
// 运输手机的方法
@Override
public void delivery(String address) {
System.out.println("来自目标类PhoneFactory的消息:运输手机至 " + address);
}
}
3、FoodFactory
package com.test.aspectj.expression;
import com.test.aspectj.expression.args.FreshFoodFactory;
import com.test.aspectj.expression.args.FrozenFoodFactory;
import org.springframework.stereotype.Component;
/**
* 食品工厂
*/
@Component
public class FoodFactory implements Factory {
// 制作产品的方法
@Override
public void make() {
System.out.println("来自目标类FoodFactory的消息:生产食品");
}
// 运输
@Override
public void delivery(String address) {
System.out.println("来自目标类FoodFactory的消息:销售食品至 " + address);
}
}
4、自定义注解 Log
package com.test.aspectj.expression.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* 自定义日志注解
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {
boolean value() default true;
}
5、切面类 AnnotationAspect,@annotation 代码示例
package com.test.aspectj.expression.annotation;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
/**
* 使用 @annotation() 来为所有加了 @Log 注解的方法织入增强
*/
@Aspect
public class AnnotationAspect {
@AfterReturning("@annotation(com.test.aspectj.expression.annotation.Log)")
public void log() {
System.out.println("来自切面里的消息:打印日志");
}
}
6、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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="com.test.aspectj.expression"/>
<bean id="annotationAspect" class="com.test.aspectj.expression.annotation.AnnotationAspect"/>
<aop:aspectj-autoproxy/>
</beans>
7、测试代码
package com.test.aspectj.expression.annotation;
import com.test.aspectj.expression.Factory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 测试ExecutionAspect切面
*/
public class AspectJExpressionDemo {
public static void main(String[] args) {
System.out.println("=============== 测试 @annotation ===============");
ApplicationContext context = new ClassPathXmlApplicationContext("classpath:spring-chapter3-aspectjannotationexpression.xml");
Factory foodFactory = (Factory) context.getBean("foodFactory");
// FoodFactory#make方法上没有@Log注解,因此不会被增强
System.out.println("FoodFactory#make方法上没有@Log注解,因此不会被增强");
foodFactory.make();
System.out.println("-----分割线-----");
Factory phoneFactory = (Factory) context.getBean("phoneFactory");
// PhoneFactory#make方法上有@Log注解,因此会被增强
System.out.println("PhoneFactory#make方法上有@Log注解,因此会被增强");
phoneFactory.make();
}
}
8、运行结果
=============== 测试 @annotation ===============
FoodFactory#make方法上没有@Log注解,因此不会被增强
来自目标类FoodFactory的消息:生产食品
-----分割线-----
PhoneFactory#make方法上有@Log注解,因此会被增强
来自目标类PhoneFactory的消息:生产手机
来自切面里的消息:打印日志
11、结论
- FoodFactory#make方法上没有@Log注解,因此不会被增强
- PhoneFactory#make方法上有@Log注解,因此会被增强
来源:oschina
链接:https://my.oschina.net/u/3777515/blog/3161926