第三章 AOP 通过Java API创建切面

本文详细介绍了如何在Spring框架中使用不同类型的切点来精确地控制AOP增强的织入位置,包括静态方法切点、动态方法切点、注解切点、表达式切点等,并提供了具体的代码示例。

    在前面使用增强的时候,我们发现增强会被织入到目标类的所有的方法中。我们如果把增强织入到目标类的特定的方法中,需要使用切点进行目标连接点的定位。然后我们可以通过切点及增强生成一个切面了。


3.4.1切点类型

静态方法切点:org.springframework.aop.support.StaticMethodMatcherPointcut

动态方法切点:org.springframework.aop.support.DynamicMethodMatcherPointcut

注解切点:org.springframework.aop.support.annotation.AnnotationMatchingPointcut

表达式切点:org.springframework.aop.support.ExpressionPointcut

流程切点:org.springframework.aop.support.ControlFlowPointcut


3.4.2静态方法来匹配切面

org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor


a、增强类

在这里使用的是3.3.2的前置增强:

http://blog.csdn.net/p_3er/article/details/9239925

只是UserDaoImpl中有两个方法,save方法和delete方法。然后我们给save方法配置切面。

b、切面类继承StaticMethodMatcherPointcutAdvisor

public class StaticAdvisor extends StaticMethodMatcherPointcutAdvisor {
	/*
	 * 切点方法匹配规则
	 */
	public boolean matches(Method method, Class<?> clazz) {
		return method.getName().equals("save");
	}

	/*
	 * 重写StaticMethodMatcherPointcut的getClassFilter()
	 * 匹配哪个类下的方法
	 */
	public ClassFilter getClassFilter() {
		ClassFilter classFilter = new ClassFilter() {
			public boolean matches(Class<?> clazz) {
				return UserDaoImpl.class.isAssignableFrom(clazz);
			}
		};
		return classFilter;
	}
}


c、配置

<!-- 增强Bean -->
	<bean id="userDaoBeforeAdvice" class="cn.framelife.spring.advice.UserDaoBeforeAdvice"></bean>
	
	<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	<!-- 管理切面类。p:advice-ref把增强放入切面 -->
	<bean id="staticAdvisor" class="cn.framelife.spring.advisor.StaticAdvisor" p:advice-ref="userDaoBeforeAdvice"></bean>
	
	<!-- 
		设置父代理类 
		p:interceptorNames 放切面,而不再是增强
	-->
	<bean id="paramProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="staticAdvisor"
		p:proxyTargetClass="true" />	
	<!-- 设置子代理类 -->
	<bean id="userDaoProxy" parent="paramProxy" p:target-ref="userDao"></bean>

d、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserDao userDao = (UserDao) context.getBean("userDaoProxy");
		//UserDaoImpl中有两个方法,save方法中有切面,delete中没切面
		userDao.save();
		System.out.println("----------");
		userDao.delete();

e、结果

我是前置增强:save
保存用户...
----------
删除用户...



3.4.3静态正则表达式方法配置切面

     这种方法不需要自己写切面类。

       org.springframework.aop.support.RegexpMethodPointcutAdvisor是正则表达式方法匹配的切面实现类。


a、增强类

在这里使用的是3.3.2的前置增强:

http://blog.csdn.net/p_3er/article/details/9239925


b、配置

<!-- 增强Bean -->
	<bean id="userDaoBeforeAdvice" class="cn.framelife.spring.advice.UserDaoBeforeAdvice"></bean>
	
	<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	<!-- 
		设置切面Bean 
		patterns 用正则表达式定义目标类全限定方法名的匹配模式串
			目标类全限定方法名,指的是带类名的方法名。如: cn.framelife.spring.dao.impl.UserDaoImpl.delete()
		<value>.*delete</value> 匹配模式串
	-->
	<bean id="regexpAdvisor" 
		class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"
		p:advice-ref="userDaoBeforeAdvice">
		<property name="patterns">
			<list>
				<value>.*delete</value>
			</list>
		</property>
	</bean>
	
	<!-- 
		设置代理类 
	-->
	<bean id="regexpProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="regexpAdvisor"
		p:target-ref="userDao"
		p:proxyTargetClass="true" />	

