Spring v4

Spring v4.2框架–事务&JDBC模板

1.Spring的AOP注解开发(基于AspectJ)

1)小试牛刀

  • a.导入spring的基本开发包

    • 在这里插入图片描述
  • b.导入aop的jar包

    • 在这里插入图片描述
  • c.引入配置文件

    • log4j的配置文件:log4j.properties
    • ### direct log messages to stdout ###
      log4j.appender.stdout=org.apache.log4j.ConsoleAppender
      log4j.appender.stdout.Target=System.err
      log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
      log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n
      
      ### direct messages to file mylog.log ###
      log4j.appender.file=org.apache.log4j.FileAppender
      log4j.appender.file.File=c\:mylog.log
      log4j.appender.file.layout=org.apache.log4j.PatternLayout
      log4j.appender.file.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n
      
      ### set log levels - for more verbose logging change 'info' to 'debug' ###
      # error warn info debug trace
      log4j.rootLogger= info, stdout
      
    • spring的配置文件:applicationContext.xml

    • <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"
      	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      	xmlns:context="http://www.springframework.org/schema/context"
      	xmlns:aop="http://www.springframework.org/schema/aop"
      	xmlns:tx="http://www.springframework.org/schema/tx"
      	xsi:schemaLocation="http://www.springframework.org/schema/beans 
      	http://www.springframework.org/schema/beans/spring-beans.xsd
      	http://www.springframework.org/schema/context
      	http://www.springframework.org/schema/context/spring-context.xsd
      	http://www.springframework.org/schema/aop
      	http://www.springframework.org/schema/aop/spring-aop.xsd
      	http://www.springframework.org/schema/tx 
      	http://www.springframework.org/schema/tx/spring-tx.xsd">
      	
      </beans> 
      
  • d.编写目标类(被增强的对象)

  • package com.nike.spring.demo1;
    
    public class OrderDao {
    	
    	public void save(){
    		System.out.println("保存订单。。。");
    	}
    	public void update(){
    		System.out.println("修改订单。。。");
    	}
    	public void delete(){
    		System.out.println("删除订单。。。");
    	}
    	public void find(){
    		System.out.println("查找订单。。。");
    	}
    }
    
    • 把目标类交给spring管理

    • 	<!--      配置目标类            -->
      	<bean id="orderDao" class="com.nike.spring.demo1.OrderDao" />
      
  • e.编写切面类并配置

    • package com.nike.spring.demo1;
      /**
       * 使用注解开发切面类
       * @author 猪猪
       *
       */
      public class MyAspectAnno {
      	
      	public void before(){
      		System.out.println("前置通知。。。");
      	}
      	public void after(){
      		System.out.println("后置通知。。。");
      	}
      }
      
    • <!--      配置切面类             -->
      <bean id="myAspectAnno" class="com.nike.spring.demo1.MyAspectAnno"/>
      
  • f.使用注解的AOP对目标类进行增强

    • 在配置文件打开注解的AOP开发

    • 	<!-- 在配置文件打开注解的AOP开发 -->
      	<aop:aspectj-autoproxy/>
      
    • 在切面类上使用注解

    • package com.nike.spring.demo1;
      
      import org.aspectj.lang.annotation.Aspect;
      import org.aspectj.lang.annotation.Before;
      
      /**
       * 使用注解开发切面类
       * @author 猪猪
       *
       */
      @Aspect //告知spring这是一个切面类
      public class MyAspectAnno {
      	
      	@Before(value="execution(* com.nike.spring.demo1.OrderDao.save(..))")
      	public void before(){
      		System.out.println("前置通知。。。");
      	}
      	
      	public void after(){
      		System.out.println("后置通知。。。");
      	}
      }
      
  • g.编写测试类

    • 引入spring的junit测试架包:spring-test-4.2.4.RELEASE.jar

    • package com.nike.spring.demo1;
      
      import javax.annotation.Resource;
      
      import org.junit.Test;
      import org.junit.runner.RunWith;
      import org.springframework.test.context.ContextConfiguration;
      import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
      
      /**
       * Spring的AOP注解开发
       * @author 猪猪
       *
       */
      @RunWith(SpringJUnit4ClassRunner.class)
      @ContextConfiguration("classpath:applicationContext.xml")
      public class SpringDemo1 {
      	@Resource(name="orderDao")
      	private OrderDao dao;
      	
      	@Test
      	public void demo01(){
      		dao.save();
      		dao.find();
      		dao.update();
      		dao.delete();
      	}
      }
      

