Spring框架:第四篇

第一章: Spring 中的 JdbcTemplate

1.1-JdbcTemplate 概述

它是 spring 框架中提供的一个对象,是对原始 Jdbc API 对象的简单封装。spring 框架为我们提供了很多 的操作模板类。

操作关系型数据的:

  • JdbcTemplate
  • HibernateTemplate

操作 nosql 数据库的:

  • RedisTemplate

操作消息队列的:

  • JmsTemplate

1.2-JdbcTemplate快速入门

Maven工程

<dependencies>
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.12</version>
    </dependency>
    <!--spring-context-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--【spring-jdbc】-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-jdbc</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--spring-tx-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-tx</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--mysql-->
    <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>5.1.26</version>
    </dependency>

  </dependencies>

数据库脚本

CREATE DATABASE  IF NOT EXISTS db1
USE db1
CREATE TABLE account(
id INT PRIMARY KEY AUTO_INCREMENT,
NAME VARCHAR(40),
money FLOAT
)CHARACTER SET utf8 COLLATE utf8_general_ci;
INSERT INTO account(NAME,money) VALUES('aaa',1000);
INSERT INTO account(NAME,money) VALUES('bbb',1000);

实体类

public class Account {
  private int id;
  private String name;
  private float money;

  public int getId() {
    return id;
  }

  public void setId(int id) {
    this.id = id;
  }

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name;
  }

  public float getMoney() {
    return money;
  }

  public void setMoney(float money) {
    this.money = money;
  }

  @Override
  public String toString() {
    return "Account{" +
            "id=" + id +
            ", name='" + name + '\'' +
            ", money=" + money +
            '}';
  }
}

JdbcTemplate测试类

public class Demo01 {
  public static void main(String[] args) {
    // 【创建Spring提供的数据源对象】
    DriverManagerDataSource ds = new DriverManagerDataSource();
    // 设置驱动
    ds.setDriverClassName("com.mysql.jdbc.Driver");
    // 设置连接
    ds.setUrl("jdbc:mysql://localhost:3306/db1");
    // 设置数据库登录用户名
    ds.setUsername("root");
    // 设置数据库登录密码
    ds.setPassword("root");

    // 【创建JdbcTemplate对象】
    JdbcTemplate jt = new JdbcTemplate();
    // 设置数据源
    jt.setDataSource(ds);
    // 向数据库中添加一条数据
    int row = jt.update("insert into account(name,money) values(?,?)","CCC",2000);
    System.out.println(row);
  }
}

1.3-Spring配置数据源及CRUD

Spring配置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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--配置数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url" value="jdbc:mysql://localhost:3306/db1"></property>
        <property name="username" value="root"></property>
        <property name="password" value="root"></property>
    </bean>
    <!--配置JdbcTemplate-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--配置AccountDao-->
    <bean id="accountDao" class="cn.lpl666.dao.impl.AccountImpl">
        <property name="jt" ref="jdbcTemplate"></property>
    </bean>
</beans>

测试类

public class Demo02 {
  public static void main(String[] args) {
    // 创建容器对象
    ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
    // 创建JdbcTemplate
    JdbcTemplate jt = ac.getBean("jdbcTemplate",JdbcTemplate.class);
    //【插入】
    // int row = jt.update("insert into account(name,money) values(?,?)","DDD",2000);
    // System.out.println(row);

    //【更新】
    // int row = jt.update("update account set name=? where id=?","ddd",4);
    // System.out.println(row);

    //【删除】
    // int row = jt.update("delete from account where id=?",4);
    // System.out.println(row);

    //【查询结果集-集合】
    // List<Account> list = jt.query("select * from account where money>?",new BeanPropertyRowMapper<Account>(Account.class),10);
    // System.out.println(list);

    //【查询结果-单个】
    // List<Account> list = jt.query("select * from account where id=?",new BeanPropertyRowMapper<Account>(Account.class),1);
    // System.out.println(list.get(0));

   //【查询单行单列】
   // Long aLong = jt.queryForObject("select count(*) from account where money>?", Long.class, 10);
   // System.out.println(aLong);
  }
}

1.4-Dao 继承 JdbcDaoSupport

JdbcDaoSupport