c、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserDao userDao = (UserDao) context.getBean("regexpProxy");
		userDao.save();
		System.out.println("----------");
		userDao.delete();


d、结果

保存用户...
----------
我是前置增强:delete
删除用户...


常用的正则表达式规则

1.*set.*表示所有类中的以set为前缀的方法。如:com.abc.UserDao.setName()

2com\.abc\.service\..*表示com.abc.service.包下所有的类的所有的方法。

Com.abc.service.a.User.setName()

Com.abc.service.UserService.save()

3com\.abc\.service\..*Service\..*表示com.abc.service包下以Service结尾的类的所有的方法。如:com.abc.service.UserService.save()

4com\.abc\.service\..*Service\.save.+匹配所有以save为前缀的方法,而且save后必须拥有一个或多个字符。如:com.abc.service.UserService.saveUser()



3.4.4动态切面

org.springframework.aop.support.DynamicMethodMatcherPointcut类即有静态切点检查的方法,也有动态切点检查的方法。由于动态切点检查会对性能造成很大的影响,我们应当避免在运行时每次都对目标类的各个方法进行检查。

Spring检查动态切点的机制:在创建代理时对目标类的每个连接点进行静态切点检查,如果仅静态切点检查就可以知道连接点是不匹配的,在运行期就不进行动态检查了;如果静态切点检查是匹配的,在运行期才进行动态方法检查。

动态切面通过DefaultPointcutAdvisor切面类与DynamicMethodMatcherPointcut切点结合起来生成。主要是针对连接点方法的参数。

a、增强类

在这里使用的是3.3.2的前置增强:

http://blog.csdn.net/p_3er/article/details/9239925

只是UserDaoImpl中多了一个out方法,带一个参数。


b、切点类继承DynamicMethodMatcherPointcut

public class DynamicPointcut extends DynamicMethodMatcherPointcut {
	//用list集合保存参数名
	private List<String> specialNames = new ArrayList<String>();
	
	public DynamicPointcut(){
		specialNames.add("aa");
		specialNames.add("bb");
	}
	/*
	 * 对类进行静态切点检查
	 */
	public ClassFilter getClassFilter() {
		
		return new ClassFilter() {
			@Override
			public boolean matches(Class<?> targetClass) {
				System.out.println("使用getClassFilter静态检查:"+targetClass.getName());
				return UserDaoImpl.class.isAssignableFrom(targetClass);
			}
		};
	}

	/*
	 *对方法进行静态切点检查
	 */
	public boolean matches(Method method, Class<?> targetClass) {
		System.out.println("使用matches(method,targetClass)方法静态检查:"+targetClass.getName()+"--"+method.getName());
		return method.getName().equals("out");
	}


	/*
	 *对方法进行动态切点检查
	 */
	public boolean matches(Method method, Class<?> targetClass, Object[] args) {
		System.out.println("使用matches(method,targetClass)方法动态检查:"+targetClass.getName()+"--"+method.getName()+"的参数");
		String name = (String)args[0];
		return specialNames.contains(name);
	}
}

c、配置

<!-- 增强Bean -->
	<bean id="userDaoBeforeAdvice" class="cn.framelife.spring.advice.UserDaoBeforeAdvice"></bean>

	<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	
	<!-- 切面 -->
	<bean id="dynamicAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor">
		<!-- 设置切点 -->
		<property name="pointcut">
			<bean class="cn.framelife.spring.pointcut.DynamicPointcut"></bean>
		</property>
		<!-- 织入增强 -->
		<property name="advice" ref="userDaoBeforeAdvice"></property>
	</bean>
	
	<!-- 
		设置代理 
	-->
	<bean id="dynamicProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="dynamicAdvisor"
		p:target-ref="userDao"
		p:proxyTargetClass="true" />	

