Spring AOP总结

news/2024/6/18 18:53:57
Spring AOP总结
[code]
Spring AOP(Aspect Oriented Programming)

应用程序通常包含两种代码:一是核心业务代码,一是和业务关系不大的代码如日志、事物处理等。
AOP的思想就是使这两种代码分离,从而降低了两种代码的偶合性,达到了易于重用和维护的目的。
AOP和OOP:
在AOP里,每个关注点的实现并不知道是否有其他关注点关注它,这是AOP和OOP的主要区别,
在AOP里组合的流向是从横切关注点到主关注点,在OOP中组合流向是从主关注点到横切关注点。
AOP和OOP所关注的对象不同,AOP是OOP有益的补充,不是对立面。

AOP的3个关键概念:
1:切入点:(PiontCut)
连接点(Jion piont):是指程序运行中的某个阶段,如方法的调用、异常的抛出等。
PiontCut就是Jion piont点的集合,它是程序中需要注入的Advice的集合。指明Advice
在什么 条件下才被触发。
2:通知(Advice):
某个连接点采用的处理逻辑,也就是向连接点注入的代码。
3:Advisor
是PiontCut和Advice的配置器,它包含PiontCut和Advice,是把Advice注入到PiontCut位置的代码。


Spring 的3种切入点的实现:
1:静态切入点:
静态切入点只限于给定的方法和目标类。不考虑方法的参数。
2:动态切入点:
动态切入点不仅限于给定的方法和目标类,还可以指定方法的参数。
动态切入点有很大的性能损耗,一般很少使用。
3:自定义切入点:
正在发展


Spring 的通知:
1:Interception Around通知

Interception Around在Jion Point的前后执行。实现Interception Around通知要实现
MethodInterceptor接口,示例代码如下:
(1)Logger logger=Logger.getLogger(this.getClass().getName());

public Object invoke(MethodInvocation arg0) throws Throwable {
// TODO Auto-generated method stub
logger.log(Level.INFO,arg0.getArguments()[0]+"开始审核数据:");
Object result=arg0.proceed();
logger.log(Level.INFO,arg0.getArguments()[0]+"审核数据结束!");

return result;
}

(2)
/* public class LoginInterceptor implements MethodInterceptor{
public Object invoke(MenthodInvocation invocation) throws Throwable{
System.out.println("开始审核数据");
Object result = invocation.proceed();
System.out.println("审核数据结束");
return result;
}
}*/

2:Before通知
(1)通过log4j.properties实现
Logger logger=Logger.getLogger(this.getClass().getName());

public void before(Method arg0, Object[] arg1, Object arg2)
throws Throwable {
// TODO Auto-generated method stub
logger.log(Level.INFO,arg1[0]+"开始数据操作:");

}
(2):
Before通知在JiontPoint的前执行。实现Befored通知要实现
MethodBeforeAdvice接口,示例代码如下:
public class LoginBeforeAdvice implements MethodBeforeAdvice{
public void before(Menthod m,Object [] atgs,Object target) throws Throwable{
System.out.println("开始审核数据");
}
}


3:After Return 通知

After Return通知在JiontPoint后执行。实现After Returnd通知要实现
AfterReturningAdvice接口,示例代码如下:
public class LoginAfterAdvice implements AfterReturningAdvice{
public void afterReturning(Method m,Object [] atgs,Object target) throws Throwable{
System.out.println("审核数据结束");
}
}

4:Throw通知
(1)Logger logger=Logger.getLogger(this.getClass().getName());
public void afterThrowing(Method arg1, Object[] arg2,Object arg3,Throwable subclass ) throws Throwable{
System.out.println(arg2[0]+"审核数据异常");
}

Throw通知在JiontPoint抛出异常时执行。实现Throw通知要实现
ThrowsAdvice接口,示例代码如下:
public class LoginThrowAdvice implements ThrowsAdvice{
public void afterThrowing(Method arg1, RemoteException ex) throws Throwable{
System.out.println("审核数据异常");
}
}

5:Introduction 通知

Introduction通知在JiontPoint 调用完毕后执行。实现Introduction通知要实现
IntroductionAdvisor接口和IntroductionInterceptor接口。

用ProxyFactoryBean创建AOP代理
使用org.springfamework.aop.framework.ProxyFactoryBean是创建AOP代理的基本方式。

