SpringBoot:3.SpringBoot使用Spring-data-jpa實(shí)現(xiàn)數(shù)據(jù)庫(kù)訪(fǎng)問(wèn)

做Web開(kāi)發(fā),首先要能將數(shù)據(jù)渲染到網(wǎng)頁(yè)中展示,其次是要獲取數(shù)據(jù)庫(kù)數(shù)據(jù)展示到視圖層,在前面的文章SpringBoot整合Thymeleaf模板引擎渲染web視圖,我們實(shí)現(xiàn)了從后端數(shù)據(jù)展示到視圖層,那么下面我們通過(guò)使用Spring Data Jpa來(lái)實(shí)現(xiàn)簡(jiǎn)單的數(shù)據(jù)庫(kù)增刪改查功能。

1.pom.xml中引入依賴(lài)的jar包

        <!--Web開(kāi)發(fā)依賴(lài)-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--junit測(cè)試依賴(lài)-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <!--jpa依賴(lài)-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>   
        <!-- 引入MySQL連接的依賴(lài)包 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.21</version>
        </dependency>                                       

因?yàn)轫?xiàng)目是基于Spring Boot 2.1.9.RELEASE版本的,而我本地使用的Mysql是5.X版本,所以這里引入mysql依賴(lài)的時(shí)候添加了版本信息,這樣可以引入相應(yīng)的版本,如果不加版本信息將會(huì)引入8.X版本的Mysql,可能在運(yùn)行項(xiàng)目測(cè)試出現(xiàn)錯(cuò)誤。

2.配置application.properties數(shù)據(jù)庫(kù)信息

# 數(shù)據(jù)庫(kù)連接信息
spring.datasource.url=jdbc:mysql://localhost:3306/test
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
# 自動(dòng)創(chuàng)建表結(jié)構(gòu)的設(shè)置
spring.jpa.properties.hibernate.hbm2ddl.auto=create-drop

spring.jpa.properties.hibernate.hbm2ddl.auto是hibernate的配置屬性,其主要作用是:自動(dòng)創(chuàng)建、更新、驗(yàn)證數(shù)據(jù)庫(kù)表結(jié)構(gòu)。該參數(shù)的幾種配置如下:

  • create:每次加載hibernate時(shí)都會(huì)刪除上一次的生成的表,然后根據(jù)你的model類(lèi)再重新來(lái)生成新表,哪怕兩次沒(méi)有任何改變也要這樣執(zhí)行,這就是導(dǎo)致數(shù)據(jù)庫(kù)表數(shù)據(jù)丟失的一個(gè)重要原因。
  • create-drop:每次加載hibernate時(shí)根據(jù)model類(lèi)生成表,但是sessionFactory一關(guān)閉,表就自動(dòng)刪除
  • update:最常用的屬性,第一次加載hibernate時(shí)根據(jù)model類(lèi)會(huì)自動(dòng)建立起表的結(jié)構(gòu)(前提是先建立好數(shù)據(jù)庫(kù)),以后加載hibernate時(shí)根據(jù)model類(lèi)自動(dòng)更新表結(jié)構(gòu),即使表結(jié)構(gòu)改變了但表中的行仍然存在不會(huì)刪除以前的行。要注意的是當(dāng)部署到服務(wù)器后,表結(jié)構(gòu)是不會(huì)被馬上建立起來(lái)的,是要等應(yīng)用第一次運(yùn)行起來(lái)后才會(huì)。
  • validate:每次加載hibernate時(shí),驗(yàn)證創(chuàng)建數(shù)據(jù)庫(kù)表結(jié)構(gòu),只會(huì)和數(shù)據(jù)庫(kù)中的表進(jìn)行比較,不會(huì)創(chuàng)建新表,但是會(huì)插入新值。

3.創(chuàng)建實(shí)體類(lèi)

創(chuàng)建一個(gè)User實(shí)體,包含id(主鍵)、name(姓名)、age(年齡)屬性,通過(guò)ORM框架其會(huì)被映射到數(shù)據(jù)庫(kù)表中,由于配置了hibernate.hbm2ddl.auto,并且配置的值為:create-drop,在應(yīng)用啟動(dòng)的時(shí)候框架會(huì)自動(dòng)去數(shù)據(jù)庫(kù)中創(chuàng)建對(duì)應(yīng)的表,應(yīng)用停止時(shí)則會(huì)自動(dòng)刪除表信息。

package com.w3cjava.entity;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;

@Entity
public class User {
    @Id
    @GeneratedValue
    private Long id;

    @Column(nullable = false)
    private String name;