// JdbcDaoSupport 是 spring 框架为我们提供的一个类,该类中定义了一个 JdbcTemplate 对象,我们可以
直接获取使用,但是要想创建该对象,需要为其提供一个数据源:具体源码如下:
public abstract class JdbcDaoSupport extends DaoSupport {
//定义对象
private JdbcTemplate jdbcTemplate;
//set 方法注入数据源,判断是否注入了,注入了就创建 JdbcTemplate
public final void setDataSource(DataSource dataSource) {
    if (this.jdbcTemplate == null || dataSource != this.jdbcTemplate.getDataSource())
    { //如果提供了数据源就创建 JdbcTemplate
    this.jdbcTemplate = createJdbcTemplate(dataSource);
    initTemplateConfig();
    }
}
//使用数据源创建 JdcbTemplate
protected JdbcTemplate createJdbcTemplate(DataSource dataSource) {
	return new JdbcTemplate(dataSource);
}
//当然,我们也可以通过注入 JdbcTemplate 对象
public final void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
    this.jdbcTemplate = jdbcTemplate;
    initTemplateConfig();
}
//使用 getJdbcTmeplate 方法获取操作模板对象
public final JdbcTemplate getJdbcTemplate() {
 	return this.jdbcTemplate;
}

Dao层

接口

public interface IAccountDao {
  /**
   * 查询所有
   * @return
   */
  List<Account> findAll();

  /**
   * 查询一个
   * @param id
   * @return
   */
  Account findOne(int id);

  /**
   * 更新一个
   * @param account
   * @return
   */
  boolean update(Account account);
}

实现类

package cn.lpl666.dao.impl;
import cn.lpl666.dao.IAccountDao;
import cn.lpl666.domain.Account;

// import cn.lpl666.util.JdbcSupport;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.support.JdbcDaoSupport;

import java.util.List;
//public class AccountImpl extends JdbcSupport implements IAccountDao
public class AccountImpl extends JdbcDaoSupport implements IAccountDao {
  @Override
  public List<Account> findAll() {
    List<Account> list = getJdbcTemplate().query("select * from account",new BeanPropertyRowMapper<Account>(Account.class));
    return list;
  }

  @Override
  public Account findOne(int id) {
    List<Account> list = getJdbcTemplate().query("select * from account where id=?",new BeanPropertyRowMapper<Account>(Account.class),id);
    return list.isEmpty()?null:list.get(0);
  }

  @Override
  public boolean update(Account account) {
    int row = getJdbcTemplate().update("update account set name=?,money=? where id=?", account.getName(), account.getMoney(), account.getId());
    return row>0;
  }
}

第二章:Spring中的事务控制

2.1-Spring 事务控制我们要明确的

第一:JavaEE 体系进行分层开发,事务处理位于业务层,Spring 提供了分层设计业务层的事务处理解决方 案。

第二:spring 框架为我们提供了一组事务控制的接口。具体在后面的第二小节介绍。这组接口是在 spring-tx-5.0.2.RELEASE.jar 中。

第三:spring 的事务控制都是基于 AOP 的,它既可以使用编程的方式实现,也可以使用配置的方式实现。本章描述的是使用配置的方式实现。

2.2-Spring 中事务控制的 API 介绍

PlatformTransactionManager

此接口是 spring 的事务管理器,它里面提供了我们常用的操作事务的方法,如下图:

我们在开发中都是使用它的实现类,如下图:

TransactionDefinition

它是事务的定义信息对象,里面有如下方法:

事务的隔离级别

事务的传播行为

赶超时间

默认值是-1,没有超时限制。如果有,以秒为单位进行设置。

是否是只读事务

建议查询时设置为只读

TransactionStatus

此接口提供的是事务具体的运行状态,方法介绍如下图:

2.3-基于 XML 的声明式事务控制(配置方式)

搭建Maven工程

<dependencies>
    <!--junit-->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.12</version>
      <scope>test</scope>
    </dependency>
    <!--spring-context-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--spring-tx-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-tx</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--spring-jdbc-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-jdbc</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--spring-test-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-test</artifactId>
      <version>5.2.3.RELEASE</version>
    </dependency>
    <!--mysql-->
    <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>5.1.26</version>
    </dependency>
    <!--aspectjweaver-->
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.9.5</version>
    </dependency>
  </dependencies>

第二步:创建 spring 的配置文件并导入约束

此处需要导入 aop 和 tx 两个名称空间


<?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"
 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/tx
 http://www.springframework.org/schema/tx/spring-tx.xsd
 http://www.springframework.org/schema/aop
 http://www.springframework.org/schema/aop/spring-aop.xsd">
</beans>

第三步:Dao层接口和实现类

接口

public interface IAccountDao {
  /**
   * 查询一个
   * @param id
   * @return
   */
  Account findOne(int id);

  /**
   * 更新一个
   * @param account
   * @return
   */
  boolean update(Account account);
}

实现类