2)AOP注解的通知类型

  • @Before:前置通知

  • @AfterReturning:后置增强

    • 	@AfterReturning("execution(* com.nike.spring.demo1.OrderDao.delete(..))")
      	public void afterReturning(){
      		System.out.println("后置通知。。。");
          }
      
    	/**
    	 * 有返回值的后置增强
    	 */
    	@AfterReturning(value="execution(* com.nike.spring.demo1.OrderDao.delete(..))",returning="result")
    	public void afterReturning(Object result){
    		System.out.println("后置通知。。。"+result);
    	}
    
  • @Around:环绕增强

    • 	@Around(value="execution(* com.nike.spring.demo1.OrderDao.update(..))")
      	public Object around(ProceedingJoinPoint point) throws Throwable{
      		System.out.println("环绕前。。。。。。");
      		Object proceed = point.proceed();
      		System.out.println("环绕后。。。。。。");
      		return proceed;		
      	}
      
  • @AfterThrowing:异常抛出通知

    • 	@AfterThrowing(value="execution(* com.nike.spring.demo1.OrderDao.find(..))")
      	public void afterThrowing(){
      		System.out.println("异常抛出增强。。。。。");
      	}
      
    • 可以获取异常信息

    • 	@AfterThrowing(value="execution(* com.nike.spring.demo1.OrderDao.find(..))",throwing="ex")
      	public void afterThrowing(Throwable ex){
      		System.out.println("异常抛出增强。。。。。"+ex.getMessage());
      	}
      
  • @After:最终增强

    • 	@After(value="execution(* com.nike.spring.demo1.OrderDao.find(..))")
      	public void after(){
      		System.out.println("最终增强。。。。。。。");
      	}
      

3)AOP注解中的切入点注解

package com.nike.spring.demo1;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

/**
 * 使用注解开发切面类
 * @author 猪猪
 *
 */
@Aspect //告知spring这是一个切面类
public class MyAspectAnno {
	
	@Before(value="MyAspectAnno.pointcut2()")
	public void before(){
		System.out.println("前置通知。。。");
	}
	/**
	 * 有返回值的后置增强
	 */
	@AfterReturning(value="MyAspectAnno.pointcut4()",returning="result")
	public void afterReturning(Object result){
		System.out.println("后置通知。。。"+result);
	}
	
	@Around(value="MyAspectAnno.pointcut3()")
	public Object around(ProceedingJoinPoint point) throws Throwable{
		System.out.println("环绕前。。。。。。");
		Object proceed = point.proceed();
		System.out.println("环绕后。。。。。。");
		return proceed;		
	}
	
	@AfterThrowing(value="MyAspectAnno.pointcut1()",throwing="ex")
	public void afterThrowing(Throwable ex){
		System.out.println("异常抛出增强。。。。。"+ex.getMessage());
	}
	
	@After(value="MyAspectAnno.pointcut1()")
	public void after(){
		System.out.println("最终增强。。。。。。。");
	}
	
	//切入点注解
	@Pointcut(value="execution(* com.nike.spring.demo1.OrderDao.find(..))")
	private void pointcut1(){}
	@Pointcut(value="execution(* com.nike.spring.demo1.OrderDao.save(..))")
	private void pointcut2(){}
	@Pointcut(value="execution(* com.nike.spring.demo1.OrderDao.update(..))")
	private void pointcut3(){}
	@Pointcut(value="execution(* com.nike.spring.demo1.OrderDao.delete(..))")
	private void pointcut4(){}
}