d、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserDao userDao = (UserDao) context.getBean("dynamicProxy");
		//UserDaoImpl中多了一个out方法,带一个参数。
		System.out.println("----------");
		userDao.save();
		System.out.println("----------");
		userDao.out("aa");
		System.out.println("----------");
		userDao.out("1111");
		System.out.println("----------");


e、结果

使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--out
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--save
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--delete
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--clone
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--toString
----------
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--save
保存用户...
----------
使用getClassFilter静态检查:cn.framelife.spring.dao.impl.UserDaoImpl
使用matches(method,targetClass)方法静态检查:cn.framelife.spring.dao.impl.UserDaoImpl--out
使用matches(method,targetClass)方法动态检查:cn.framelife.spring.dao.impl.UserDaoImpl--out的参数
我是前置增强:out
out输出名字为:aa
----------
使用matches(method,targetClass)方法动态检查:cn.framelife.spring.dao.impl.UserDaoImpl--out的参数
out输出名字为:1111
----------


3.4.5流程切面

一个类中的某一方法使用目标类的两个方法,那么我们可以使用流程切面给这两个方法都积入增强。

如:UserServiceImpl(使用类)operate方法中使用UserDaoImpl(目标类)的两个方法。

流程切面使用ControlFlowPointcutDefaultPointcutAdvisor结合形成。

a、增强类

在这里使用的是3.3.2的前置增强:

http://blog.csdn.net/p_3er/article/details/9239925

只是UserDaoImpl中有两个方法,save方法和delete方法。然后我们给save方法配置切面。


bUserServiceImpl

public class UserServiceImpl implements UserService {
	private UserDao userDao;
	
	public void setUserDao(UserDao userDao) {
		this.userDao = userDao;
	}

	public void addUser() {
		userDao.save();
	}

	@Override
	public void operate() {
		//这里同时使用了userDao的两个方法
		userDao.delete();
		userDao.save();
	}
}

c、配置

<!-- 增强Bean -->
	<bean id="userDaoBeforeAdvice" class="cn.framelife.spring.advice.UserDaoBeforeAdvice"></bean>

	<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	<!-- 切点 -->
	<bean id="controlFlowPointcut" class="org.springframework.aop.support.ControlFlowPointcut">
		<!-- 指定流程切点的类 -->
		<constructor-arg type="java.lang.Class" value="cn.framelife.spring.service.impl.UserServiceImpl"></constructor-arg>
		<!-- 指定流程切点的方法 -->
		<constructor-arg type="java.lang.String" value="operate"></constructor-arg>
	</bean>
	
	<!-- 切面 -->
	<bean id="controlFlowAdvisor" 
		class="org.springframework.aop.support.DefaultPointcutAdvisor"
		p:pointcut-ref="controlFlowPointcut"
		p:advice-ref="userDaoBeforeAdvice"/>	
	
	<!-- 
		设置代理 
	-->
	<bean id="controlFlowProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="controlFlowAdvisor"
		p:target-ref="userDao"
		p:proxyTargetClass="true" />
	
	<!-- 把使用目标类的Bean交由Spring管理 -->
	<bean id="userService" class="cn.framelife.spring.service.impl.UserServiceImpl">
		<property name="userDao" ref="controlFlowProxy"></property>
	</bean>


d、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserService userService = (UserService) context.getBean("userService");
		userService.operate();
		System.out.println("----------------");
		userService.addUser();


e、结果

我是前置增强:delete
删除用户...
我是前置增强:save
保存用户...
----------------
保存用户...


3.4.6复合切面

在前面的例子中,我们所定义的切面都只有一个切点而已。有时候我们一个切面需要多个切点,也就是多个条件才能决定连接点。多个切点组成一个切点,这样的切点是复合切点。由复合切点加上增强形成的切面,称为复合切面。

a、增强类

在这里使用的是3.3.2的前置增强:

http://blog.csdn.net/p_3er/article/details/9239925


