使用Spring 的AOP 对客户管理的 DAO 进行增强 使用Spring 完成转账的事务管理

案例一:使用Spring的AOP对客户管理的DAO进行增强
1.1 案例需求
1.1.1 需求描述
对于CRM的系统而言,现在有很多的DAO类,比如客户的DAO,联系人DAO等等。客户提出一个需求要开发人员实现一个功能对所有的DAO的类中以save开头的方法实现权限的校验,需要时管理员的身份才可以进行保存操作。

1.2 相关知识点
1.2.1 Spring使用AspectJ进行AOP的开发:注解的方式
1.2.1.1 引入相关的jar包:

  • spring的传统AOP的开发的包

spring-aop-4.2.4.RELEASE.jar

com.springsource.org.aopalliance-1.0.0.jar

  • aspectJ的开发包:

com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

spring-aspects-4.2.4.RELEASE.jar

1.2.1.2 引入Spring的配置文件
引入AOP约束:

<beansxmlns=“http://www.springframework.org/schema/beans

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="

   http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd

   http://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd">

1.2.1.3 编写目标类:
public class ProductDao {

public voidsave(){

    System.out.println("保存商品...");

}

public voidupdate(){

    System.out.println("修改商品...");

}

public voiddelete(){

    System.out.println("删除商品...");

}

public voidfind(){

    System.out.println("查询商品...");

}

}

1.2.1.4 配置目标类:

 <bean id="productDao" class="cn.itcast.spring.demo4.ProductDao"></bean>  

1.2.1.5 开启aop注解的自动代理:
aop:aspectj-autoproxy/

1.2.1.6 AspectJ的AOP的注解:
@Aspect:定义切面类的注解

通知类型:

*@Before            :前置通知

*@AfterReturing     :后置通知

*@Around            :环绕通知

* @After             :最终通知

*@AfterThrowing     :异常抛出通知.

@Pointcut:定义切入点的注解

1.2.1.7 编写切面类:
@Aspect

public class MyAspectAnno {

@Before("MyAspectAnno.pointcut1()")

public void before(){

    System.out.println("前置通知===========");

}



@Pointcut("execution(*cn.itcast.spring.demo4.ProductDao.save(..))")

private void pointcut1(){}

}

1.2.1.8 配置切面:

 <bean id="myAspectAnno" class="cn.itcast.spring.demo4.MyAspectAnno"></bean> 

1.2.1.9 其他通知的注解:
@Aspect

public class MyAspectAnno {

@Before("MyAspectAnno.pointcut1()")

public void before(){

    System.out.println("前置通知===========");

}



@AfterReturning("MyAspectAnno.pointcut2()")

public void afterReturning(){

    System.out.println("后置通知===========");

}



@Around("MyAspectAnno.pointcut3()")

public Object around(ProceedingJoinPointjoinPoint) throws Throwable{

    System.out.println("环绕前通知==========");

    Object obj = joinPoint.proceed();

    System.out.println("环绕后通知==========");

    return obj;

}



@AfterThrowing("MyAspectAnno.pointcut4()")

public void afterThrowing(){

    System.out.println("异常抛出通知========");

}



@After("MyAspectAnno.pointcut4()")

public void after(){

    System.out.println("最终通知==========");

}



@Pointcut("execution(*cn.itcast.spring.demo4.ProductDao.save(..))")

private void pointcut1(){}

@Pointcut("execution(* cn.itcast.spring.demo4.ProductDao.update(..))")

private void pointcut2(){}

@Pointcut("execution(*cn.itcast.spring.demo4.ProductDao.delete(..))")

private void pointcut3(){}

@Pointcut("execution(*cn.itcast.spring.demo4.ProductDao.find(..))")

private void pointcut4(){}

}

案例二:Spring的事务管理完成转账的案例
1.3 案例需求:
1.3.1 需求描述:
完成一个转账的功能,需要进行事务的管理,使用Spring的事务管理的方式完成.

1.4 相关知识点
1.4.1 Spring的JDBC的模板:
1.4.1.1 Spring提供了很多持久层技术的模板类简化编程:
1.4.1.2 创建数据库和表:

1.4.1.3 引入相关开发包:
Spring的基本的开发包需要引入的:6个.

1.4.1.4 创建一个测试类:
@Test

// JDBC模板的基本使用:

public void demo1(){

    DriverManagerDataSource dataSource =new DriverManagerDataSource();

    dataSource.setDriverClassName("com.mysql.jdbc.Driver");

    dataSource.setUrl("jdbc:mysql:///spring_day03");

    dataSource.setUsername("root");

    dataSource.setPassword("123");

   

    JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);

    jdbcTemplate.update("insert into account values (null,?,?)","会希",10000d);

}

1.4.2 将连接池的配置交给Spring管理:
1.4.2.1 Spring内置的连接池的配置:
【引入Spring的配置文件】

【配置内置连接池】

<!-- 配置Spring的内置连接池 -->

<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">

<propertyname="driverClassName"value="com.mysql.jdbc.Driver"/>

<propertyname="url"value="jdbc:mysql:///spring_day02"/>

<propertyname="username"value="root"/>