2.Spring的JDBC模板的使用

Spring对持久层页提供了解决方案:ORM模块JdbcTemplate

Spring提供了很多持久层的模板简化编程:

ORM持久化技术模板类
JDBCorg.springframework.jdbc.core.JdbcTemplate
Hibernate3.0org.springframework.orm.hibernate3.HibernateTemplate
IBatis(MyBatis)org.springframework.orm.ibatis.SqlMapClientTemplate
JPAorg.springframework.orm.jpa.JpaTemplate

1)入门程序

  • a.创建项目,引入jar包
    • 引入基本开发包:6个

    • 引入数据库驱动jar包:mysql-connector-java-5.1.7-bin.jar

    • spring的jdbc模板jar包:

    • spring的单元测试的包:spring-test-4.2.4.RELEASE.jar

    • 在这里插入图片描述

  • b.创建数据库和表

    • CREATE DATABASE spring4_day03;
      USE spring4_day03;
      CREATE TABLE account(
      	id INT PRIMARY KEY AUTO_INCREMENT,
      	NAME VARCHAR(30),
      	money DOUBLE
      );
      
  • c.向数据库插入数据

    • 	/**
      	 * JdbcTemplate的使用类似于DbUtils
      	 */
      	@Test
      	public void demo01(){
      		//创建数据库连接池
      		DriverManagerDataSource dataSource = new DriverManagerDataSource();
      		dataSource.setDriverClassName("com.mysql.jdbc.Driver");
      		dataSource.setUrl("jdbc:mysql://localhost:3306/spring4_day03");
      		dataSource.setUsername("root");
      		dataSource.setPassword("971102");
      		//创建jdbc模板
      		JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
      		jdbcTemplate.update("insert into account values(null,?,?)", "王彬彬",800000);
      	}
      
  • d.将连接池和JdbcTemplate都交给spring管理

    • 创建spring的配置文件

    • <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"
      	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      	xmlns:context="http://www.springframework.org/schema/context"
      	xmlns:aop="http://www.springframework.org/schema/aop"
      	xmlns:tx="http://www.springframework.org/schema/tx"
      	xsi:schemaLocation="http://www.springframework.org/schema/beans 
      	http://www.springframework.org/schema/beans/spring-beans.xsd
      	http://www.springframework.org/schema/context
      	http://www.springframework.org/schema/context/spring-context.xsd
      	http://www.springframework.org/schema/aop
      	http://www.springframework.org/schema/aop/spring-aop.xsd
      	http://www.springframework.org/schema/tx 
      	http://www.springframework.org/schema/tx/spring-tx.xsd">
      	
      
      </beans> 
      
    • 配置spring的内置连接池

    • 	<!-- 配置spring的内置连接池 -->
      	<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
      		<!-- 属性注入 -->
      		<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
      		<property name="url" value="jdbc:mysql://localhost:3306/spring4_day03"/>
      		<property name="username" value="root"/>
      		<property name="password" value="971102"/>
      	</bean>
      	<!-- 配置JdbcTemplate -->
      	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
      		<property name="dataSource" ref="dataSource"/>
      	</bean>
      
    • 编写测试类

    • package com.nike.spring.jdbc.demo1;
      
      import javax.annotation.Resource;
      
      import org.junit.Test;
      import org.junit.runner.RunWith;
      import org.springframework.jdbc.core.JdbcTemplate;
      import org.springframework.test.context.ContextConfiguration;
      import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
      
      @RunWith(SpringJUnit4ClassRunner.class)
      @ContextConfiguration("classpath:applicationContext.xml")
      public class JdbcDemo2 {
      	@Resource(name="jdbcTemplate")
      	private JdbcTemplate template;
      
      	@Test
      	public void demo01(){
      		template.update("insert into account values(null,?,?)", "李世民",465451);
      	}
      }
      
  • 使用DBCP连接池

    • 导入DBCP的jar包

    • 在这里插入图片描述

    • 配置DBCP连接池

    • 	<!-- 配置DBCP连接池 -->
      	<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
      		<!-- 注入属性 -->
      		<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
      		<property name="url" value="jdbc:mysql://localhost:3306/spring4_day03"/>
      		<property name="username" value="root"/>
      		<property name="password" value="971102"/>
      	</bean>
      	<!-- 配置JdbcTemplate -->
      	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
      		<property name="dataSource" ref="dataSource"/>
      	</bean>
      
    • 编写测试类

    • 	@Test
      	public void demo02(){
      		template.update("insert into account values(null,?,?)", "薛平贵",154451);
      	}
      
  • 使用C3P0数据库连接池

    • 导入C3P0的jar包

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

    • 配置C3P0连接池

    • 	<!-- 配置C3P0连接池 -->
      	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
      		<!-- 属性注入 -->
      		<property name="driverClass" value="com.mysql.jdbc.Driver"/>
      		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/spring4_day03"/>
      		<property name="user" value="root"/>
      		<property name="password" value="971102"/>
      	</bean>
      	<!-- 配置JdbcTemplate -->
      	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
      		<property name="dataSource" ref="dataSource"/>
      	</bean>
      
    • 引入外部属性文件jdbc.properties

    • jdbc.driverClass=com.mysql.jdbc.Driver
      jdbc.jdbcUrl=jdbc:mysql://localhost:3306/spring4_day03
      jdbc.user=root
      jdbc.password=971102
      
    • 在配置文件中引入配置文件:2种方式

    • 第一种:通过一个bean标签引入(很少使用)

    • 	<!-- 引入属性文件 -->
      	<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
      		<property name="location" value="classpath:jdbc.properties"/>
      	</bean>
      
    • 第二种:通过context标签引入属性文件

    • <context:property-placeholder location="classpath:jdbc.properties"/>
      
    • 属性文件的使用

    • 	<!-- 配置C3P0连接池 -->
      	<!-- 引入属性文件 -->
      	<context:property-placeholder location="classpath:jdbc.properties"/>
      	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
      		<!-- 属性注入 -->
      		<property name="driverClass" value="${jdbc.driverClass}"/>
      		<property name="jdbcUrl" value="${jdbc.jdbcUrl}"/>
      		<property name="user" value="${jdbc.user}"/>
      		<property name="password" value="${jdbc.password}"/>
      	</bean>
      	<!-- 配置JdbcTemplate -->
      	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
      		<property name="dataSource" ref="dataSource"/>
      	</bean>
      
    • 模板的crud操作

    • package com.nike.spring.demo1;
      
      import java.sql.ResultSet;
      import java.sql.SQLException;
      import java.util.List;
      
      import javax.annotation.Resource;
      
      import org.junit.Test;
      import org.junit.runner.RunWith;
      import org.springframework.jdbc.core.JdbcTemplate;
      import org.springframework.jdbc.core.RowMapper;
      import org.springframework.test.context.ContextConfiguration;
      import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
      
      import com.nike.spring.domain.Account;
      
      @RunWith(SpringJUnit4ClassRunner.class)
      @ContextConfiguration("classpath:applicationContext.xml")
      public class JdbcDemo2 {
      	@Resource(name="jdbcTemplate")
      	private JdbcTemplate template;
      	
      	//增
      	@Test
      	public void demo01(){
      		template.update("insert into account values(null,?,?)", "猪八戒",54451);
      	}
      	
      	//改
      	@Test
      	public void demo02(){
      		template.update("update account set money=? where id=?",100,1);
      	}
      	
      	//删
      	@Test
      	public void demo03(){
      		template.update("delete from account where id=?",5);
      	}
      	
      	//查询单个用户单个字段值
      	@Test
      	public void demo04(){
      		String name = template.queryForObject("select name from account where id=?",String.class,4);
      		System.out.println(name);
      	}
      	//统计个数
      	@Test
      	public void demo05(){
      		Long num = template.queryForObject("select count(1) from account",Long.class);
      		System.out.println(num);
      	}
      	
      	@Test
      	//查询单个对象
      	public void demo06(){
      		Account user = template.queryForObject("select id,name,money from account where id=?",new RowMapper<Account>(){
      
      			@Override
      			public Account mapRow(ResultSet rs, int rowNum) throws SQLException {
      				Account user = new Account();
      				user.setId(rs.getInt("id"));
      				user.setName(rs.getString("name"));
      				user.setMoney(rs.getDouble("money"));
      				return user;
      			}} ,2);
      		System.out.println(user);
      	}
      	
      	@Test
      	//查询多个用户
      	public void demo07(){
      		List<Account> query = template.query("select * from account",new RowMapper<Account>(){
      
      			@Override
      			public Account mapRow(ResultSet rs, int arg1) throws SQLException {
      				Account user = new Account();
      				user.setId(rs.getInt("id"));
      				user.setName(rs.getString("name"));
      				user.setMoney(rs.getDouble("money"));
      				return user;
      			}
      		
      		});
      		System.out.println(query);
      	}
      }
      