b、一个普通类中有一个获取Pointcut的方法

public class MyPointcut {
//获取Pointcut的方法名是get开头
	public Pointcut getMyComposablePointcut(){
		//创建一个复合切点
		ComposablePointcut cp = new ComposablePointcut();

		//创建一个流程切点(参数:使用类、类中的方法)这里使用的是3.4.5中的UserServiceImpl类
		Pointcut pt1 = new ControlFlowPointcut(UserServiceImpl.class,"operate");
		
		//创建一个静态方法切点
		Pointcut pt2 = new StaticMethodMatcherPointcut() {
			public boolean matches(Method method, Class<?> clazz) {
				return UserDaoImpl.class.isAssignableFrom(clazz)&&method.getName().equals("delete");
			}
		};

		//两个切点进行交集操作.
		return cp.intersection(pt1).intersection(pt2);
	}
}

c、配置

<!-- 增强Bean -->
	<bean id="userDaoBeforeAdvice" class="cn.framelife.spring.advice.UserDaoBeforeAdvice"></bean>

	<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	<!-- 切点所在类 -->
	<bean id="myPoincut" class="cn.framelife.spring.pointcut.MyPointcut"></bean>
	
	<!-- 
		切面
		p:pointcut #{ myPoincut.myComposablePointcut} 是由MyPointcut的getMyComposablePointcut方法获取的
	-->
	<bean id="composableAdvisor" 
		class="org.springframework.aop.support.DefaultPointcutAdvisor"
		p:pointcut="#{ myPoincut.myComposablePointcut}"
		p:advice-ref="userDaoBeforeAdvice"/>	
	
	<!-- 
		设置代理 
	-->
	<bean id="composableProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="composableAdvisor"
		p:target-ref="userDao"
		p:proxyTargetClass="true" />
	
	<!-- 把使用目标类的Bean交由Spring管理 -->
	<bean id="userService" class="cn.framelife.spring.service.impl.UserServiceImpl">
		<property name="userDao" ref="composableProxy"></property>
	</bean>


d、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserService userService = (UserService) context.getBean("userService");
		userService.operate();

e、结果

我是前置增强:delete
删除用户...
保存用户...

3.4.7引介切面

a、使用3.3.6引介增强里面的东西(接口与增强类)


b、配置

<!-- 目标Bean -->
	<bean id="userDao" class="cn.framelife.spring.dao.impl.UserDaoImpl"></bean>
	
	<!-- 
		切面
		constructor-arg 设置引介增强
	-->
	<bean id="introductionAdvisor" 
		class="org.springframework.aop.support.DefaultIntroductionAdvisor">
		<constructor-arg>
			<bean class="cn.framelife.spring.advice.IntroductionAdvice"></bean>
		</constructor-arg>
	</bean>	
	
	<!-- 
		设置代理 
	-->
	<bean id="incluctionProxy" 
		class="org.springframework.aop.framework.ProxyFactoryBean"
		p:interceptorNames="introductionAdvisor"
		p:target-ref="userDao"
		p:proxyTargetClass="true" />


c、测试

ApplicationContext context = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
		UserDao userDao = (UserDao) context.getBean("aProxy");
		userDao.save();
		
		System.out.println("-------------");
		
		AInterface a = (AInterface)userDao;
		a.say();
		
		System.out.println("-------------");
		userDao.save();


d、结果

方法执行前执行
org.springframework.aop.framework.Cglib2AopProxy$CglibMethodInvocation
save
保存用户...
方法执行后执行
-------------
方法执行前执行
org.springframework.aop.framework.Cglib2AopProxy$CglibMethodInvocation
say
UserDao要说话
方法执行后执行
-------------
方法执行前执行
org.springframework.aop.framework.Cglib2AopProxy$CglibMethodInvocation
save
保存用户...
方法执行后执行