1:使用ProxyFactoryBean代理目标类中的所有方法(参考代码:workspace20070228\SpringAOP1)
示例代码:
<beans>
<bean id="log" class="logAround"/>
<bean id="logBefore" class="logBefore"/>
<bean id="logAfter" class="logAfter"/>
<bean id="logThrow" class="logThrow"/>
<bean id="timebook" class="TimeBook"/>
<!-设定代理类-->
<bean id="logProxy" class ="org.springframework.aop.framework.ProxyFactoryBean">
<!--代理的是接口-->
<property name="proxyInterfaces">
<value>TimeBookInterface</value>
</property>
<!--要代理的目标类-->
<property name="target">
<ref bean="timebook"/>
</property>
<!--程序中的Advice-->
<property name="interceptorNames">
<list>
<value>logBefore</value>
<value>logAfter</value>
<value>logThrow</value>
</list>
</property>
</bean>
</beans>


2:使用ProxyFactoryBean代理目标类中的指定方法

示例代码:
<beans>
<bean id="log" class="logAround"/>
<bean id="logBefore" class="logBefore"/>
<bean id="logAfter" class="logAfter"/>
<bean id="logThrow" class="logThrow"/>
<bean id="timebook" class="TimeBook"/>
<!--代理目标类的指定方法-->
<bean id ="logAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
<property name="advice">
<ref bean="log"/>
</property>
<!--指定要代理的方法-->
<property name="patterns">
<value>.*doCheck.*</value>
</property>
</bean>
<!-设定代理类-->
<bean id="logProxy" class ="org.springframework.aop.framework.ProxyFactoryBean">

<!--要代理的目标类-->
<property name="target">
<ref bean="timebook"/>
</property>
<!--程序中的Advice-->
<property name="interceptorNames">
<list>
<value>logAdvisor</value>
</list>
</property>
</bean>
</beans>

---------------------------------------------------------

正则表达式:
1:.表示可以匹配任何一个字符
2:[]表示只有[]里指定的字符才能匹配
3:*表示匹配次数
4:?表示可以匹配1或0次
5:\是正则表达式的连接符

---------------------------------------------------------------
Spring 中两种AOP代理方式
1:动态代理
动态代理是指代理的是接口,Spring默认的是动态代理
2:CGLIB代理
<beans>
<bean id="log" class="logAround"/>
<bean id="logBefore" class="logBefore"/>
<bean id="logAfter" class="logAfter"/>
<bean id="logThrow" class="logThrow"/>
<bean id="timebook" class="TimeBook"/>
<bean id="logProxy" class ="org.springframework.aop.framework.ProxyFactoryBean">
<property name="proxyTargetClass">
<value>true</value>
</property>
<property name="target">
<ref bean="timebook"/>
</property>
<property name="interceptorNames">
<list>
<value>logBefore</value>
<value>logAfter</value>
<value>logThrow</value>
</list>
</property>
</bean>
</beans>

-----------------------------------------------------------------------
Spring 中的自动代理方式

自动代理可以跨越多个类,不管哪个类中的方法只要符合要求都可以代理
<beans>
<bean id="log" class="logAround"/>
<bean id="logBefore" class="logBefore"/>
<bean id="logAfter" class="logAfter"/>
<bean id="logThrow" class="logThrow"/>
<bean id="timebook" class="TimeBook"/>
<bean id="timework" class="TimeWork"/>
<!--使用自动代理-->
<bean id="autoProxy" class ="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"/>
<bean id="logBeforAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
<property name="advice">
<ref bean="logBefore"/>
</property>
<property name="patterns">
<value>.*do.*</value>
</property>
</bean>
</beans>

------------------------------------------------------------
Spring中的事务处理

事务处理是由多个步骤组成,这些步骤之间有一定的逻辑关系,作为一个整体的操作过程,所有的步骤必须同时成功或失败。
1:提交 当所有的操作步骤都被完整执行后,称为该事物被提交。
2:回滚 由于某个操作失败,导致所有的步骤都没被提交则事物必须回滚,回到事物执行前的状态。

事务的特性:
ACID:原子性(Atomicity)一致性(Consistency)隔离性(Isolation)持久性(Durablity)

Spring中的事务处理是基于动态AOP机制的实现。

1:编程式事务处理:

spring 提供的TransactionTemplate能够以编程的方式实现事务控制。

HelloADO.java:
private DataSource dataSource;
private PlatformTransactionManager transactionManager;
public int create(String msg){
DefaultTransactionDefinition def = new DefaultTransactionDefinition();
TransactionStatus status = transactionManager.getTransaction(def);

try{
JdbcTemplate jt = new JdbcTemplate(dataSource);
int i=jt.update("insert into st(name,password) values('zz','zz')");
return i;
}catch (Exception e){
transactionManager.rollback(status);
return 0;
}
finally {
transactionManager.commit(status);
}
}
applicationContext.xml

