AOP与Spring控制AOP

来源:互联网 发布:vscode和visualstudio 编辑:程序博客网 时间:2024/04/29 12:49

Spring AOP与事务代理机制

Spring提供了TransactionProxyFactoryBean作为动态代理的生成工厂,该工厂产生目标bean的动态AOP事务代理。事务代理根据所配置的事务属性自动管理事务操作。该事务代理对事务的透明管理依赖于一个在ApplicationContext范围全局可见的PlatformTransactionManager。Spring提供两个默认的事务管理器实现:DataSourceTransactionManager和JtaTransactionManager。前者只能支持单个jdbc数据源,后者可以支持多个数据源,可以做分布式事务管理。一般由容器提供支持JTA的事务管理器实现,Spring的JtaTransactionManager会自动通过JNDI检测TransactionManager或UserTransaction的存在,也可以在配置中指定JNDI的设置。如果想不依赖于容器实现分布式事务管理,可以采用开源的事务管理器实现JOTM。事务管理器必须要知道被管理的DataSource的位置,如果是DataSourceTransactionManager,在事务管理器配置中直接指定JDBC数据源的引用,如果是JtaTransactionManager,容器负责告知事务管理器需要管理的数据源。不管是哪一种情况,都需要在Dao中配置与事务管理器相对应的数据源。

 

     1. AOP是什么?
  
  AOP是OOP的延续,是Aspect Oriented Programming的缩写,意思是面向方面编程。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP可以说也是这种目标的一种实现。

AOP中的概念:

l  Aspect(切面):指横切性关注点的抽象即为切面,它与类相似,只是两者的关注点不一样,类是对物体特征的抽象,而切面是横切性关注点的抽象。

l  Joinpoint(连接点):所谓连接点是指那些被拦截到的点,在spring中,这些点指的是方法,因为spring只支持方法类型的连接点,实际上joinpoint还可以是field或类构造器

l  Pointcut(切入点):所谓切入点是指我们要对那些joinpoint进行拦截的定义。

l  Advice(通知):所谓通知是指拦截到joinpoint之后所要做的事情就是通知。通知分为前置通知,后置通知,异常通知,最终通知,环绕通知

l  Target(目标对象):代理的目标对象

l  Weave(织入):将aspects应用到target对象并导致proxy对象创建的过程称为织入。