<propertyname="password"value="123"/>

</bean>

【将模板配置到Spring中】

<!-- 配置JDBC模板 -->

<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">

<propertyname="dataSource"ref="dataSource"/>

</bean>

【编写测试类】

**** 引入spring-aop.jar

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(“classpath:applicationContext.xml”)

public class SpringDemo2 {

@Resource(name="jdbcTemplate")

private JdbcTemplate jdbcTemplate;



@Test

public void demo1(){

    jdbcTemplate.update("insertinto account values (null,?,?)","凤姐",10000d);

}

}

1.4.2.2 Spring中配置DBCP连接池:
【引入dbcp连接池的jar包】

【配置连接池】

<!-- 配置DBCP连接池 -->

<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">

<propertyname="driverClassName"value="com.mysql.jdbc.Driver"/>

<propertyname="url"value="jdbc:mysql:///spring_day02"/>

<propertyname="username"value="root"/>

<propertyname="password"value="123"/>

</bean>

1.4.2.3 配置c3p0连接池:
【引入相应的jar包】

com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar

【配置连接池】

<!-- 配置C3P0连接池 -->

<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">

<propertyname="driverClass"value="com.mysql.jdbc.Driver"/>

<propertyname="jdbcUrl"value="jdbc:mysql:///spring_day02"/>

<propertyname="user"value="root"/>

<propertyname="password"value="123"/>

</bean>

1.4.3 JDBC模板的CRUD的操作:
1.4.3.1 JDBC模板CRUD的操作:
@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(“classpath:applicationContext.xml”)

public class SpringDemo3 {

@Resource(name="jdbcTemplate")

private JdbcTemplate jdbcTemplate;



@Test

// 插入操作

public voiddemo1(){

    jdbcTemplate.update("insertinto account values (null,?,?)","冠希",10000d);

}



@Test

// 修改操作

public voiddemo2(){

    jdbcTemplate.update("updateaccount set name=?,money =? where id = ?","思雨",10000d,5);

}



@Test

// 删除操作

public voiddemo3(){

    jdbcTemplate.update("deletefrom account where id = ?", 5);

}



@Test

// 查询一条记录

public voiddemo4(){

   Accountaccount = jdbcTemplate.queryForObject("select * from account where id = ?",new MyRowMapper(), 1);

    System.out.println(account);

}



@Test

// 查询所有记录

public voiddemo5(){

    List<Account>list = jdbcTemplate.query("select * from account",new MyRowMapper());

    for (Account account: list) {

        System.out.println(account);

    }

}



class MyRowMapper implementsRowMapper<Account>{



    @Override

    public Account mapRow(ResultSetrs, introwNum) throwsSQLException {

        Accountaccount = newAccount();

        account.setId(rs.getInt("id"));

        account.setName(rs.getString("name"));

        account.setMoney(rs.getDouble("money"));

        return account;

    }

   

}

}

1.4.4 事务的回顾:
1.4.4.1 什么是事务:
事务逻辑上的一组操作,组成这组操作的各个逻辑单元,要么一起成功,要么一起失败.

1.4.4.2 事务特性:
原子性 :强调事务的不可分割.

一致性 :事务的执行的前后数据的完整性保持一致.

隔离性 :一个事务执行的过程中,不应该受到其他事务的干扰

持久性 :事务一旦结束,数据就持久到数据库

1.4.4.3 如果不考虑隔离性引发安全性问题:
脏读 :一个事务读到了另一个事务的未提交的数据

不可重复读 :一个事务读到了另一个事务已经提交的update的数据导致多次查询结果不一致.

虚读 :一个事务读到了另一个事务已经提交的insert的数据导致多次查询结果不一致.

1.4.4.4 解决读问题:设置事务隔离级别
未提交读 :脏读,不可重复读,虚读都有可能发生

已提交读 :避免脏读。但是不可重复读和虚读有可能发生

可重复读 :避免脏读和不可重复读.但是虚读有可能发生.

串行化的 :避免以上所有读问题.

1.4.5 Spring进行事务管理一组API
1.4.5.1 PlatformTransactionManager:平台事务管理器.
***** 真正管理事务的对象

org.springframework.jdbc.datasource.DataSourceTransactionManager使用Spring JDBC或iBatis进行持久化数据时使用

org.springframework.orm.hibernate3.HibernateTransactionManager 使用Hibernate版本进行持久化数据时使用

1.4.5.2 TransactionDefinition:事务定义信息
事务定义信息:

  • 隔离级别

  • 传播行为

  • 超时信息

  • 是否只读

1.4.5.3 TransactionStatus:事务的状态
记录事务的状态

1.4.5.4 Spring的这组接口是如何进行事务管理:
平台事务管理根据事务定义的信息进行事务的管理,事务管理的过程中产生一些状态,将这些状态记录到TransactionStatus里面

1.4.5.5 事务的传播行为
PROPAGION_XXX :事务的传播行为

* 保证同一个事务中

PROPAGATION_REQUIRED 支持当前事务,如果不存在 就新建一个(默认)

PROPAGATION_SUPPORTS 支持当前事务,如果不存在,就不使用事务