3.Spring的事务管理

a.事务

  • 事务的概念
    • 逻辑上的一组操作,组成这组操作的各个单元,要么全都成功,要么全都失败;
  • 事务的特性
    • 原子性:事务不可分割
    • 一致性:事务执行前后数据的完整性要保持一致
    • 隔离性:一个事务的执行不应该受到其他事务的干扰
    • 持久性:一个事务结束,数据就持久化到数据库
  • 如果不考虑事务的隔离性,就会引发安全性问题
    • 读问题
      • 脏读:一个事务读到另一个事务未提交的数据
      • 幻读:一个事务读到另一个事务已经提交的insert的数据,导致一个事务中的多次查询结果不一致
      • 不可重复读:一个事务读到另一个事务已经提交的update的数据,导致一个事务中的多次查询结果不一致
    • 解决读问题(设置事务的隔离级别)
      • Read uncommitted(读未提交):不能解决任何读问题,会引发脏读问题
      • Read committed(读已提交):可解决脏读问题
      • Repeatable read(可重复读):可解决脏读,不可重复读问题
      • Serializable(可串行化):可解决所有读问题,不允许事务并发
    • 写问题
      • 丢失更新

b.spring提供了两大类的事务管理方式(编程式事务,声明式事务)

  • 事务管理API
    • PlatformTransactionManager:平台事务管理器
      • 接口,spring用于管理事务的真正的对象,已知实现类有:DataSourceTransactionManagerHibernateTransactionManager
    • TransactionDefinition:事务定义信息
      • 用于定义事务相关信息,比如隔离级别,超时信息,传播行为,是否只读。
    • TransactionStatus:事务状态
      • 用于记录在事务管理过程中事务的状态的对象
  • 事务管理API的关系:
    • spring进行事务管理的时候,首先平台事务管理器会根据事务定义信息来进行事务管理,在事务管理过程中,就会产生各种状态,将这些状态的信息记录到事务状态的对象中
  • 事务的传播行为:spring中提供了7种事务传播行为
  • 如果遇到特别复杂的业务逻辑,有可能出现业务层之间的方法相互调用。
  • 事务的传播行为主要用来解决业务层方法相互调用的问题。
  • 7种传播行为分成3类:
    • 保证多个操作在同一个事务中
      • PROPAGATION_REQUIRED:Ⓜ️默认值,如果A中有事务,使用A中的事务,如果A中没有事务,创建一个新的事务,将操作包含到事务中。
      • PROPAGATION_SUPPORTS:支持事务,如果A中有事务,就使用A中的事务,如果A中没有事务,就不使用事务。
      • PROPAGATION_MANDATORY:如果A中有事务,就是用A中的事务,如果A中没有事务,抛出异常,不执行。
    • 保证多个操作不在同一个事务中
      • PROPAGATION_REQUIRES_NEW:如果A中有事务,就会将A中的事务挂起,创建新的事务,只包含自身的操作,如果A中没有事务,创建一个新事务,包含自身的操作。
      • PROPAGATION_NOT_SUPPORTED:如果A中有事务,就会将A中的事务挂起。不使用事务管理。
      • PROPAGATION_NEVER:如果A中有事务,直接报异常。
    • 嵌套式事务
      • PROPAGATION_NESTED:如果A中有事务,按照A的事务执行,执行完成后,设置一个保存点,执行B中的操作,如果没有异常,则执行通过,如果有异常,可以选择回滚到最初始的位置,也可以回滚到保存点。

