编程式事务管理:
配置文件:定义了名为transactionManager的事务管理器,使用dataSource作为数据源。最后,将dataSource、transactionManager和transactionTemplate注入到名为transactionExample的类中,以便在该类中使用它们
<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/how2java
</value>
</property>
<property name="username">
<value>root</value>
</property>
<property name="password">
<value>123456</value>
</property>
</bean>
<!-- 定义TransactionTemplate模板 -->
<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager">
<ref bean="transactionManager"/>
</property>
<property name="propagationBehaviorName">
<value>PROPAGATION_REQUIRED</value>
</property>
</bean>
<!-- 定义事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource">
<ref bean="dataSource" />
</property>
</bean>
<!-- 为TransactionExample注入数据源 、事务管理器、TransactionTemplate模板-->
<bean id="transactionExample"
class="com.mr.transaction.TransactionExample">
<property name="dataSource">
<ref bean="dataSource" />
</property>
<property name="transactionManager">
<ref bean="transactionManager" />
</property>
<property name="transactionTemplate">
<ref bean="transactionTemplate"/>
</property>
</bean>
TransactionExample(核心)类:
定义添加数据的方法,会对数据库进行两次插入操作,但在第二次插入前,有一行制造异常的代码,用于测试事务是否配置成功。如果事务配置成功,则两次插入操作都应该回滚,数据库中不会有任何记录。如果事务配置失败,则第一次插入操作会生效,数据库中会出现第一条记录。
public class TransactionExample {
DataSource dataSource;//注入数据源
PlatformTransactionManager transactionManager;//注入事务管理器
TransactionTemplate transactionTemplate;//注入TransactionTemplate模板
public DataSource getDataSource() {
return dataSource;
}
public void setDataSource(DataSource dataSource) {
this.dataSource = dataSource;
}
public PlatformTransactionManager getTransactionManager() {
return transactionManager;
}
public void setTransactionManager(PlatformTransactionManager transactionManager) {
this.transactionManager = transactionManager;
}
public TransactionTemplate getTransactionTemplate() {
return transactionTemplate;
}
public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
this.transactionTemplate = transactionTemplate;
}
public void transactionOperation() {
transactionTemplate.execute(new TransactionCallback() {
public Object doInTransaction(TransactionStatus status) {
Connection conn = DataSourceUtils.getConnection(dataSource);//获得数据库连接
try {
Statement stmt = conn.createStatement();
//执行两次添加方法
stmt.execute("insert into tb_user(name,age,sex) values('小强','26','男')");
int a=0;//制造异常测试事务是否配置成功
a=9/a;
stmt.execute("insert into tb_user(name,age,sex) values('小红','22','女')");
System.out.println("操作执行成功!");
} catch (Exception e) {
transactionManager.rollback(status);//事务回滚
System.out.println("操作执行失败,事务回滚!");
System.out.println("原因:"+e.getMessage());
}
return null;
测试类:从ApplicationContext对象中获取transactionExample Bean,并调用它的transactionOperation()方法
public class Manager {
public static void main(String[] args) {
ApplicationContext factory = new ClassPathXmlApplicationContext("applicationContext.xml"); //装载配置文件
TransactionExample transactionExample = (TransactionExample) factory.getBean("transactionExample");
transactionExample.transactionOperation();//执行添加方法
}
总结:编程式事务管理是一种通过编写代码来控制事务的方式,而不是通过配置文件来控制。在这种方式中,使用Spring框架提供的TransactionTemplate模板来执行一段代码块,并在这个代码块中进行数据库操作。如果这些操作成功,则事务将被提交,否则事务将被回滚。编程式事务管理需要在代码中显式地管理事务,包括开始、提交或回滚事务,以及处理异常情况。这种方式可以提供更精细的控制,但也需要更多的代码编写和维护工作。
声明式事务管理:
声明式事务管理是通过使用注解或XML配置来实现的,通过声明式的方式来管理事务的提交和回滚
配置文件:
定义了以下三个Bean:
-
dataSource:定义了一个数据源,用于连接数据库。
-
transactionManager:定义了一个事务管理器,用于管理数据库事务。
-
transactionProxy:定义了一个事务代理,用于在添加用户的方法中添加事务控制。
其中,设置了PROPAGATION_REQUIRED属性,表示当方法执行时如果当前存在事务,则加入该事务;如果当前没有事务,则新建一个事务,在第二次操作出现异常时,事务会自动回滚,保证数据的一致性
<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/how2java</value>
</property>
<property name="username">
<value>root</value>
</property>
<property name="password">
<value>123456</value>
</property>
</bean>
<!-- 定义事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource">
<ref bean="dataSource" />
</property>
</bean>
<!-- 定义TransactionProxy -->
<bean id="transactionProxy"
class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<property name="transactionManager">
<ref local="transactionManager" />
</property>
<property name="target">
<bean id="addDAO" class="com.mr.dao.AddDAO">
<property name="dataSource">
<ref local="dataSource" />
</property>
</bean>
</property>
<property name="proxyTargetClass" value="true" />
<property name="transactionAttributes">
<props>
<prop key="add*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
</beans>
AddDAO:
用于执行数据库操作。它封装了添加用户的操作,并提供了一个addUser方法,用于向数据库中添加用户。由于是使用声明式事务管理,因此在addUser方法中不需要编写事务控制的代码,Spring会自动为其添加事务控制逻辑。
public class AddDAO extends JdbcDaoSupport {
//添加用户的方法
public void addUser(User user){
//执行添加方法的sql语句
String sql="insert into tb_user (name,age,sex) values('" +
user.getName() + "','" + user.getAge()+ "','" + user.getSex()+ "')";
//执行两次添加方法
getJdbcTemplate().execute(sql);
int a=0;//制造异常测试事务是否配置成功
a=9/a;
getJdbcTemplate().execute(sql);
}
主类:
通过Spring容器获取了一个名为"transactionProxy"的Bean,该Bean是AddDAO类的代理对象,用于在addUser方法中添加事务控制逻辑。接着,实例化了一个User对象,并设置了姓名、年龄和性别等属性。最后,调用addDAO的addUser方法,将User对象添加到数据库中,由于该方法被事务代理包装,因此在方法执行过程中会自动添加事务控制逻辑,保证数据的一致性
public class Manager {
public static void main(String[] args) {
ApplicationContext factory = new ClassPathXmlApplicationContext("applicationContext.xml"); //装载配置文件
AddDAO addDAO = (AddDAO)factory.getBean("transactionProxy");//获取AddDAO
User user = new User();//实例化User实体对象
user.setName("111");//设置姓名
user.setAge(30);//设置年龄
user.setSex("1");//设置性别
addDAO.addUser(user);//执行数据库添加方法
运行结果和上述编程式事务管理一样,主要不同之处在于于实现方式。声明式事务管理通过配置文件来实现,而编程式事务管理需要在代码中显式地调用事务控制类