<beans>
<bean id ="dataSource" class ="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName">
<value>com.mysql.jdbc.Driver</value>
</property>
<property name="url">
<value>jdbc:mysql://localhost:3306/newdb</value>
</property>
<property name="username">
<value>root</value>
</property>
<property name="password">
<value>lxl</value>
</property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource">
<ref bean="dataSource"/>
</property>
</bean>
<bean id="helloDAO" class ="HelloDAO">
<property name="dataSource">
<ref bean="dataSource"/>
</property>
<property name="transactionManager">
<ref bean="transactionManager"/>
</property>
</bean>
</beans>

2:声明式事务处理:

HelloADO.java:

public class HelloDAO {
private DataSource dataSource ;
private JdbcTemplate jdbcTemplate;
public void setDataSource(DataSource dataSource) {
this.dataSource = dataSource;
jdbcTemplate = new JdbcTemplate(dataSource);
}
public void create(String name){
jdbcTemplate.update("insert into st(name,password)values('lxl','lxl')");
}
}
applicationContext.xml
<beans>
<bean id ="dataSource" class ="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName">
<value>com.mysql.jdbc.Driver</value>
</property>
<property name="url">
<value>jdbc:mysql://localhost:3306/newdb</value>
</property>
<property name="username">
<value>root</value>
</property>
<property name="password">
<value>lxl</value>
</property>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource">
<ref bean="dataSource"/>
</property>
</bean>
<bean id="helloDAO" class ="HelloDAO">
<property name="dataSource">
<ref bean="dataSource"/>
</property>
</bean>
<!-- 声明式事务处理-->
<bean id="helloDAOProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<property name="transactionManager">
<ref bean="transactionManager"/>
</property>
<property name="target">
<ref bean="helloDAO"/>
</property>
<property name="transactionAttributes">
<props>
<!-- 对create方法进行事务管理,PROPAGATION_REQUIRED表示如果没有事务就新建一个事务-->
<prop key="create*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
</beans>
[/code]

http://www.niftyadmin.cn/n/3854437.html

相关文章

Android学习笔记43-XML文件解析(Pull方式)

解析XML文档的方法是很多的&#xff0c;除了常用的SAX和DOM方法以外&#xff0c;还可以使用Java自带的Pull来解析XML文档。 1.使用Pull解析器解析XML文档 Pull解析器的运行方式和SAX解析器很相似&#xff0c;它提供了类似的事件&#xff0c;比如文档开始、文档结束、开始元素和…

Android Tools Project Site

Android Tools Project Site Search this siteProjects OverviewScreenshotsRelease StatusRoadmapDownloadPreview ChannelRecent ChangesTechnical docsNew Build SystemKnown IssuesTipsBuild OverviewContributingFeedbackTechnical docs‎ > ‎New Build System‎ >…

12款浏览器兼容性测试工具推荐

对于前端开发工程师来说&#xff0c;确保代码在各种主流浏览器的各个版本中都能正常工作是件很费时的事情&#xff0c;幸运的是&#xff0c;有很多优秀的工具可以帮助测试浏览器的兼容性&#xff0c;让我们一起看看这些很棒的工具。 Spoon Browser Sandbox 点击你需要测试的浏…

总结Struts,Hibernate,Spring框架

总结Struts&#xff0c;Hibernate&#xff0c;Spring框架 [code] Hibernate工作原理及为什么要用&#xff1f; 原理&#xff1a; 1.读取并解析配置文件 2.读取并解析映射信息&#xff0c;创建SessionFactory 3.打开Sesssion 4.创建事务Transation 5.持久化操作 6.提交事务 7.关…

visual studio 2012 update3

http://www.microsoft.com/zh-cn/download/confirmation.aspx?id39305

Struts+Spring+Hibernate练习

StrutsSpringHibernate练习[code]工具&#xff1a;Eclipse3.1、MyEclipse4.03、Tomcat5.5.9、Properties Editor插件、MySql4.1.13新建工程&#xff1a;名称为 login创建Struts框架创建 index.jsp&#xff0c;增加一链接指向 login.jsp按下Ctrl N&#xff0c;创建 login.jsp、…

地址代码uboot之start.S源码分析

每日一贴,今天的容内关键字为地址代码 /arch/arm920t/cpu/start.s件文就是uboot的第一阶段&#xff0c;uboot的前4K容内&#xff0c;即nandflash中的前4K容内会被拷贝到cpu中的SRAM中运行&#xff0c;这一小段代码担任初始化硬件环境&#xff0c;并将残余的Uboot代码加载到内存…

[转]搜狗2012.9.23校园招聘会笔试题

转载请标明出处&#xff0c;原文地址&#xff1a;http://blog.csdn.net/hackbuteer1/article/details/8016173 C/C类 1、以下程序的输出是&#xff08;12&#xff09; [cpp] view plaincopy class Base { public: Base(int j) : i(j) { } virtual ~Base() { …