以下是Java Agent、AOP切面和SDK埋点的详细介绍及对比分析: ### 1. **Java Agent** **定义**: Java Agent是JVM层面的字节码增强技术,通过`javaagent`参数在程序启动时加载,利用`java.lang.instrument`包动态修改类字节码(如添加埋点逻辑)。 **核心特点**: - **非侵入性**:无需修改业务代码,直接操作字节码实现埋点。 - **全局性**:可监控JVM内所有类(包括第三方库),适合全链路追踪。 - **动态性**:支持运行时重新加载类(需配合热部署工具)。 - **性能开销**:字节码操作可能引入轻微性能损耗,但通常可接受。 **适用场景**: - 需要监控JVM内所有方法调用(如全链路性能分析)。 - 无法修改源码的遗留系统埋点。 - 动态代理无法覆盖的场景(如静态方法、私有方法)。 **示例**: ```java // Agent入口类 public class MyAgent { public static void premain(String args, Instrumentation inst) { inst.addTransformer(new ClassFileTransformer() { @Override public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ...) { if (className.equals("com/example/TargetClass")) { // 修改字节码插入埋点逻辑 return modifyBytecode(); } return null; } }); } } ``` ### 2. **AOP切面** **定义**: AOP(面向切面编程)通过切面(Aspect)将横切关注点(如埋点、日志)与业务逻辑分离,利用切点(Pointcut)匹配目标方法,通过增强(Advice)插入埋点代码。 **核心特点**: - **解耦性**:埋点逻辑与业务代码完全分离,降低耦合度。 - **灵活性**:支持多种增强类型(如`@Before`、`@After`、`@Around`)。 - **可维护性**:切面集中管理埋点规则,修改无需触达业务类。 - **依赖框架**:通常需结合Spring AOP或AspectJ使用。 **适用场景**: - 需要精细化控制埋点位置(如特定方法、异常捕获)。 - 业务代码频繁变更,需快速调整埋点策略。 - 与Spring生态集成的系统(如微服务、Web应用)。 **示例**(Spring AOP): ```java @Aspect @Component public class TrackingAspect { @Around("execution(* com.example.Service.*(..))") public Object track(ProceedingJoinPoint joinPoint) throws Throwable { // 埋点逻辑:记录方法入参、耗时等 long start = System.currentTimeMillis(); Object result = joinPoint.proceed(); long duration = System.currentTimeMillis() - start; log.info("Method {} took {}ms", joinPoint.getSignature(), duration); return result; } } ``` ### 3. **SDK埋点** **定义**: SDK埋点通过手动调用埋点API(如`Tracker.trackEvent()`)或封装工具类实现数据采集,需在业务代码中显式插入埋点逻辑。 **核心特点**: - **可控性**:开发者完全掌握埋点位置和时机。 - **侵入性**:需修改业务代码插入埋点调用。 - **性能最优**:无字节码操作或代理开销,性能损耗最低。 - **灵活性受限**:埋点规则变更需修改代码并重新部署。 **适用场景**: - 对性能要求极高的场景(如高频交易系统)。 - 埋点逻辑复杂,需结合业务上下文(如用户行为分析)。 - 无法使用AOPJava Agent的环境(如部分移动端或嵌入式系统)。 **示例**: ```java public class UserService { private final Tracker tracker = new Tracker(); public void login(String userId) { // 手动埋点 tracker.trackEvent("LOGIN", Map.of("userId", userId)); // 业务逻辑... } } ``` ### **对比总结** | **维度** | **Java Agent** | **AOP切面** | **SDK埋点** | |----------------|-----------------------------|-----------------------------|-----------------------------| | **侵入性** | 无(字节码层面) | 无(代码解耦) | 有(需手动调用) | | **性能开销** | 中(字节码操作) | 低(代理调用) | 最低(直接调用) | | **灵活性** | 高(全局监控) | 高(切点匹配) | 低(需改代码) | | **维护成本** | 高(字节码调试复杂) | 中(切面集中管理) | 低(代码直观) | | **适用场景** | 全链路追踪、遗留系统 | 业务解耦、快速迭代 | 高性能、复杂逻辑 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值