JdbcTemplate是spring框架中提供的一个模板对象,是对原始繁琐的Jdbc API对象的简单封装。java
JdbcTemplate jdbcTemplate = new JdbcTemplate(DataSource dataSource);
int update(); 执行增、删、改语句 List<T> query(); 查询多个 T queryForObject(); 查询一个 new BeanPropertyRowMapper<>(); 实现ORM映射封装
查询数据库全部帐户信息到Account实体中mysql
public class JdbcTemplateTest { @Test public void testFindAll() throws Exception { // 建立核心对象 JdbcTemplate jdbcTemplate = new JdbcTemplate(JdbcUtils.getDataSource()); // 编写sql String sql = "select * from account"; // 执行sql List<Account> list = jdbcTemplate.query(sql, new BeanPropertyRowMapper<>(Account.class)); } }
基于Spring的xml配置实现帐户的CRUD案例web
1. 建立java项目,导入坐标 2. 编写Account实体类 3. 编写AccountDao接口和实现类 4. 编写AccountService接口和实现类 5. 编写spring核心配置文件 6. 编写测试代码
<dependencies> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.47</version> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.15</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.8.13</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>5.1.5.RELEASE</version> </dependency> </dependencies>
public class Account { private Integer id; private String name; private Double money; }
public interface AccountDao { public List<Account> findAll(); public Account findById(Integer id); public void save(Account account); public void update(Account account); public void delete(Integer id); }
@Repository public class AccountDaoImpl implements AccountDao { @Autowired private JdbcTemplate jdbcTemplate; @Override public List<Account> findAll() { // 编写sql String sql = "select * from account"; // 执行sql return jdbcTemplate.query(sql, new BeanPropertyRowMapper<>(Account.class)); } @Override public Account findById(Integer id) { // 编写sql String sql = "select * from account where id = ?"; // 执行sql return jdbcTemplate.queryForObject(sql, new BeanPropertyRowMapper<>(Account.class),id); } @Override public void save(Account account) { // 编写sql String sql = "insert into account values(null,?,?)"; // 执行sql jdbcTemplate.update(sql, account.getName(), account.getMoney()); } @Override public void update(Account account) { // 编写sql String sql = "update account set name = ?,money = ? where id = ?"; // 执行sql jdbcTemplate.update(sql, account.getName(), account.getMoney(),account.getId()); } @Override public void delete(Integer id) { // 编写sql String sql = "delete from account where id = ?"; // 执行sql jdbcTemplate.update(sql, id); } }
public interface AccountService { public List<Account> findAll(); public Account findById(Integer id); public void save(Account account); public void update(Account account); public void delete(Integer id); }
@Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao; @Override public List<Account> findAll() { return accountDao.findAll(); } @Override public Account findById(Integer id) { return accountDao.findById(id); } @Override public void save(Account account) { accountDao.save(account); } @Override public void update(Account account) { accountDao.update(account); } @Override public void delete(Integer id) { accountDao.delete(id); } }
<?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" 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"> <context:component-scan base-package="com.lagou"/> <context:property-placeholder location="classpath:jdbc.properties"/> <bean > <property name="driverClassName" value="${jdbc.driver}"></property> <property name="url" value="${jdbc.url}"></property> <property name="username" value="${jdbc.username}"></property> <property name="password" value="${jdbc.password}"></property> </bean> <bean > <constructor-arg name="dataSource" ref="dataSource"></constructor-arg> </bean> </beans>
@RunWith(SpringJUnit4Cla***unner.class) @ContextConfiguration("classpath:applicationContext.xml") public class AccountServiceTest { @Autowired private AccountService accountService; //测试保存 @Test public void testSave() { Account account = new Account(); account.setName("lucy"); account.setMoney(100d); accountService.save(account); } //测试查询 @Test public void testFindById() { Account account = accountService.findById(3); System.out.println(account); } //测试查询全部 @Test public void testFindAll() { List<Account> accountList = accountService.findAll(); for (Account account : accountList) { System.out.println(account); } } //测试修改 @Test public void testUpdate() { Account account = new Account(); account.setId(3); account.setName("rose"); account.setMoney(2000d); accountService.update(account); } //测试删除 @Test public void testDelete() { accountService.delete(3); } }
1. 建立java项目,导入坐标 2. 编写Account实体类 3. 编写AccountDao接口和实现类 4. 编写AccountService接口和实现类 5. 编写spring核心配置文件 6. 编写测试代码
<dependencies> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.47</version> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.15</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.8.13</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>5.1.5.RELEASE</version> </dependency> </dependencies>
public class Account { private Integer id; private String name; private Double money; // setter getter.... }
public interface AccountDao { public void out(String outUser, Double money); public void in(String inUser, Double money); }
@Repository public class AccountDaoImpl implements AccountDao { @Autowired private JdbcTemplate jdbcTemplate; @Override public void out(String outUser, Double money) { jdbcTemplate.update("update account set money = money - ? where name = ?", money, outUser); } @Override public void in(String inUser, Double money) { jdbcTemplate.update("update account set money = money + ? where name = ?", money, inUser); } }
public interface AccountService { public void transfer(String outUser, String inUser, Double money); }
@Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao; @Override public void transfer(String outUser, String inUser, Double money) { accountDao.out(outUser, money); accountDao.in(inUser, money); } }
<?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" 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"> <!--IOC注解扫描--> <context:component-scan base-package="com.lagou"/> <!--加载jdbc配置文件--> <context:property-placeholder location="classpath:jdbc.properties"/> <!--把数据库链接池交给IOC容器--> <bean > <property name="driverClassName" value="${jdbc.driver}"></property> <property name="url" value="${jdbc.url}"></property> <property name="username" value="${jdbc.username}"></property> <property name="password" value="${jdbc.password}"></property> </bean> <!--把JdbcTemplate交给IOC容器--> <bean > <constructor-arg name="dataSource" ref="dataSource"></constructor-arg> </bean> </beans>
@RunWith(SpringJUnit4Cla***unner.class) @ContextConfiguration("classpath:applicationContext.xml") public class AccountServiceTest { @Autowired private AccountService accountService; @Test public void testTransfer() throws Exception { accountService.transfer("tom", "jerry", 100d); } }
Spring的事务控制能够分为编程式事务控制和声明式事务控制。spring
开发者直接把事务的代码和业务代码耦合到一块儿,在实际开发中不用。sql
开发者采用配置的方式来实现的事务控制,业务代码与事务代码实现解耦合,使用的AOP思想。数据库
PlatformTransactionManager接口,是spring的事务管理器,里面提供了咱们经常使用的操做事务的方法。编程
方法 | 说明 |
---|---|
TransactionStatus getTransaction(TransactionDefinition definition); | 获取事务的状态信息 |
void commit(TransactionStatus status); | 提交事务 |
void rollback(TransactionStatus status); | 回滚事务 |
* PlatformTransactionManager 是接口类型,不一样的 Dao 层技术则有不一样的实现类。 * Dao层技术是jdbcTemplate或mybatis时: DataSourceTransactionManager * Dao层技术是hibernate时: HibernateTransactionManager * Dao层技术是JPA时: JpaTransactionManager
TransactionDefinition接口提供事务的定义信息(事务隔离级别、事务传播行为等等)mybatis
方法 | 说明 |
---|---|
int getIsolationLevel() | 得到事务的隔离级别 |
int getPropogationBehavior() | 得到事务的传播行为 |
int getTimeout() | 得到超时时间 |
boolean isReadOnly() | 是否只读 |
设置隔离级别,能够解决事务并发产生的问题,如脏读、不可重复读和虚读(幻读)。并发
* ISOLATION_DEFAULT 使用数据库默认级别 * ISOLATION_READ_UNCOMMITTED 读未提交 * ISOLATION_READ_COMMITTED 读已提交 * ISOLATION_REPEATABLE_READ 可重复读 * ISOLATION_SERIALIZABLE 串行化
事务传播行为指的就是当一个业务方法【被】另外一个业务方法调用时,应该如何进行事务控制。app
参数 | 说明 |
---|---|
REQUIRED | 若是当前没有事务,就新建一个事务,若是已经存在一个事务中,加入到这个事务中。通常的选择(默认值) |
SUPPORTS | 支持当前事务,若是当前没有事务,就以非事务方式执行(没有事务) |
MANDATORY | 使用当前的事务,若是当前没有事务,就抛出异常 |
REQUERS_NEW | 新建事务,若是当前在事务中,把当前事务挂起 |
NOT_SUPPORTED | 以非事务方式执行操做,若是当前存在事务,就把当前事务挂起 |
NEVER | 以非事务方式运行,若是当前存在事务,抛出异常 |
NESTED | 若是当前存在事务,则在嵌套事务内执行。若是当前没有事务,则执行REQUIRED 相似的操做 |
* read-only(是否只读):建议查询时设置为只读 * timeout(超时时间):默认值是-1,没有超时限制。若是有,以秒为单位进行设置
TransactionStatus 接口提供的是事务具体的运行状态。
方法 | 说明 |
---|---|
boolean isNewTransaction() | 是不是新事务 |
boolean hasSavepoint() | 是不是回滚点 |
boolean isRollbackOnly() | 事务是否回滚 |
boolean isCompleted() | 事务是否完成 |
能够简单的理解三者的关系:事务管理器经过读取事务定义参数进行事务管理,而后会产生一系列的事务状态。
1)配置文件
<!--事务管理器交给IOC--> <bean > <property name="dataSource" ref="dataSource"/> </bean>
2)业务层代码
@Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao; @Autowired private PlatformTransactionManager transactionManager; @Override public void transfer(String outUser, String inUser, Double money) { // 建立事务定义对象 DefaultTransactionDefinition def = new DefaultTransactionDefinition(); // 设置是否只读,false支持事务 def.setReadOnly(false); // 设置事务隔离级别,可重复读mysql默认级别 def.setIsolationLevel(TransactionDefinition.ISOLATION_REPEATABLE_READ); // 设置事务传播行为,必须有事务 def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED); // 配置事务管理器 TransactionStatus status = transactionManager.getTransaction(def); try { // 转帐 accountDao.out(outUser, money); accountDao.in(inUser, money); // 提交事务 transactionManager.commit(status); } catch (Exception e) { e.printStackTrace(); // 回滚事务 transactionManager.rollback(status); } } }
Spring中的事务控制主要就是经过这三个API实现的
* PlatformTransactionManager 负责事务的管理,它是个接口,其子类负责具体工做 * TransactionDefinition 定义了事务的一些相关参数 * TransactionStatus 表明事务运行的一个实时状态
理解三者的关系:事务管理器经过读取事务定义参数进行事务管理,而后会产生一系列的事务状态。
在 Spring 配置文件中声明式的处理事务来代替代码式的处理事务。底层采用AOP思想来实现的。
声明式事务控制明确事项:
使用spring声明式事务控制转帐业务。
1. 引入tx命名空间 2. 事务管理器通知配置 3. 事务管理器AOP配置 4. 测试事务控制转帐业务代码
1)引入tx命名空间
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w2.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/s chema/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>
2)事务管理器通知配置
<!--事务管理器--> <bean > <property name="dataSource" ref="dataSource"></property> </bean> <!--通知加强--> <tx:advice transaction-manager="transactionManager"> <!--定义事务的属性--> <tx:attributes> <tx:method name="*"/> </tx:attributes> </tx:advice>
3)事务管理器AOP配置
<!--aop配置--> <aop:config> <!--切面配置--> <aop:advisor advice-ref="txAdvice" pointcut="execution(* com.lagou.serivce..*.*(..))"> </aop:advisor> </aop:config>
4)测试事务控制转帐业务代码
@Override public void transfer(String outUser, String inUser, Double money) { accountDao.out(outUser, money); // 制造异常 int i = 1 / 0; accountDao.in(inUser, money); }
<tx:method name="transfer" isolation="REPEATABLE_READ" propagation="REQUIRED" timeout="-1" read-only="false"/> * name:切点方法名称 * isolation:事务的隔离级别 * propogation:事务的传播行为 * timeout:超时时间 * read-only:是否只读
<tx:attributes> <tx:method name="save*" propagation="REQUIRED"/> <tx:method name="delete*" propagation="REQUIRED"/> <tx:method name="update*" propagation="REQUIRED"/> <tx:method name="find*" read-only="true"/> <tx:method name="*"/> </tx:attributes>
* 平台事务管理器配置 * 事务通知的配置 * 事务aop织入的配置
步骤分析
1. 修改service层,增长事务注解 2. 修改spring核心配置文件,开启事务注解支持
步骤分析
@Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao; @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.REPEATABLE_READ, timeout = -1, readOnly = false) @Override public void transfer(String outUser, String inUser, Double money) { accountDao.out(outUser, money); int i = 1 / 0; accountDao.in(inUser, money); } }
2)修改spring核心配置文件,开启事务注解支持
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w2.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"> <!--省略以前datsSource、jdbcTemplate、组件扫描配置--> <!--事务管理器--> <bean > <property name="dataSource" ref="dataSource"></property> </bean> <!--事务的注解支持--> <tx:annotation-driven/> </beans>
核心配置类
@Configuration // 声明为spring配置类 @ComponentScan("com.lagou") // 扫描包 @Import(DataSourceConfig.class) // 导入其余配置类 @EnableTransactionManagement // 事务的注解驱动 public class SpringConfig { @Bean public JdbcTemplate getJdbcTemplate(@Autowired DataSource dataSource) { return new JdbcTemplate(dataSource); } @Bean("transactionManager") public PlatformTransactionManager getPlatformTransactionManager(@Autowired DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } }
数据源配置类
@PropertySource("classpath:jdbc.properties") public class DataSourceConfig { @Value("${jdbc.driver}") private String driver; @Value("${jdbc.url}") private String url; @Value("${jdbc.username}") private String username; @Value("${jdbc.password}") private String password; @Bean public DataSource getDataSource() { DruidDataSource dataSource = new DruidDataSource(); dataSource.setDriverClassName(driver); dataSource.setUrl(url); dataSource.setUsername(username); dataSource.setPassword(password); return dataSource; } }
* 平台事务管理器配置(xml、注解方式) * 事务通知的配置(@Transactional注解配置) * 事务注解驱动的配置 <tx:annotation-driven/>、@EnableTransactionManagement
应用上下文对象是经过 new ClasspathXmlApplicationContext(spring配置文件) 方式获取的,可是每次从容器中得到Bean时都要编写 new ClasspathXmlApplicationContext(spring配置文件) ,这样的弊端是配置文件加载屡次,应用上下文对象建立屡次。 Version:0.9 StartHTML:0000000105 EndHTML:0000002684 StartFragment:0000000141 EndFragment:0000002644
解决思路分析: 在Web项目中,可使用ServletContextListener监听Web应用的启动,咱们能够在Web应用启动时,就加载Spring的配置文件,建立应用上下文对象ApplicationContext,在将其存储到最大的域servletContext域中,这样就能够在任意位置从域中得到应用上下文ApplicationContext对象了。
上面的分析不用手动实现,Spring提供了一个监听器ContextLoaderListener就是对上述功能的封装,该监听器内部加载Spring配置文件,建立应用上下文对象,并存储到ServletContext域中,提供了一个客户端工具WebApplicationContextUtils供使用者得到应用上下文对象。
因此咱们须要作的只有两件事:
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.1.5.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>5.1.5.RELEASE</version> </dependency>
<!--全局参数--> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:applicationContext.xml</param-value>• </context-param> <!--Spring的监听器--> <listener> <listener-class> org.springframework.web.context.ContextLoaderListener </listener-class> </listener>
ApplicationContext applicationContext = WebApplicationContextUtils.getWebApplicationContext(servletContext); Object obj = applicationContext.getBean("id");