Spring AOP(4)

本文详细介绍了如何在Spring框架中使用XML配置实现面向切面编程(AOP),涵盖了定义业务Bean、切面、配置切入点及各种通知类型的方法,并通过实例展示了不同场景下通知的执行顺序。
在第三节里面,完满讲了使用@AspectJ注解实现Spring AOP,它需要运行在Java5以上的版本中,对于Java1.4之前的版本,我们也想使用Spring AOP,那么怎么办呢?
一种是像1,2节里面讲的那样,定义Advice实现MethodBeforeAdvice、MethodAfterAdvice、ThrowsAdvice、MethodInterceptor接口之一,然后包装在Advisor中,最后使用BeanPostProcessor(如BeanNameAutoProxyCreator、DefaultAdvisorAutoProxyCreator)创建业务Bean的代理对象。显然这种方式很繁琐。
第二种选择是使用Spring的<aop:...>命名空间,这是在2.0版本以后引入的。它可以运行在Java1.4基础上。本节主要介绍使用<aop:...>基于xml配置实现Spring AOP。
1、定义业务Bean

package spring.aop;
public class UserService {
public void getUser(int id){
System.out.println("the user id: "+id);
}
}

2、定义切面

public class UserAspect {
//业务方法执行前会执行此操作
public void before() {
System.out.println("before advice");
}
//业务方法正常执行结束后会执行此操作
public void afterReturn() {
System.out.println("after-returning advice");
}
//相当于finally,无论业务方法是否产生异常,执行后都会执行此操作
public void after() {
System.out.println("after advice");
}
}

这里的切面就是一个简单的POJO,不需要实现任何接口,不需要继承任何类。里面的方法就是一个Advice(包括before、after、after-throwing、after-return、around五种类型),而Pointcut在xml配置。
3、配置xml

<bean id="userService" class="spring.aop.UserService"/>
<bean id="aspect" class="spring.aop.UserAspect"/>
<aop:config>
<aop:pointcut id="pointcut" expression="execution(* spring.aop.UserService.* (..))"/>
<aop:aspect ref="aspect">
<aop:before pointcut-ref="pointcut" method="before"/>
<aop:after-returning pointcut-ref="pointcut" method="afterReturn" >
<aop:after pointcut-ref="pointcut" method="after">
</aop:aspect>
</aop:config>


测试代码:

ApplicationContext con=new ClassPathXmlApplicationContext("applicationContext.xml");
UserService us=(UserService)con.getBean("userService");
us.getUser(12);

输出结果:
before advice
the user id: 12
after-returning advice
after advice

4、环绕通知

public Object around(ProceedingJoinPoint pjp) throws Throwable{
try{
System.out.println("around advice: before");
Object obj=pjp.proceed();//必须调用此方法,否则后续处理终断
System.out.println("around advice: after-returning");
return obj;
}catch(Exception e){
throw e;
}
}

对应的xml配置:

<aop:aspect ref="aspect">
<aop:around pointcut-ref="pointcut" method="around"/>
</aop:aspect>

环绕通知使我们有机会在方法执行的前后都作出相应的处理,功能最为强大,但必须包含一个处理连接点参数ProceedingJoinPoint pjp,并在与处理(before)之后调用pjp.proceed(),忘记次调用会带来莫名其妙的结果,所以能使用其他Advice进行处理的,尽量使用其他更简单的Advice。

5、异常

public void exception(Exception e) {
//记录异常
System.out.println("exception ["+e+"]");
}

这里参数Exception e是必须的。在xml配置中也必须指明此参数:throwing="e"
对应的xml配置:

<aop:aspect ref="aspect">
<aop:after-throwing pointcut-ref="pointcut" method="exception" throwing="e"/>
</aop:aspect>

6、使用带参数的Advice
上面的例子中除了around和after-throwing含有参数,且around中的参数不需要xml中配置,其他的Advice都是无参数的,要想使用带有自定义参数的Advice,怎么办呢?此时就需要重新配置Pointcut了:
例子,一个带参数的before Advice:
定义业务Bean:

public class UserService {
public void login(User user){
System.out.println("user id: "+u.getId());
}
}

定义before Advice:

public void before(User user) {
System.out.println("user "+u.getName()+"try to login...");
}

在xml中配置Pointcut:

<aop:config>
<aop:pointcut id="pointcut" expression="execution(* spring.aop.UserService.* (User)) and args(u)"/>
<aop:aspect ref="aspect">
<aop:before pointcut-ref="pointcut" method="before" arg-names="u"/>
</aop:aspect>
</aop:config>

关键在于这一行:expression="execution(* spring.aop.UserService.* (User)) and args(u)",指明了切入点为spring.aop.UserService的任何方法,[b][color=red]并且[/color][/b]此方法含有一个类型的User的参数,参数名为u(可以与业务Bean中的参数名不一样,实际上是它的一个别名),<aop:before../>中arg-names就是引用的这个别名(可以与Advice中的参数名不一样)。

[color=red]注意:
其他Advice不能共享此Pointcut,除非Advice中的参数与此Pointcut中的参数一致。[/color]


有人会问,如果Advice中只使用业务Bean方法的[color=red][b]部分参数[/b][/color],该如何做呢?
答案是:依然利用Pointcut配置。
Demo:
定义业务Bean:

public class UserService {
public void login(String name,String psw){
System.out.println("login...");
}
}

定义before Advice:

public void before(String n) {
System.out.println("user "+n+" try to login...");
}

在xml中配置Pointcut:

<aop:config>
<aop:pointcut id="pointcut" expression="execution(* spring.aop.UserService.* (String,..)) and args(n,..)"/>
<aop:aspect ref="aspect">
<aop:before pointcut-ref="pointcut" method="before" arg-names="n"/>
</aop:aspect>
</aop:config>

(String,..)声明切入点至少含有一个String类型的参数,显然可以匹配UserService中的login(String name,String psw);
args(n,..)声明给login(String name,String psw)的第一个参数起了个别名“n”传递给Advice,如果<aop:before...>中arg-names不是“n”,将抛出异常。

7、Advice的顺序
1)一般情况下,before、after-throwing、after的执行顺序是一定的,即:
before-->after-throwing-->after
而before与around中的proceed()方法调用之前的处理则是按照谁配在前谁先处理的原则,after与around中的proceed()方法调用之后的处理也是如此;
当异常抛出时,after-returning操作不会被处理,而after-throwing、after依次被处理。
2)如果是基于注解的方式,在测试中发现是按照如下顺序执行增强的:
before advice
around advice: before
login...
around advice: after-returning
after-returning advice
after advice
植物实例分割数据集 一、基础信息 数据集名称:植物实例分割数据集 图片数量: - 训练集:9,600张图片 - 验证集:913张图片 - 测试集:455张图片 总计:10,968张图片 分类类别:59个类别,对应数字标签0至58,涵盖多种植物状态或特征。 标注格式:YOLO格式,适用于实例分割任务,包含多边形标注点。 数据格式:图像文件,来源于植物图像数据库,适用于计算机视觉任务。 二、适用场景 • 农业植物监测AI系统开发:数据集支持实例分割任务,帮助构建能够自动识别植物特定区域并分类的AI模型,辅助农业专家进行精准监测和分析。 • 智能农业应用研发:集成至农业管理平台,提供实时植物状态识别功能,为作物健康管理和优化种植提供数据支持。 • 学术研究与农业创新:支持植物科学与人工智能交叉领域的研究,助力发表高水平农业AI论文。 • 农业教育与培训:数据集可用于农业院校或培训机构,作为学生学习植物图像分析和实例分割技术的重要资源。 三、数据集优势 • 精准标注与多样性:标注采用YOLO格式,确保分割区域定位精确;包含59个类别,覆盖多种植物状态,具有高度多样性。 • 数据量丰富:拥有超过10,000张图像,大规模数据支持模型充分学习和泛化。 • 任务适配性强:标注兼容主流深度学习框架(如YOLO、Mask R-CNN等),可直接用于实例分割任务,并可能扩展到目标检测或分类等任务。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值