c.转账环境的搭建

  • 创建了业务层的接口和实现类

  • 创建了dao层的接口和实现类

  • 配置service和dao,交给spring管理

  • 在DAO中编写账户转入转出的方法

  • 	<!-- 配置service -->
    	<bean id="accountService" class="com.nike.tx.demo1.AccountServiceImpl">
    		<property name="accountDao" ref="accountDao"></property>
    	</bean>
    	<!-- 配置dao -->
    	<bean id="accountDao" class="com.nike.tx.demo1.AccountDaoImpl">
    		<property name="dataSource" ref="dataSource"/>
    	</bean>
    	
    	<!-- 配置连接池和jdbc模板 -->
    		<!-- 配置C3P0连接池 -->
    	<!-- 引入属性文件 -->
    	<context:property-placeholder location="classpath:jdbc.properties"/>
    	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    		<!-- 属性注入 -->
    		<property name="driverClass" value="${jdbc.driverClass}"/>
    		<property name="jdbcUrl" value="${jdbc.jdbcUrl}"/>
    		<property name="user" value="${jdbc.user}"/>
    		<property name="password" value="${jdbc.password}"/>
    	</bean>
    
package com.nike.tx.demo1;

import org.springframework.jdbc.core.support.JdbcDaoSupport;

public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {

	@Override
	public void outMoney(String from, Double money) {
		this.getJdbcTemplate().update("update account set money=money-? where name=?",money,from);
	}

	@Override
	public void inMoney(String to, Double money) {
		this.getJdbcTemplate().update("update account set money=money+? where name=?",money,to);
	}
}
package com.nike.tx.demo1;
/**
 * 转账Dao接口
 * @author 猪猪
 */