l  Introduction(引入):在不修改类代码的前提下,Introduction可以在运行期为类动态地添加一些方法或field。

  
  2. 切面意义何在?
  
  就可以在这层切面上进行统一的集中式权限管理。而业务逻辑组件则无需关心权限方面的问题。也就是说,通过切面,我们可以将系统中各个不同层次上的问题隔离开来,实现统一集约式处理。各切面只需集中于自己领域内的逻辑实现。这一方面使得开发逻辑更加清晰,专业化分工更加易于进行;另一方面,由于切面的隔离,降低了耦合性,我们就可以在不同的应用中将各个切面组合使用,从而使得代码可重用性大大增强。
  
  3. AOP应用范围
  
  Authentication 权限
  Caching 缓存
  Context passing 内容传递
  Error handling 错误处理
  Lazy loading 懒加载
  Debugging  调试
  logging, tracing, profiling and monitoring 记录跟踪 优化 校准
  Performance optimization 性能优化
  Persistence  持久化
  Resource pooling 资源池
  Synchronization 同步
  Transactions 事务
  
  三.Spring事务处理
  
  1.Spring事务管理能给我们带来什么?
  
  对于传统的基于特定事务资源的事务处理而言(如基于JDBC 的数据库访问),Spring并不会对其产生什么影响,我们照样可以成功编写并运行这样的代码。同时,Spring还提供了一些辅助类可供我们选择使用,这些辅助类简化了传统的数据库操作流程,在一定程度上节省了工作量,提高了编码效率。
  
  对于依赖容器的参数化事务管理而言,Spring则表现出了极大的价值。Spring本身也是一个容器,只是相对EJB容器而言,Spring显得更为轻便小巧。我们无需付出其他方面的代价,即可通过Spring实现基于容器的事务管理(本质上来讲,Spring的事务管理是基于动态AOP)。
  
  2. Hibernate in Spring
  applicationContext.xml
  
  <!-- Hibernate SessionFactory --><bean id="sessionFactory“
  class="org.springframework.orm.hibernate.LocalSessionFactoryBean">
  <property name="dataSource"><ref local="dataSource" /></property>
  <property name="mappingResources">
  <list><!-- Add list of .hbm.xml files here -->
  <value>org/mzone/model/Tuser.hbm.xml</value>
  <value>org/mzone/model/Article.hbm.xml</value>
  </list>
  </property>
  <property name="hibernateProperties">
  <props>
  <prop  key="hibernate.dialect">net.sf.hibernate.dialect.SybaseDialec</prop>
  <prop key="hibernate.show_sql">True</prop>
  </props>
  </property>
  </bean><!-- Transaction manager for a single Hibernate SessionFactory (alternative to JTA) -->
  <bean id="transactionManager"
  class="org.springframework.orm.hibernate.HibernateTransactionManager">
  <property name="sessionFactory"><ref local="sessionFactory" /></property>
  </bean>
  <bean id="baseTxProxy" lazy-init="true"
  
  class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
  <property name="transactionManager"><ref bean="transactionManager"/></property>
  <property name="target">
  <ref local=" userManagerTarget " />
  </property>
  <property name="transactionAttributes">
  <props>
  <prop key="save*">PROPAGATION_REQUIRED</prop>
  <prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
  <prop key="remove*">PROPAGATION_REQUIRED</prop>
  <prop key="*">PROPAGATION_REQUIRED</prop>
  </props>
  </property>
  </bean>
  <bean id="userManagerTarget" class="org.mzone.service.impl.UserManagerImpl">
  <property name="userDAO"><ref local="userDAO"/></property>
  <property name="articleDao"><ref local="articleDAO"/></property>
  </bean>
  
  UserDAO.java ArticleDAO.java
  public class UserDAOImpl extends HibernateDaoSupport implements UserDAO {
  public void saveUser(Tuser user) {
  getHibernateTemplate().saveOrUpdate(user);
  }}public class ArticleDAOImpl extends HibernateDaoSupport implements ArticleDAO {
  public void saveArticle(Article article) {
  getHibernateTemplate().saveOrUpdate(article);
  }}
  
  HibernateDaoSupport
  
  实现了HibernateTemplate和SessionFactory实例的关联。HibernateTemplate对Hibernate Session操作进行了封装,而HibernateTemplate.execute方法则是一封装机制的核心,感兴趣可以研究一下其实现机制。
  
  借助HibernateTemplate我们可以脱离每次数据操作必须首先获得Session实例、启动事务、提交/回滚事务以及烦杂的try/catch/finally的繁琐操作。从而获得以上代码中精干集中的逻辑呈现效果。
  
  org.mzone.service.impl.UserManagerImpl
  public class UserManagerImpl implements UserManager {
  private UserDAO userDao;
  private ArticleDAO articleDao;
  public void saveUserAndArticle(Tuser user, Article article) {
  userDao.saveUser(user);
  articleDao.saveArticle(article);
  }}
  
  测试代码
  InputStream is = new FileInputStream("applicationContext.xml");
  XmlBeanFactory factory = new XmlBeanFactory(is);UserManager userManager =
  (UserManager )factory.getBean(" baseTxProxy ");
  user = new Tuser();
  article = new Article();
  user.setUsername("hellboys_topic 1");
  user.setPassword("12345678_topic 1");
  article.setTitle("hellboys_topic 1");
  article.setContent("hellboys_topic 1");
  userManager.saveUserAndArticle(user,article);
  
  注意问题
  UserManager userManager =
  (UserManager )factory.getBean(" baseTxProxy ");UserManager userManager =
  (UserManagerImpl) ctx.getBean("baseTxProxy");java.lang.ClassCastException
  
  原因在于Spring的AOP实现机制,前面曾经提及,Spring中的事务管理实际上是基于动态AOP机制实现,为了实现动态AOP,Spring在默认情况下会使用Java DynamicProxy,但是,Dynamic Proxy要求其代理的对象必须实现一个接口,该接口定义了准备进行代理的方法。而对于没有实现任何接口的Java Class,需要采用其他方式,Spring通过CGLib10实现这一功能。
  CGLib可以在运行期对Class行为进行修改。由于其功能强大,性能出众,常常被作为Java Dynamic Proxy
  之外的动态Proxy模式的实现基础。在Spring、Hibernate中都用到了CGLib类库。