通過(guò)學(xué)習(xí)Spring的IOC技術(shù)和AOP技術(shù),我們可以管理類(lèi)之間的依賴(lài)并從另一個(gè)角度來(lái)設(shè)計(jì)和實(shí)現(xiàn)項(xiàng)目中身份校驗(yàn)、日志記錄等功能。為使項(xiàng)目能真正的面向用戶,我們需要把數(shù)據(jù)進(jìn)行持久化:用戶更改提交的信息可以再次被訪問(wèn)到。因此,接下來(lái)的文章中我們將學(xué)習(xí)Spring的數(shù)據(jù)訪問(wèn)功能。
我們將會(huì)先學(xué)習(xí)Spring基于JDBC的數(shù)據(jù)訪問(wèn)方式,為提高代碼效率我們將進(jìn)一步學(xué)習(xí)講解Mybatis框架。
為了更好更方便的理解文章內(nèi)容需要以下知識(shí)儲(chǔ)備和環(huán)境準(zhǔn)備工作:
- 基本的SQL知識(shí);
- 基礎(chǔ)的java訪問(wèn)數(shù)據(jù)庫(kù)知識(shí);
- 本機(jī)安裝mysql;
一、DataSource配置
DataSource
是JDBC規(guī)范的一部分,它是一個(gè)通用連接工廠類(lèi)。通過(guò)它容器或框架可以隱藏連接池和事務(wù)管理的代碼細(xì)節(jié)。Spring通過(guò)DataSource
來(lái)獲取數(shù)據(jù)庫(kù)的連接,因此我們使用Spring來(lái)訪問(wèn)數(shù)據(jù)庫(kù),首先要配置DataSource
bean。
Spring支持三種DataSource
配置方式:
- JDBC方式
- DBCP方式
- C3P0方式
我們會(huì)以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配置方式模版,我們將類(lèi)DriverManagerDataSource
定義為dataSource的bean,并且為該dataSource配置屬性。需要注意的是該配置模版中我們使用了${}
方式為屬性設(shè)置值,其中真實(shí)的值我們配置在jdbc.properties
文件中,使用<context:property-placeholder location="jdbc.properties"/>
將配置信息加載,Spring會(huì)使用該文件內(nèi)的鍵值對(duì)替換${}
。jdbc.properties
文件樣式如下:
jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/score_manager
jdbc.username=root
jdbc.password=root
其中jdbc.driverClassName
設(shè)置數(shù)據(jù)驅(qū)動(dòng)類(lèi),這里使用mysql數(shù)據(jù)驅(qū)動(dòng),jdbc.url
指定連接數(shù)據(jù)庫(kù),這里指向本地?cái)?shù)據(jù)庫(kù),端口3306,數(shù)據(jù)庫(kù)名為score_manager。jdbc.username
和jdbc.password
設(shè)置訪問(wèn)數(shù)據(jù)庫(kù)的用戶名和密碼,具體根據(jù)你數(shù)據(jù)庫(kù)配置進(jìn)行更改。
當(dāng)然,如果是用于學(xué)習(xí)或試驗(yàn),我們可以直接將jdbc.properties
的內(nèi)容寫(xiě)到DataSource
配置文件中,上述配置文件可寫(xiě)成如下方式,效果相同:
<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配置方式相似,這里就不進(jìn)一步陳述了。但需要注意的是,具體的driverClassName
和url
配置方式,需要查看具體的數(shù)據(jù)庫(kù)產(chǎn)品文檔說(shuō)明。
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訪問(wèn)數(shù)據(jù)
Spring框架的JDBC抽象為開(kāi)發(fā)者提供高效、便捷的數(shù)據(jù)庫(kù)訪問(wèn)操作。Spring為我們提供了如下工具類(lèi)來(lái)訪問(wèn)數(shù)據(jù)庫(kù):
- JdbcTemplate
- NamedParameterJdbcTemplate
- SimpleJdbcInsert
- SimpleJdbcCall
其中JdbcTemplate是Spring JDBC抽象中最核心的類(lèi);NamedParameterJdbcTemplate類(lèi)封裝了JdbcTemplate,使用命名參數(shù)來(lái)代替SQL語(yǔ)句中的?
。
2.1 JdbcTemplate
Jdbc配置
代碼模版
public class StudentDAOImpl implements StudentDAO{
private JdbcTemplate jdbcTemplate;
//數(shù)據(jù)訪問(wèn)方法
......
//屬性注入
public void setDataSource(DataSource dataSource) {
this.jdbcTemplate = new JdbcTemplate(dataSource);
}
}
如上代碼實(shí)現(xiàn),我們持有JdbcTemplate
類(lèi)的私有屬性,并且使用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提供了豐富的方法用于執(zhí)行數(shù)據(jù)庫(kù)訪問(wèn):查詢(xún)、插入、更新、刪除。我們這里大致介紹下query
、update
等方法。
查找
查詢(xún)
JdbcTemplate提供了多種查詢(xún)方法,若我們期望查詢(xún)的為單個(gè)符合條件的結(jié)果,則可食用queryForObject
方法,若多行,則可直接使用query
方法。
查詢(xún)符合數(shù)據(jù)的行數(shù):
int rowCount = this.jdbcTemplate.queryForObject("select count(*) from student", Integer.class);
根據(jù)多個(gè)條件查詢(xún)某個(gè)結(jié)果
String name = this.jdbcTemplate.queryForObject("select name from student where id = ? and gender=? ",String.class,1,"F");
當(dāng)然我們也可以使用如下語(yǔ)句:
String name = this.jdbcTemplate.queryForObject("select name from student where id = ? and gender=? ",new Object[]{1,"F"},String.class);
查詢(xún)自定義的數(shù)據(jù)類(lèi)型對(duì)象:
首先我們定義數(shù)據(jù)類(lèi)如下:
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+"]";
}
}
查詢(xún)語(yǔ)句如下:
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;
}
});
當(dāng)然,當(dāng)我們需要查詢(xún)多個(gè)滿足對(duì)象的結(jié)果是,可以使用如下方法:
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;
}
});
更新
插入數(shù)據(jù):
this.jdbcTemplate.update(
"insert into student (id,name,gender,age) values(?,?,?,?)",
student.getId(),student.getName(),student.getGender(),student.getAge());
更新數(shù)據(jù)
this.jdbcTemplate.update("UPDATE student set gender=? where id = ?",student.getGender(),student.getId());
刪除數(shù)據(jù)
this.jdbcTemplate.update("delete from student where id=?",student.getId());
2.2 NamedParameterJdbcTemplate
NamedParameterJdbcTemplate
封裝了JdbcTemplate
,但是正如它名字所體現(xiàn)的那樣,它不實(shí)用?
來(lái)構(gòu)造SQL語(yǔ)句,而是使用參數(shù)命名的方式來(lái)構(gòu)造SQL語(yǔ)句,在內(nèi)部完成SQL語(yǔ)句的組裝。接下來(lái)我們簡(jiǎn)要介紹下該類(lèi)用法。
配置
public class StudentDAOImpl implements StudentDAO{
private NamedParameterJdbcTemplate namedParameterJdbcTemplate;
//數(shù)據(jù)訪問(wèn)方法
......
//屬性注入
public void setDataSource(DataSource dataSource) {
this. namedParameterJdbcTemplate = new NamedParameterJdbcTemplate(dataSource);
}
}
XML配置和JdbcTemplate相同,可直接參考JdbcTemplate
配置。
查詢(xún)數(shù)據(jù)
String sql = "select * from student where id = :studentId";
SqlParameterSource namedParameters = new MapSqlParameterSource("studentId", id);
return this.namedParameterJdbcTemplate.queryForObject(sql, namedParameters, Integer.class);
如上所示,我們使用MapSqlParameterSource
為sql語(yǔ)句中的參數(shù)賦值,當(dāng)我們需要為多個(gè)參數(shù)賦值時(shí),我們還可以使用Map類(lèi)型的類(lèi):
Map<String, String> namedParameters = Collections.singletonMap("studentId", id);
我們創(chuàng)建了一個(gè)Map,可以為SQL語(yǔ)句中的多個(gè)參數(shù)賦值。
三、JDBC訪問(wèn)實(shí)戰(zhàn)
學(xué)習(xí)了Spring JDBC知識(shí),我們接下來(lái)通過(guò)代碼實(shí)戰(zhàn)來(lái)鞏固一下(假設(shè)此時(shí)你本地已安裝配置好MySQL數(shù)據(jù)庫(kù))。該例子使用JdbcTemplate來(lái)訪問(wèn)數(shù)據(jù)庫(kù),盡量使用到前文講述到的知識(shí),你可以下載代碼在本地運(yùn)行,并通過(guò)修改來(lái)鞏固所需到的知識(shí)。不過(guò)在運(yùn)行代碼之前你需要做好下面的準(zhǔn)備工作:
- 創(chuàng)建數(shù)據(jù)庫(kù):score_manager。
- 創(chuàng)建數(shù)據(jù)表:student,表結(jié)構(gòu)如下:
列名 | 類(lèi)型 |
---|---|
id | int |
name | varchar(20) |
gender | char(1) |
age | int |
代碼下載地址
四、總結(jié)
本文我們學(xué)習(xí)了如何使用Spring JDBC來(lái)訪問(wèn)數(shù)據(jù)庫(kù),并且以常用數(shù)據(jù)庫(kù)MySQL為例進(jìn)行代碼實(shí)戰(zhàn)練習(xí)。Spring的JDBC抽象極大的提升了程序員對(duì)數(shù)據(jù)庫(kù)開(kāi)發(fā)的效率,不需要再關(guān)心數(shù)據(jù)庫(kù)連接獲取、準(zhǔn)備和運(yùn)行句子、檢索結(jié)果集、關(guān)閉連接等繁雜的工作。但通過(guò)前面的實(shí)戰(zhàn)練習(xí)也許你也發(fā)現(xiàn)構(gòu)造SQL語(yǔ)句,為句子中的參數(shù)賦值;若獲取自定義定義對(duì)象則需要從結(jié)果集獲取數(shù)據(jù)并對(duì)自定義對(duì)象賦值。我們將在下篇文章學(xué)習(xí)如何解決這些問(wèn)題,進(jìn)一步提升工作效率。