public interface AccountDao {
	/**
	 * 账户转出
	 * @param from 转出账户
	 * @param money 转账金额
	 */
	void outMoney(String from,Double money);
	
	/**
	 * 账户转入
	 * @param to 转入账户
	 * @param money 转账金额
	 */
	void inMoney(String to,Double money);
}
package com.nike.tx.demo1;

public class AccountServiceImpl implements AccountService{
	
	//注入Dao
	private AccountDao accountDao;
	
	public void setAccountDao(AccountDao accountDao) {
		this.accountDao = accountDao;
	}

	@Override
	public void transfer(String from, String to, Double money) {
		accountDao.outMoney(from, money);
		accountDao.inMoney(to, money);
	}

}
package com.nike.tx.demo1;
/**
 * 转账的业务层接口
 * @author 猪猪
 */
public interface AccountService {
	/**
	 * 转账的方法
	 * @param from 转出账号
	 * @param to 转入账号
	 * @param money 转账金额
	 */
	void transfer(String from,String to,Double money);
}

d.进行测试

编程式事务管理

(1)配置平台事务管理器

	<!-- 配置平台事务管理器 -->
	<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<property name="dataSource" ref="dataSource"></property>
	</bean>

(2)spring提供了事务的模板类

配置事务的管理的模板类

	<!-- 配置事务管理的模板 -->
	<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
		<property name="transactionManager" ref="transactionManager"></property>
	</bean>