public class AccountDaoImpl extends JdbcDaoSupport implements IAccountDao {
  @Override
  public Account findOne(int id) {
    String sql = "SELECT * FROM account WHERE id = ?";
    List<Account> list =  getJdbcTemplate().query(sql,new BeanPropertyRowMapper<Account>(Account.class),id);
    return list==null?null:list.get(0);
  }

  @Override
  public boolean update(Account account) {
    String sql = "UPDATE account SET NAME=?,money=? WHERE id = ?";
    int row = getJdbcTemplate().update(sql, account.getName(), account.getMoney(), account.getId());
    return row>0;
  }
}

第四步:Service层接口和实现类

接口

public interface IAccountService {
  /**
   * 转账
   * @param oneId  转账用户id
   * @param twoId  收账用户id
   * @param money  转账钱数
   */
  void transfer(int oneId,int twoId,float money);
}

实现类

public class AccountSerice implements IAccountService {
  private IAccountDao dao = null;
  public void setDao(IAccountDao dao) {
    this.dao = dao;
  }

  @Override
  public void transfer(int oneId, int twoId, float money) {
    Account one = dao.findOne(oneId);
    Account two = dao.findOne(twoId);
    one.setMoney(one.getMoney()-money);
    two.setMoney(two.getMoney()+money);
    dao.update(one);
    //int i = 10/0;
    dao.update(two);

  }
}

第五步: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: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/tx
        http://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--【配置数据源】-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url" value="jdbc:mysql://localhost:3306/db1"></property>
        <property name="username" value="root"></property>
        <property name="password" value="root"></property>
    </bean>
    <!--【配置AccountDao】-->
    <bean id="accountDao" class="cn.lpl666.dao.impl.AccountDaoImpl">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--【配置AccountService】-->
    <bean id="accountService" class="cn.lpl666.service.impl.AccountSerice">
        <property name="dao" ref="accountDao"></property>
    </bean>
    <!--【配置事务管理器】-->
    <bean id="transferManger" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--【配置事务通知】-->
    <tx:advice id="txAdivce" transaction-manager="transferManger">
        <!--配置事务属性-->
        <tx:attributes>
            <!--propagation:用于指定事务的传播行为。默认值是REQUIRED,表示一定会有事务,增删改的选择。查询方法可以选择SUPPORTS。-->
            <tx:method name="*" propagation="REQUIRED"/>
            <tx:method name="find*" propagation="SUPPORTS"></tx:method>
        </tx:attributes>
    </tx:advice>
    <!--【配置AOP】-->
    <aop:config>
        <!--配置管理包的表达式-->
        <aop:pointcut id="servicePath" expression="execution(* cn.lpl666.service.impl.*.*(..))"></aop:pointcut>
        <!--建立切入点表达式和事务通知的对应关系-->
        <aop:advisor advice-ref="txAdivce" pointcut-ref="servicePath"></aop:advisor>
    </aop:config>
</beans>

在 tx:advice 标签内部 配置事务的属性

  • read-only:是否是只读事务。默认 false,不只读。
  • isolation:指定事务的隔离级别。默认值是使用数据库的默认隔离级别。
  • propagation:指定事务的传播行为。
  • timeout:指定超时时间。默认值为:-1。永不超时。
  • rollback-for:用于指定一个异常,当执行产生该异常时,事务回滚。产生其他异常,事务不回滚。 没有默认值,任何异常都回滚。
  • no-rollback-for:用于指定一个异常,当产生该异常时,事务不回滚,产生其他异常时,事务回 滚。没有默认值,任何异常都回滚。

2.4-基于注解的事务控制方式1

Dao层实现类

@Repository("accountDao")
public class AccountDaoImpl  implements IAccountDao {
  @Autowired
  @Qualifier("jdbcTemplate")
  private JdbcTemplate jdbcTemplate = null;
  @Override
  public Account findOne(int id) {
    String sql = "SELECT * FROM account WHERE id = ?";
    List<Account> list =  jdbcTemplate.query(sql,new BeanPropertyRowMapper<Account>(Account.class),id);
    return list==null?null:list.get(0);
  }

  @Override
  public boolean update(Account account) {
    String sql = "UPDATE account SET NAME=?,money=? WHERE id = ?";
    int row = jdbcTemplate.update(sql, account.getName(), account.getMoney(), account.getId());
    return row>0;
  }
}

服务层实现类

Service("accountService")
@Transactional(propagation = Propagation.SUPPORTS,readOnly = true)
public class AccountService implements IAccountService {
  @Autowired
  @Qualifier("accountDao")
  private IAccountDao dao = null;