    @Column(nullable = false)
    private Integer age;

    public User() {
    }

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Long getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

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

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }   
}

4.創(chuàng)建數(shù)據(jù)訪(fǎng)問(wèn)接口

下面針對(duì)User實(shí)體創(chuàng)建對(duì)應(yīng)的Repository接口實(shí)現(xiàn)對(duì)該實(shí)體的數(shù)據(jù)訪(fǎng)問(wèn),如下代碼:

package com.w3cjava.dao;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;

import com.w3cjava.entity.User;
/**
 * 
 * @class  UserRepository
 * @version 1.0.0
 * @author cos
 * @desc   創(chuàng)建對(duì)應(yīng)的Repository接口實(shí)現(xiàn)對(duì)該實(shí)體的數(shù)據(jù)訪(fǎng)問(wèn)
 *
 */
public interface UserRepository extends JpaRepository<User, Long> {

    User findByName(String name);

    User findByNameAndAge(String name, Integer age);

    @Query("from User u where u.name=:name")
    User findUser(@Param("name") String name);

}

通過(guò)繼承JpaRepository接口,已經(jīng)實(shí)現(xiàn)了基礎(chǔ)的增(save)刪(delete)改(save)查(findAll、findOne),具體可以查看API文檔,因此開(kāi)發(fā)過(guò)程中,基本的數(shù)據(jù)訪(fǎng)問(wèn)層不需要開(kāi)發(fā)者再自己定義。

項(xiàng)目開(kāi)發(fā)過(guò)程中,往往對(duì)數(shù)據(jù)的訪(fǎng)問(wèn)都比較復(fù)雜,這時(shí)候就需要程序員自己定義相對(duì)復(fù)雜的Sql語(yǔ)句。具體可以參考API文檔。

Spring Data Jpa還提供了通過(guò)解析方法名創(chuàng)建查詢(xún),比如上面的代碼:

    User findByName(String name);

    User findByNameAndAge(String name, Integer age);

5.單元測(cè)試

整合Spring Data Jpa所需要的基本類(lèi)和配置已經(jīng)準(zhǔn)備好,下面進(jìn)行簡(jiǎn)單的單元測(cè)試。

package com.w3cjava.service;

import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;

import com.w3cjava.SpringBootSpringDataJpaApplication;
import com.w3cjava.dao.UserRepository;
import com.w3cjava.entity.User;

@RunWith(SpringRunner.class)
@SpringBootTest(classes = {SpringBootSpringDataJpaApplication.class})
public class UserServiceTest {
    @Autowired
    private UserRepository userRepository;

    @Test
    public void test() throws Exception {

        // 創(chuàng)建10條記錄
        userRepository.save(new User("A", 10));
        userRepository.save(new User("B", 20));
        userRepository.save(new User("C", 30));
        userRepository.save(new User("D", 40));
        userRepository.save(new User("E", 50));
        userRepository.save(new User("F", 60));
        userRepository.save(new User("G", 70));
        userRepository.save(new User("H", 80));
        userRepository.save(new User("I", 90));
        userRepository.save(new User("J", 100));

        // 測(cè)試findAll, 查詢(xún)所有記錄
        Assert.assertEquals(10, userRepository.findAll().size());

        // 測(cè)試findByName, 查詢(xún)姓名為FFF的User
        Assert.assertEquals(60, userRepository.findByName("F").getAge().longValue());

        // 測(cè)試findUser, 查詢(xún)姓名為FFF的User
        Assert.assertEquals(60, userRepository.findUser("F").getAge().longValue());

        // 測(cè)試findByNameAndAge, 查詢(xún)姓名為FFF并且年齡為60的User
        Assert.assertEquals("F", userRepository.findByNameAndAge("F", 60).getName());

        // 測(cè)試刪除姓名為AAA的User
        userRepository.delete(userRepository.findByName("A"));

        // 測(cè)試findAll, 查詢(xún)所有記錄, 驗(yàn)證上面的刪除是否成功
        Assert.assertEquals(9, userRepository.findAll().size());

    }
}

6.參考文章

Cannot load driver class: com.mysql.jdbc.Driver: https://blog.csdn.net/linmengmeng_1314/article/details/86291180
spring Boot--junit單元測(cè)試: https://blog.csdn.net/adminBfl/article/details/86518748
源碼:03.Spring-Boot-SpringDataJpa

歡迎掃面下列二維碼關(guān)注“余弦的自留地”公眾微信號(hào)

在這里插入圖片描述

萬(wàn)物之中,希望至美

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容