一. IOC 和 DI
IOC : 控制反转,将对象的创建权反转给了 Spring。
DI : 依赖注入,前提是必须要有 IOC 的环境,Spring 管理这个类的时候将类的依赖的属性注入(设置)进来。
二. 工厂类
// 1. 加载类路径
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
// 2. 加载磁盘中的 文件路径
ApplicationContext applicationContext = new FileSystemXmlApplicationContext("C:/Users/Administrator/Desktop/applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
三. Bean 的作用范围配置 (xml 文件中)
1. scope : Bean 的作用范围。
! - singleton : 默认,Spring 会采用单例模式创建这个对象。
! - prototype : 多例模式。(Struts2 和 Spring 整合时会用到)
- request : 应用在 Web 项目中,Spring 创建这个类以后,将这个类存入到 request 范围中。
- session : 应用在 Web 项目中,Spring 创建这个类以后,将这个类存入到 session 范围中。
- globalsession : 应用在 Web 项目中,必须在 porlet 环境下使用,如果没有这个环境,相对于 session。
四. Spring 的属性注入方式(xml)
1. 构造方法
// 构造方法 (需要提供构造器)
<!-- 1. 构造方法 -->
<bean id="shop" class="com.q.spring.demo2.ShopDao">
<constructor-arg name="name" value="apple" />
<constructor-arg name="price" value="100" />
</bean>
@Test
public void demo1(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
ShopDao shopDao = (ShopDao) applicationContext.getBean("shop");
System.out.println(shopDao);
}
2. set 方式的属性注入
<!-- 2. set 方法 -->
<bean id="shop2" class="com.q.spring.demo2.ShopDao2">
<property name="name" value="pear" />
<property name="price" value="22" />
</bean>
<!-- 2. set 方法 (p名称空间属性注入) -->
<bean id="shop2" class="com.q.spring.demo2.ShopDao2" p:name="q" p:price="2222"></bean>
<!-- 2. set 方法 (SpEL 属性注入) -->
<bean id="shop2" class="com.q.spring.demo2.ShopDao2">
<property name="name" value="#{'q1'}"></property>
<property name="price" value="#{111}"></property>
</bean>
// set 方式的属性注入(需要提供 set 方法)
@Test
public void demo2(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
ShopDao2 shopDao = (ShopDao2) applicationContext.getBean("shop2");
System.out.println(shopDao);
}
3. set 方法注入对象类型 (属性中依赖类)
<!-- 3. set 方法注入对象类型的属性 -->
<bean id="person" class="com.q.spring.demo2.Person">
<property name="name" value="q" />
<property name="shopDao2" ref="shop2" />
</bean>
<!-- 3. set 方法注入对象类型的属性 (p名称空间属性注入) -->
<bean id="person" class="com.q.spring.demo2.Person" p:name="q1" p:shopDao2-ref="shop2"></bean>
<!-- 3. set 方法注入对象类型的属性 (SpEL 属性注入) -->
<bean id="person" class="com.q.spring.demo2.Person">
<property name="name" value="#{'q1'}"></property>
<property name="shopDao2" value="#{shop2}"></property>
</bean>
// set 方法注入对象类型(需要提供 set 方法)
@Test
public void demo3(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Person person = (Person) applicationContext.getBean("person");
System.out.println(person);
}
4. p名称空间属性注入
<bean id="person" class="com.q.spring.demo2.Person" p:name="q1" p:shopDao2-ref="shop2"></bean>
5. SpEL 属性注入 (Spring 3.0 之后)
<bean id="shop2" class="com.q.spring.demo2.ShopDao2">
<property name="name" value="#{'q1'}"></property>
<property name="price" value="#{111}"></property>
</bean>
6. 集合类型的注入 ( arrs list set map )
private String[] arrs;
private List<String> list;
private Set<String> set;
private Map<String, String> map;
<bean id="typealls" class="com.q.spring.demo2.typeAlls">
<!-- 数组类型 -->
<property name="arrs">
<list>
<value>a</value>
<value>b</value>
<value>c</value>
</list>
</property>
<!-- list集合 -->
<property name="list">
<list>
<value>a1</value>
<value>b1</value>
<value>c1</value>
</list>
</property>
<!-- set集合 -->
<property name="set">
<set>
<value>a1</value>
<value>b1</value>
<value>c1</value>
</set>
</property>
<!-- map 集合 -->
<property name="map">
<map>
<entry key="b2" value="b2" />
<entry key="b2" value="b2" />
<entry key="b2" value="b2" />
</map>
</property>
</bean>
五. 分模块开发的配置
1. 直接在一个xml文件中导入
<import resource="applicationContext.xml" />
2. 在创建 ClassPathXmlApplicationContext 时可传多个参数路径
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml1","applicationContext.xml2");
六. Spring 的 IOC 注解开发
1. 引入约束: 使用注解开发引入 Context 约束
<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" 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">
</beans>
2. 开启 Spring 的组件扫描
<!-- 配置组件扫描(哪些包下的类使用IOC注解) -->
<context:component-scan base-package="包路径"></context:component-scan>
3. 在类上添加注解
- @Component : 组件
- 装饰一个类,将这个类交给 Spring 管理。
- 这个注解有三个衍生注解: - @Controller : web 层
- @Service : service 层
- @Repository : dao 层
import org.springframework.stereotype.Component;
// value 可省略: @Component("shopDao")
@Component(value = "userDao") // 相当于在 <bean id="userDao" class="com.q.spring.demo2.userDemo2" />
public class userDemo2 implements UserDao {
public void save(){
}
}
4. 注解方式设置属性值
注解方式:使用注解方式,可以没有 set 方法。
- 属性如果有 set 方法,需要将属性注入的注解加到 set 方法上。
// 如:
@Value("添加注解")
public void setName(String name){
this.name = name;
}
- 属性如果没有 set 方法,需要将属性注入的注解添加到属性上。
// 如:
@Value("添加注解")
private String name;
5. 属性注入的注解
# 普通属性:
@Value : 设置普通属性的值
# 对象类型属性:
@Autowired : 设置对象类型的属性值。但是按照类型完成属性注入(按照类名的一致)。
- 可以通过 @Qualifier 和 @Autowired 配合使用来修改为按照名称属性注入。
@Resource : 完成对象类型的属性注入,按照名称完成属性注入。 (常用)。
6. Bean 的其他注解
# 生命周期相关注解:
@PostConstruct : 初始化方法
@PreDestroy : 销毁方法
# Bean 作用范围的注解:
@scope : 作用范围
参数:
- singleton : 默认单例
- prototype : 多例
七. Spring 的 AOP 的 XML 开发
1. applocationContext.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:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 配置目标对象 :被增强的对象 -->
<bean id="productDao" class="com.q.spring.demo1.ProductDaoImpl" />
<!-- 将切面类交给 Spring 管理 -->
<bean id="myAspect" class="com.q.spring.demo1.MyAspectXml" />
<aop:config>
<!-- 通过AOP的配置完成对目标类产生代理 -->
<aop:pointcut expression="execution(* com.q.spring.demo1.ProductDaoImpl.save(..))" id="pointcut1" />
<aop:pointcut expression="execution(* com.q.spring.demo1.ProductDaoImpl.delete(..))" id="pointcut2" />
<aop:pointcut expression="execution(* com.q.spring.demo1.ProductDaoImpl.update(..))" id="pointcut3" />
<aop:pointcut expression="execution(* com.q.spring.demo1.ProductDaoImpl.find(..))" id="pointcut4" />
<!-- 配置切面 -->
<aop:aspect ref="myAspect">
<!-- 前置通知:获得切入点信息 -->
<aop:before method="checkPri" pointcut-ref="pointcut1" />
<!-- 后置通知 -->
<aop:after-returning method="writeLog" pointcut-ref="pointcut2" returning="result" />
<!-- 环绕通知 -->
<aop:around method="around" pointcut-ref="pointcut3" />
<!-- 异常抛出通知 -->
<aop:after-throwing method="afterThrowing" pointcut-ref="pointcut4" throwing="ex" />
<!-- 最终通知 -->
<aop:after method="after" pointcut-ref="pointcut4" />
</aop:aspect>
</aop:config>
</beans>
2. ProductDao.java 接口文件
package com.q.spring.demo1;
public interface ProductDao {
public void save();
public void update();
public String delete();
public void find();
}
3. ProductDaoImpl.java DAO文件
package com.q.spring.demo1;
public class ProductDaoImpl implements ProductDao {
@Override
public void save() {
System.out.println("save");
}
@Override
public void update() {
System.out.println("update");
}
@Override
public String delete() {
System.out.println("delete");
return "Delete!!!";
}
@Override
public void find() {
System.out.println("find");
// int i = 1/0;
}
}
4. MyAspectXML.java
package com.q.spring.demo1;
// 切面类
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
public class MyAspectXml {
// 前置通知
public void checkPri(JoinPoint joinpoint){
System.out.println("权限校验 "+ joinpoint);
}
// 后置通知
public void writeLog(Object result){
System.out.println("日志记录 "+ result);
}
// 环绕通知 : 性能监控
public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
System.out.println("环绕前通知...");
Object obj = joinPoint.proceed();
System.out.println("环绕后通知...");
return obj;
}
// 异常抛出通知
public void afterThrowing(Throwable ex){
System.out.println("异常抛出通知..."+ ex.getMessage());
}
// 最终通知:类似于 finally
public void after(){
System.out.println("最终通知...");
}
}
5. 测试文件
package com.q.spring.demo1;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringDemo1 {
@Resource(name="productDao")
private ProductDao productDao;
@Test
public void demo1(){
productDao.save();
productDao.update();
productDao.delete();
productDao.find();
}
}
八. Srping 的 AOP 注解
1. applocationContext.xml 文件
<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"
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">
<!-- 在配置文件中开启注解的AOP开发 -->
<aop:aspectj-autoproxy />
<!-- 配置目标类 -->
<bean id="orderDao" class="com.q.spring.demo1.OrderDao" />
<!-- 配置切面类 -->
<bean id="myAspect" class="com.q.spring.demo1.MyAspectAnno" />
</beans>
2. OrderDao.java 文件
package com.q.spring.demo1;
public class OrderDao {
public void save(){
System.out.println("save...");
}
public void update(){
System.out.println("update...");
}
public String delete(){
System.out.println("delete...");
return "Delete!!!";
}
public void find(){
System.out.println("find...");
// int i = 1/0;
}
}
3. MyAspectAnno.java 文件
package com.q.spring.demo1;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
@Aspect
public class MyAspectAnno {
// 前置通知
// execution(* com.q.spring.demo1.OrderDao.save(..))
@Before(value = "MyAspectAnno.pointcut1()")
public void befor(){
System.out.println("前置增强");
}
// 后置通知
@AfterReturning(value = "MyAspectAnno.pointcut3()", returning = "result")
public void afterReturning(Object result){
System.out.println("后置增强 "+ result);
}
// 环绕通知
@Around(value = "MyAspectAnno.pointcut2()")
public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
System.out.println("环绕前增强");
Object obj = joinPoint.proceed();
System.out.println("环绕后增强");
return obj;
}
// 异常抛出通知
@AfterThrowing(value = "MyAspectAnno.pointcut4()", throwing = "e")
public void afterThrowing(Throwable e){
System.out.println("异常抛出增强: "+e);
}
// 最终通知
@After(value = "MyAspectAnno.pointcut4()")
public void afterThrowing(){
System.out.println("最终增强");
}
// 切入点注解
@Pointcut(value = "execution(* com.q.spring.demo1.OrderDao.save(..))")
private void pointcut1(){}
@Pointcut(value = "execution(* com.q.spring.demo1.OrderDao.update(..))")
private void pointcut2(){}
@Pointcut(value = "execution(* com.q.spring.demo1.OrderDao.delete(..))")
private void pointcut3(){}
@Pointcut(value = "execution(* com.q.spring.demo1.OrderDao.find(..))")
private void pointcut4(){}
}
4. SpringDemo1.java 文件
package com.q.spring.demo1;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringDemo1 {
@Resource(name = "orderDao")
private OrderDao orderDao;
@Test
public void demo1(){
orderDao.save();
orderDao.update();
orderDao.delete();
orderDao.find();
}
}
九. Spring 的 JDBC 模板
Spring 是一站式框架,有持久层解决方案
1. DBCP 连接池
# 文档..
2. C3P0 连接池
# 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"
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">
<!-- 配置C3P0连接池 -->
<!-- <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">-->
<!-- <property name="driverClass" value="com.mysql.jdbc.Driver" />-->
<!-- <property name="jdbcUrl" value="jdbc:mysql:///spring_jdbc" />-->
<!-- <property name="user" value="root" />-->
<!-- <property name="password" value="chaoqi" />-->
<!-- </bean>-->
<!-- 引入属性文件 -->
<context:property-placeholder location="classpath:jdbc.properties" />
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}" />
<property name="jdbcUrl" value="${jdbc.url}" />
<property name="user" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
</bean>
<!-- 配置 Spring 的JDBC的模板 -->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource" />
</bean>
</beans>
# 增删改查操作
package com.q.jdbc.demo1;
import com.q.jdbc.domain.Account;
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 javax.annotation.Resource;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.List;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class JdbcDemo2 {
@Resource(name = "jdbcTemplate")
private JdbcTemplate jdbcTemplate;
// 增
@Test
public void demo1(){
jdbcTemplate.update("insert into account values(null,?,?)","q4",152d);
}
// 改
@Test
public void demo2(){
jdbcTemplate.update("update account set name=?, money = ? where id=? ","q1test",11d,2 );
}
// 删
@Test
public void demo3(){
jdbcTemplate.update("delete from account where id=? ",2);
}
// 查
@Test
public void demo4(){
String name = jdbcTemplate.queryForObject("select name from account where id=?",String.class,3);
System.out.println(name);
}
// 统计查询
@Test
public void demo5(){
Long count = jdbcTemplate.queryForObject("select count(*) from account",Long.class);
System.out.println(count);
}
// 封装到一个对象中
@Test
public void demo6(){
Account account = jdbcTemplate.queryForObject("select * from account where id = ?", new MyRowMapper(), 3);
System.out.println(account);
}
// 查询多条记录
@Test
public void demo7(){
List<Account> list = jdbcTemplate.query("select * from account", new MyRowMapper());
for(Account account : list){
System.out.println(account);
}
}
class MyRowMapper implements RowMapper<Account>{
@Override
public Account mapRow(ResultSet res, int rowNum) throws SQLException {
Account account = new Account();
account.setId(res.getInt("id"));
account.setName(res.getString("name"));
account.setMoney(res.getDouble("money"));
return account;
}
}
}
十. Spring 的事务传播行为
1. 报则保证多个操作在同一个事务中
PROPAGATION_REQUIRED : 默认值,如果A中有事务,使用A中的事务,如果A没有,创建一个新的事务,将操作包含进来。
PROPAGATION_SUPPORTS : 支持事务
,如果A中有事务,使用A中的事务,如果没有,则不使用事务。
PROPAGATION_MANDATORY : 如果A中有事务,使用A中的事务,如果A中没有事务,则抛异常。
2. 保证多个操作不在同一个事务中
PROPAGATION_REQUIRES_NEW : 如果A中有事务,将A的事务挂起(暂停), 创建新事务,只包含自身操作。
如果A中没有事务,则创建一个新事务,包含自身操作
PROPAGATION_NOT_SUPPORTED : 如果A中有事务,将A的事务挂起,不使用事务管理。
PROPAGATION_NEVER : 如果A中有事务,则抛异常。
3. 嵌套式事务
PROPAGATION_NESTED : 嵌套事务,如果A中有事务,按照A的事务执行,执行完毕后,设置一个保存点,执行B中的操作,如果没有异常,执行通过,如果有异常,可以选择回滚到最初始的位置,也可以回滚到保存点。
十一. Spring 声明式事务管理一 (XML)
XML 步骤:
1. 引入 aop 开发包。
2. 配置事务管理器。
3. 配置增强<tx:Advice>
4. 配置aop
# 1. 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/cache
http://www.springframework.org/schema/cache/spring-cache.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置 Service -->
<bean id="accountService" class="com.q.tx.demo2.AccountServiceImpl">
<property name="accountDao" ref="accountDao" />
</bean>
<!-- 配置 DAO -->
<bean id="accountDao" class="com.q.tx.demo2.AccountDaoImpl">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 配置连接池的JDBC模板 -->
<!-- 引入属性文件 -->
<context:property-placeholder location="classpath:jdbc.properties" />
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}" />
<property name="jdbcUrl" value="${jdbc.url}" />
<property name="user" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
</bean>
<!-- 配置事务管理 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 配置事务的增强 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" propagation="REQUIRED" />
<!-- 实际开发中: -->
<!-- <tx:method name="save*" propagation="REQUIRED" isolation="DEFAULT" />-->
<!-- <tx:method name="update*" propagation="REQUIRED" />-->
<!-- <tx:method name="delete*" propagation="REQUIRED" />-->
<!-- <tx:method name="find*" read-only="true" />-->
</tx:attributes>
</tx:advice>
<!-- AOP的配置 -->
<aop:config>
<aop:pointcut id="pointcut1" expression="execution(* com.q.tx.demo2.AccountServiceImpl.*(..))"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1" />
</aop:config>
</beans>
# jdbc.propertie.xml 文件属性设置
jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql:///spring_jdbc
jdbc.username=root
jdbc.password=chaoqi
# 2. AccountDaoImpl.java 文件
package com.q.tx.demo2;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
// 转账Dao实现
public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
@Override
public void outMoney(Integer from, Double money) {
this.getJdbcTemplate().update("update account set money = money - ? where id = ?", money, from);
}
@Override
public void inMoney(Integer to, Double money) {
this.getJdbcTemplate().update("update account set money = money + ? where id = ?", money, to);
}
}
# 3. AccountServiceImpl.java 文件
package com.q.tx.demo2;
// 转账的业务层实现类
public class AccountServiceImpl implements AccountService {
// 注入 DAO
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
/*
from : 转出账号
to : 转人账号
money : 转账金额
*/
@Override
public void transfer(Integer from, Integer to, Double money) {
accountDao.outMoney(from, money);
// int i = 1/0;
accountDao.inMoney(to, money);
}
}
# 4. 测试文件
package com.q.tx.demo2;
// 测试类
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:tx2.xml")
public class SpringDemo1 {
@Resource(name = "accountService")
private AccountService accountService;
@Test
public void demo1(){
accountService.transfer(1,3,100d);
}
}
十二. Spring 声明式事务管理二 (注解方式)
注解 步骤:
1. 引入 aop 开发包。
2. 配置事务管理器。
3. 开启注解事务。
4. 在业务层添加注解
# 1. 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/cache
http://www.springframework.org/schema/cache/spring-cache.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置 Service -->
<bean id="accountService" class="com.q.tx.demo3.AccountServiceImpl">
<property name="accountDao" ref="accountDao" />
</bean>
<!-- 配置 DAO -->
<bean id="accountDao" class="com.q.tx.demo3.AccountDaoImpl">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 配置连接池的JDBC模板 -->
<!-- 引入属性文件 -->
<context:property-placeholder location="classpath:jdbc.properties" />
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}" />
<property name="jdbcUrl" value="${jdbc.url}" />
<property name="user" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
</bean>
<!-- 配置事务管理器 第一步 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 开启注解事务 第二步 -->
<tx:annotation-driven transaction-manager="transactionManager" />
</beans>
# 2. AccountDaoImpl 文件
package com.q.tx.demo3;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
// 转账Dao实现
public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
@Override
public void outMoney(Integer from, Double money) {
this.getJdbcTemplate().update("update account set money = money - ? where id = ?", money, from);
}
@Override
public void inMoney(Integer to, Double money) {
this.getJdbcTemplate().update("update account set money = money + ? where id = ?", money, to);
}
}
# 3. AccountServiceImpl 文件
package com.q.tx.demo3;
import org.springframework.transaction.annotation.Transactional;
// 转账的业务层实现类
// 第三步 需要在要进行事务操作的类上加 @Transactional
@Transactional
public class AccountServiceImpl implements AccountService {
// 注入 DAO
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
/*
from : 转出账号
to : 转人账号
money : 转账金额
*/
@Override
public void transfer(Integer from, Integer to, Double money) {
accountDao.outMoney(from, money);
// int i = 1/0;
accountDao.inMoney(to, money);
}
}
# 4. 测试类
package com.q.tx.demo3;
// 测试类
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:tx3.xml")
public class SpringDemo1 {
@Resource(name = "accountService")
private AccountService accountService;
@Test
public void demo1(){
accountService.transfer(1,3,100d);
}
}