  @Transactional(propagation = Propagation.REQUIRED,readOnly = false)
  @Override
  public void transfer(int oneId, int twoId, float money) {
    Account one = dao.findOne(oneId);
    Account two = dao.findOne(twoId);
    one.setMoney(one.getMoney()-money);
    two.setMoney(two.getMoney()+money);
    dao.update(one);
    // int i = 10/0;
    dao.update(two);

  }
}

配置文件

<?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/tx
        http://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">
    <!-- 告知 spring 创建容器时要扫描的包 -->
    <context:component-scan base-package="cn.lpl666"/>
    <!--配置数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url" value="jdbc:mysql://localhost:3306/db1"></property>
        <property name="username" value="root"></property>
        <property name="password" value="root"></property>
    </bean>
    <!--配置jdbcTemplate-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--配置事务管理器-->
    <bean id="transferManger" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!-- 开启spring对注解事务的支持-->
    <tx:annotation-driven transaction-manager="transferManger"></tx:annotation-driven>

</beans>

2.5-基于注解的事务控制方式2

纯注解

Dao层实现类

@Repository("accountDao")
public class AccountDaoImpl  implements IAccountDao {
  @Autowired
  @Qualifier("jdbcTemplate")
  private JdbcTemplate jdbcTemplate = null;
  @Override
  public Account findOne(int id) {
    String sql = "SELECT * FROM account WHERE id = ?";
    List<Account> list =  jdbcTemplate.query(sql,new BeanPropertyRowMapper<Account>(Account.class),id);
    return list==null?null:list.get(0);
  }

  @Override
  public boolean update(Account account) {
    String sql = "UPDATE account SET NAME=?,money=? WHERE id = ?";
    int row = jdbcTemplate.update(sql, account.getName(), account.getMoney(), account.getId());
    return row>0;
  }
}

服务层实现类

@Service("accountService")
@Transactional(propagation = Propagation.SUPPORTS,readOnly = true)
public class AccountService implements IAccountService {
  @Autowired
  @Qualifier("accountDao")
  private IAccountDao dao = null;

  @Transactional(propagation = Propagation.REQUIRED,readOnly = false)
  @Override
  public void transfer(int oneId, int twoId, float money) {
    Account one = dao.findOne(oneId);
    Account two = dao.findOne(twoId);
    one.setMoney(one.getMoney()-money);
    two.setMoney(two.getMoney()+money);
    dao.update(one);
    int i = 10/0;
    dao.update(two);

  }
}

JDBC配置类

package cn.lpl666.config;

import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.PropertySource;
import org.springframework.context.annotation.Scope;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.datasource.DriverManagerDataSource;

import javax.sql.DataSource;
@PropertySource("classpath:jdbc.properties")
public class JdbcConfig {
  @Value("${jdbc.driver}")
  private String driver;
  @Value("${jdbc.url}")
  private String url;
  @Value("${jdbc.user}")
  private String username;
  @Value("${jdbc.password}")
  private String password;

  /**
   * 创建QueryRunner对象
   * @param ds
   * @return
   */
  @Bean(name="jdbcTemplate")
  public JdbcTemplate createJdbcTemplate(@Qualifier("dataSource") DataSource ds){
    return new JdbcTemplate(ds);
  }

  /**
   * 创建数据源对象
   * @return
   */
  @Bean(name = "dataSource")
  public DataSource createDataSource() {
     DriverManagerDataSource dm = new DriverManagerDataSource();
     dm.setDriverClassName(driver);
     dm.setUrl(url);
     dm.setUsername(username);
     dm.setPassword(password);
     return dm;
  }
}

JDBC配置文件

jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/db1
jdbc.user=root
jdbc.password=root

事务管理配置类

public class TransactionConfig {
  @Bean(name="transactionManager")
  public PlatformTransactionManager createTransactionManager(@Qualifier("dataSource") DataSource dataSource){
    return new DataSourceTransactionManager(dataSource);
  }

}

Spring配置类

@Configuration
@ComponentScan("cn.lpl666")  // spring容器要扫描的包
@Import({JdbcConfig.class,TransactionConfig.class})  // 导入其他配置类
@EnableTransactionManagement
public class SpringConfiguration {
}

测试类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = SpringConfiguration.class)
public class AccountTest2 {
  @Autowired
  @Qualifier("accountService")
  private IAccountService service = null;
  @Test
  public void transfer(){
    service.transfer(1,2,100);
  }
}
posted @ 2020-02-23 17:52  雷哒哒  阅读(161)  评论(0编辑  收藏  举报