PROPAGATION_MANDATORY 支持当前事务,如果不存在,抛出异常

  • 保证没有在同一个事务中

PROPAGATION_REQUIRES_NEW 如果有事务存在,挂起当前事务,创建一个新的事务

PROPAGATION_NOT_SUPPORTED 以非事务方式运行,如果有事务存在,挂起当前事务

PROPAGATION_NEVER 以非事务方式运行,如果有事务存在,抛出异常

PROPAGATION_NESTED 如果当前事务存在,则嵌套事务执行

1.5 案例代码
1.5.1 搭建转账的环境:
1.5.1.1 创建业务层和DAO的类
public interface AccountService{

public voidtransfer(String from,Stringto,Double money);

}

public class AccountServiceImpl implements AccountService {

// 业务层注入DAO:

private AccountDao accountDao;



public voidsetAccountDao(AccountDao accountDao) {

    this.accountDao=accountDao;

}



@Override

/**

 * from:转出的账号

 * to:转入的账号

 * money:转账金额

 */

public voidtransfer(String from, Stringto, Double money){

    accountDao.outMoney(from,money);

    accountDao.inMoney(to,money);

}

}

public interface AccountDao {

public void outMoney(String from,Doublemoney);



public void inMoney(String to,Double money);

}

public class AccountDaoImpl extends JdbcDaoSupportimplements AccountDao {

@Override

public voidoutMoney(String from, Doublemoney) {

    this.getJdbcTemplate().update("update account set money = money - ? where name =?",money,from);

}



@Override

public voidinMoney(String to,Double money) {

    this.getJdbcTemplate().update("update account set money = money + ? where name =?",money,to); 

}

}

1.5.1.2 配置业务层和DAO

<bean id="accountService" class="cn.itcast.transaction.demo1.AccountServiceImpl">

<propertyname="accountDao"ref="accountDao"/>

</bean>



<!-- 配置DAO的类 -->

<bean id="accountDao" class="cn.itcast.transaction.demo1.AccountDaoImpl">

<propertyname="dataSource"ref="dataSource"/>

</bean>

1.5.1.3 编写测试类
@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(“classpath:applicationContext2.xml”)

public class SpringDemo4 {

@Resource(name="accountService")

private AccountServiceaccountService;



@Test

// 转账的测试:

public voiddemo1(){

    accountService.transfer("会希","凤姐", 1000d);

}

}

1.5.2 Spring的编程式事务(了解)
手动编写代码完成事务的管理:

1.5.2.1 配置事务管理器

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

     <propertyname="dataSource"ref="dataSource"/>

</bean>

1.5.2.2 配置事务管理的模板

<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">

<propertyname="transactionManager"ref="transactionManager"/>

</bean>

1.5.2.3 需要在业务层注入事务管理模板

<bean id="accountService" class="cn.itcast.transaction.demo1.AccountServiceImpl">

<propertyname="accountDao"ref="accountDao"/>

<!-- 注入事务管理模板 -->

<propertyname="transactionTemplate"ref="transactionTemplate"/>

</bean>

1.5.2.4 手动编写代码实现事务管理
public voidtransfer(final Stringfrom, finalStringto, finalDoublemoney) {

    transactionTemplate.execute(new TransactionCallbackWithoutResult() {

       

        @Override

        protected voiddoInTransactionWithoutResult(TransactionStatus status){

            accountDao.outMoney(from,money);

            int d= 1 / 0;

            accountDao.inMoney(to,money);

           

        }

    });

   

   

}

1.5.3 Spring的声明式事务管理XML方式(*****):思想就是AOP.
不需要进行手动编写代码,通过一段配置完成事务管理

1.5.3.1 引入AOP开发的包
aop联盟.jar

Spring-aop.jar

aspectJ.jar

spring-aspects.jar

1.5.3.2 恢复转账环境

1.5.3.3 配置事务管理器

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

<propertyname="dataSource"ref="dataSource"/>

</bean>

1.5.3.4 配置事务的通知

<tx:adviceid="txAdvice"transaction-manager="transactionManager">

<tx:attributes>

   <!--

         isolation="DEFAULT"     隔离级别

         propagation="REQUIRED"  传播行为

         read-only="false"   只读

         timeout="-1"        过期时间

         rollback-for=""     -Exception

         no-rollback-for=""  +Exception

      -->

     <tx:methodname="transfer"propagation="REQUIRED"/>

</tx:attributes>

</tx:advice>

1.5.3.5 配置aop事务
aop:config

<aop:pointcut expression="execution(*cn.itcast.transaction.demo2.AccountServiceImpl.transfer(..))"id="pointcut1"/>

<aop:advisor advice-ref="txAdvice"pointcut-ref="pointcut1"/>

</aop:config>

1.5.4 Spring的声明式事务的注解方式: (*****)
1.5.4.1 引入jar包:
1.5.4.2 恢复转账环境:

1.5.4.3 配置事务管理器:

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

    <propertyname="dataSource"ref="dataSource"/>

</bean>

1.5.4.4 开启事务管理的注解:

<tx:annotation-driventransaction-manager="transactionManager"/>

1.5.4.5 在使用事务的类上添加一个注解:@Transactional

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值