Spring學(xué)習(xí)手冊(cè)(12)—— Spring JDBC數(shù)據(jù)庫(kù)訪問(wèn)

通過(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ù),首先要配置DataSourcebean。
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.usernamejdbc.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)一步陳述了。但需要注意的是,具體的driverClassNameurl配置方式,需要查看具體的數(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)、插入、更新、刪除。我們這里大致介紹下queryupdate等方法。

查找

查詢(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)備工作:

  1. 創(chuàng)建數(shù)據(jù)庫(kù):score_manager。
  2. 創(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)一步提升工作效率。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,698評(píng)論 6 539
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,202評(píng)論 3 426
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 177,742評(píng)論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,580評(píng)論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,297評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,688評(píng)論 1 327
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,693評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,875評(píng)論 0 289
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,438評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,183評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,384評(píng)論 1 372
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,931評(píng)論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,612評(píng)論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 35,022評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 36,297評(píng)論 1 292
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,093評(píng)論 3 397
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,330評(píng)論 2 377

推薦閱讀更多精彩內(nèi)容

  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,911評(píng)論 6 342
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,816評(píng)論 18 139
  • 主要內(nèi)容 定義Spring的數(shù)據(jù)訪問(wèn)支持 配置數(shù)據(jù)庫(kù)資源 使用Spring提供的JDBC模板 寫(xiě)在前面:經(jīng)過(guò)上一篇...
    程序熊大閱讀 8,782評(píng)論 1 31
  • 第一、2017年的夢(mèng)想清單 工作 1.暑假努力實(shí)習(xí),了解出版社的美編工作。 2.畫(huà)更多的透視圖,不局限于人像,爭(zhēng)取...
    ARTQRT閱讀 315評(píng)論 2 3
  • 整個(gè)知識(shí)管理體系當(dāng)中,閱讀占的比重非常大,其實(shí)無(wú)論是寫(xiě)作,做其他的輸出,還是做個(gè)人品牌,我們都會(huì)發(fā)現(xiàn)一個(gè)問(wèn)題,就是...
    牛小丫53閱讀 177評(píng)論 0 2