通过学习Spring的IOC技术和AOP技术,我们可以管理类之间的依赖并从另一个角度来设计和实现项目中身份校验、日志记录等功能。为使项目能真正的面向用户,我们需要把数据进行持久化:用户更改提交的信息可以再次被访问到。因此,接下来的文章中我们将学习Spring的数据访问功能。
我们将会先学习Spring基于JDBC的数据访问方式,为提高代码效率我们将进一步学习讲解Mybatis框架。
为了更好更方便的理解文章内容需要以下知识储备和环境准备工作:
- 基本的SQL知识;
- 基础的java访问数据库知识;
- 本机安装mysql;
一、DataSource配置
DataSource
是JDBC规范的一部分,它是一个通用连接工厂类。通过它容器或框架可以隐藏连接池和事务管理的代码细节。Spring通过DataSource
来获取数据库的连接,因此我们使用Spring来访问数据库,首先要配置DataSource
bean。
Spring支持三种DataSource
配置方式:
- JDBC方式
- DBCP方式
- C3P0方式
我们会以XML配置方式展示三种配置方式,但是本文我们依然主要使用JDBC方式。
1.1 JDBC配置方式
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<context:property-placeholder location="jdbc.properties"/>
上面为Spring JDBC配置方式模版,我们将类DriverManagerDataSource
定义为dataSource的bean,并且为该dataSource配置属性。需要注意的是该配置模版中我们使用了${}
方式为属性设置值,其中真实的值我们配置在jdbc.properties
文件中,使用<context:property-placeholder location="jdbc.properties"/>
将配置信息加载,Spring会使用该文件内的键值对替换${}
。jdbc.properties
文件样式如下:
jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/score_manager
jdbc.username=root
jdbc.password=root
其中jdbc.driverClassName
设置数据驱动类,这里使用mysql数据驱动,jdbc.url
指定连接数据库,这里指向本地数据库,端口3306,数据库名为score_manager。jdbc.username
和jdbc.password
设置访问数据库的用户名和密码,具体根据你数据库配置进行更改。
当然,如果是用于学习或试验,我们可以直接将jdbc.properties
的内容写到DataSource
配置文件中,上述配置文件可写成如下方式,效果相同:
<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/score_manager"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</bean>
1.2 DBCP配置方式
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<context:property-placeholder location="jdbc.properties"/>
该配置方式与JDBC配置方式相似,这里就不进一步陈述了。但需要注意的是,具体的driverClassName
和url
配置方式,需要查看具体的数据库产品文档说明。
1.3 C3P0配置方式
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
<property name="driverClass" value="${jdbc.driverClassName}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="user" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<context:property-placeholder location="jdbc.properties"/>
二、Spring JDBC访问数据
Spring框架的JDBC抽象为开发者提供高效、便捷的数据库访问操作。Spring为我们提供了如下工具类来访问数据库:
- JdbcTemplate
- NamedParameterJdbcTemplate
- SimpleJdbcInsert
- SimpleJdbcCall
其中JdbcTemplate是Spring JDBC抽象中最核心的类;NamedParameterJdbcTemplate类封装了JdbcTemplate,使用命名参数来代替SQL语句中的?
。
2.1 JdbcTemplate
Jdbc配置
代码模版
public class StudentDAOImpl implements StudentDAO{
private JdbcTemplate jdbcTemplate;
//数据访问方法
......
//属性注入
public void setDataSource(DataSource dataSource) {
this.jdbcTemplate = new JdbcTemplate(dataSource);
}
}
如上代码实现,我们持有JdbcTemplate
类的私有属性,并且使用set注入的方式将配置好的Datasource
传入,初始化jdbcTemplate变量。
XML文件配置:
<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/score_manager"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</bean>
<bean id="studentDao" class="com.liangwei.learnspring.dao.impl.StudentDAOImpl">
<property name="dataSource" ref="dataSource"/>
</bean>
JdbcTemplate提供了丰富的方法用于执行数据库访问:查询、插入、更新、删除。我们这里大致介绍下query
、update
等方法。
查找
查询
JdbcTemplate提供了多种查询方法,若我们期望查询的为单个符合条件的结果,则可食用queryForObject
方法,若多行,则可直接使用query
方法。
查询符合数据的行数:
int rowCount = this.jdbcTemplate.queryForObject("select count(*) from student", Integer.class);
根据多个条件查询某个结果
String name = this.jdbcTemplate.queryForObject("select name from student where id = ? and gender=? ",String.class,1,"F");
当然我们也可以使用如下语句:
String name = this.jdbcTemplate.queryForObject("select name from student where id = ? and gender=? ",new Object[]{1,"F"},String.class);
查询自定义的数据类型对象:
首先我们定义数据类如下:
public class Student {
private int id;
private String name;
private String gender;
private int age;
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 String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString(){
return "["+id+","+name+","+gender+","+age+"]";
}
}
查询语句如下:
Student student = this.jdbcTemplate.queryForObject("select * from student where id = ?", new Object[]{1},
new RowMapper<Student>() {
public Student mapRow(ResultSet rs, int rowNum) throws SQLException {
Student tmp = new Student();
tmp.setId(Integer.parseInt(rs.getString("id")));
tmp.setName(rs.getString("name"));
tmp.setGender(rs.getString("gender"));
tmp.setAge(Integer.parseInt(rs.getString("age")));
return tmp;
}
});
当然,当我们需要查询多个满足对象的结果是,可以使用如下方法:
List<Student> students = this.jdbcTemplate.query("select * from student", new RowMapper<Student>() {
public Student mapRow(ResultSet rs, int rowNum) throws SQLException {
Student tmp = new Student();
tmp.setId(Integer.parseInt(rs.getString("id")));
tmp.setName(rs.getString("name"));
tmp.setGender(rs.getString("gender"));
tmp.setAge(Integer.parseInt(rs.getString("age")));
return tmp;
}
});
更新
插入数据:
this.jdbcTemplate.update(
"insert into student (id,name,gender,age) values(?,?,?,?)",
student.getId(),student.getName(),student.getGender(),student.getAge());
更新数据
this.jdbcTemplate.update("UPDATE student set gender=? where id = ?",student.getGender(),student.getId());
删除数据
this.jdbcTemplate.update("delete from student where id=?",student.getId());
2.2 NamedParameterJdbcTemplate
NamedParameterJdbcTemplate
封装了JdbcTemplate
,但是正如它名字所体现的那样,它不实用?
来构造SQL语句,而是使用参数命名的方式来构造SQL语句,在内部完成SQL语句的组装。接下来我们简要介绍下该类用法。
配置
public class StudentDAOImpl implements StudentDAO{
private NamedParameterJdbcTemplate namedParameterJdbcTemplate;
//数据访问方法
......
//属性注入
public void setDataSource(DataSource dataSource) {
this. namedParameterJdbcTemplate = new NamedParameterJdbcTemplate(dataSource);
}
}
XML配置和JdbcTemplate相同,可直接参考JdbcTemplate
配置。
查询数据
String sql = "select * from student where id = :studentId";
SqlParameterSource namedParameters = new MapSqlParameterSource("studentId", id);
return this.namedParameterJdbcTemplate.queryForObject(sql, namedParameters, Integer.class);
如上所示,我们使用MapSqlParameterSource
为sql语句中的参数赋值,当我们需要为多个参数赋值时,我们还可以使用Map类型的类:
Map<String, String> namedParameters = Collections.singletonMap("studentId", id);
我们创建了一个Map,可以为SQL语句中的多个参数赋值。
三、JDBC访问实战
学习了Spring JDBC知识,我们接下来通过代码实战来巩固一下(假设此时你本地已安装配置好MySQL数据库)。该例子使用JdbcTemplate来访问数据库,尽量使用到前文讲述到的知识,你可以下载代码在本地运行,并通过修改来巩固所需到的知识。不过在运行代码之前你需要做好下面的准备工作:
- 创建数据库:score_manager。
- 创建数据表:student,表结构如下:
列名 | 类型 |
---|---|
id | int |
name | varchar(20) |
gender | char(1) |
age | int |
代码下载地址
四、总结
本文我们学习了如何使用Spring JDBC来访问数据库,并且以常用数据库MySQL为例进行代码实战练习。Spring的JDBC抽象极大的提升了程序员对数据库开发的效率,不需要再关心数据库连接获取、准备和运行句子、检索结果集、关闭连接等繁杂的工作。但通过前面的实战练习也许你也发现构造SQL语句,为句子中的参数赋值;若获取自定义定义对象则需要从结果集获取数据并对自定义对象赋值。我们将在下篇文章学习如何解决这些问题,进一步提升工作效率。