(3)在业务层注入事务的管理模板

	<!-- 配置service -->
	<bean id="accountService" class="com.nike.tx.demo1.AccountServiceImpl">
		<property name="accountDao" ref="accountDao"/>
		<!-- 注入事务管理的模板 -->
		<property name="template" ref="transactionTemplate"/>
	</bean>

(4)编写事务管理的代码

package com.nike.tx.demo1;

import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;

public class AccountServiceImpl implements AccountService{
	//注入事务管理的模板
	private TransactionTemplate template;
	//注入Dao
	private AccountDao accountDao;
	
	public void setTemplate(TransactionTemplate template) {
		this.template = template;
	}

	public void setAccountDao(AccountDao accountDao) {
		this.accountDao = accountDao;
	}

	@Override
	public void transfer(String from, String to, Double money) {
		template.execute(new TransactionCallbackWithoutResult() {
			
			@Override
			protected void doInTransactionWithoutResult(TransactionStatus arg0) {
				accountDao.outMoney(from, money);
				int i = 1/0;
				accountDao.inMoney(to, money);
				
			}
		});
	}
}

e.声明式事务–AOP

  • 声明式事务xml

    • 引入AOP开发包
    • 回复转账环境
    • 配置声明式事务
  • 声明式事务注解

  • <!-- 配置平台事务管理器 -->
    	<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    
  • 配置增强(切面)

  • 	<!-- 配置事务的增强 -->
    	<tx:advice transaction-manager="transactionManager">
    		<tx:attributes>
    			<!-- 事务管理规则 -->
    			<tx:method name="transfer" propagation="REQUIRED"/>
    		</tx:attributes>
    	</tx:advice>
    
  • AOP配置把增强应用到类上

  • 	<!-- AOP的配置 -->
    	<aop:config>
    		<aop:pointcut expression="execution(* com.nike.tx.demo2.AccountServiceImpl.*(..))" id="pc1"/>
    		<aop:advisor advice-ref="txAdvice" pointcut-ref="pc1"/>
    	</aop:config>
    

声明式事务–注解

	<!-- 配置平台事务管理器 -->
	<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<property name="dataSource" ref="dataSource"></property>
	</bean>

开启注解事务

	<!-- 开启注解事务 -->
	<tx:annotation-driven transaction-manager="transactionManager"/>

在业务层上添加事务管理

@Transactional
public class AccountServiceImpl implements AccountService{
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Cloud是一个分布式微服务架构下的一站式解决方案,它是各个微服务架构落地技术的集合体,被称为微服务全家桶。\[2\]Spring Cloud集成了Ribbon和Eureka,可以在使用Feign时提供负载均衡的HTTP客户端。\[1\] 与此相比,Spring Boot专注于快速方便地开发单个个体微服务。它可以独立使用开发项目,而Spring Cloud则是关注全局的微服务协调治理框架,将使用Spring Boot开发的单体微服务整合并管理起来,为各个微服务之间提供配置发现、服务发现、路由、微代理、时间总线、全局锁、决策竞选、分布式会话等集成服务。可以说,Spring Boot可以离开Spring Cloud单独使用开发项目,但是Spring Cloud离不开Spring Boot,它们之间存在依赖关系。\[2\] 相比之下,Dubbo和Spring Cloud有一些区别。Spring Cloud利用Spring Boot的开发便利性,简化了分布式系统基础设施的开发,为开发人员提供了快速构建分布式系统的工具,包括配置管理、服务发现、断路器、路由、微代理、事件总线、全局锁、决策竞选、分布式会话等等。而Dubbo则是阿里巴巴开源的一款高性能Java RPC框架,主要用于提供分布式服务治理的解决方案。\[3\] #### 引用[.reference_title] - *1* *2* *3* [SpringCloud详解](https://blog.csdn.net/m0_56116754/article/details/126387535)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v4^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值