Spring Boot集成JPA访问数据库

终于到了数据库操作部分了,通常我们对于数据库的操作无非是增删改查,对于单表操作而言,SQL语句大都是类似的,同时时候我们的项目会移植到不同的数据库上运行,为了适应这种情况,有时候还得写不同的SQL来适配。

为了解决这种情况(当然也不可能完全解决啦),我们会借用一些ORM框架来减少我们的工作负担。本章我们来学习如何在Spring Boot中集成JPA框架来访问数据库。

JPA全称Java Persistence API.JPA通过JDK 5.0注解或XML描述对象-关系表的映射关系,并将运行期的实体对象持久化到数据库中。

JPA的查询语言是面向对象而非面向数据库的,它以面向对象的自然语法构造查询语句,可以看成是Hibernate HQL的等价物。JPA定义了独特的JPQL(Java Persistence Query Language),JPQL是EJB QL的一种扩展,它是针对实体的一种查询语言,操作对象是实体,而不是关系数据库的表,而且能够支持批量更新和修改、JOIN、GROUP BY、HAVING 等通常只有 SQL 才能够提供的高级查询特性,甚至还能够支持子查询。

为了抽象出不同对象的“增删改查”操作,通常我们会写一个模板Dao来简化我们的开发,而Spring-data-jpa可以更加减轻我们的开发工作量,使得数据访问层变成只是一层接口的编写方式,如下所示:

package com.bluecoffee.repository;

import com.bluecoffee.domain.Book;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;

/**
 * Created by qianlong on 16/9/27.
 */
public interface BookDao extends JpaRepository<Book,Long> {

    Book findByTitle(String title);

    Book findByTitleAndAuthor(String title,String author);

    @Query("from Book b where b.title=:title")
    Book findBook(@Param("title") String bookTitle);
}

我们只需要通过编写一个继承自JpaRepository的接口就能完成数据访问,下面以一个具体实例来体验Spring-data-jpa给我们带来的强大功能。

使用示例

Spring-data-jpa依赖于Hibernate。如果您已经对Hibernate有一定了解,那你可以毫不费力的看懂并上手使用Spring-data-jpa。如果您还是Hibernate新手,您可以先按如下方式入门,再建议回头学习一下Hibernate以帮助这部分的理解和进一步使用。

引入JPA依赖

pom.xml中添加JPA和MySQL数据库驱动的依赖

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.39</version>
        </dependency>

配置数据库连接信息

spring.datasource.url=jdbc:mysql://localhost:3306/spring_boot
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.driver-class-name=com.mysql.jdbc.Driver

#spring.jpa.properties.hibernate.hbm2ddl.auto=create
#spring.jpa.properties.hibernate.hbm2ddl.auto=create-drop
spring.jpa.properties.hibernate.hbm2ddl.auto=update
#spring.jpa.properties.hibernate.hbm2ddl.auto=validate

spring.jpa.properties.hibernate.hbm2ddl.auto是hibernate的配置属性,其主要作用是:自动创建、更新、验证数据库表结构。该参数的几种配置如下:

  • create:每次加载hibernate时都会删除上一次的生成的表,然后根据你的model类再重新来生成新表,哪怕两次没有任何改变也要这样执行,这就是导致数据库表数据丢失的一个重要原因。
  • create-drop:每次加载hibernate时根据model类生成表,但是sessionFactory一关闭,表就自动删除。
  • update:最常用的属性,第一次加载hibernate时根据model类会自动建立起表的结构(前提是先建立好数据库),以后加载hibernate时根据model类自动更新表结构,即使表结构改变了但表中的行仍然存在不会删除以前的行。要注意的是当部署到服务器后,表结构是不会被马上建立起来的,是要等应用第一次运行起来后才会。
  • validate:每次加载hibernate时,验证创建数据库表结构,只会和数据库中的表进行比较,不会创建新表,但是会插入新值。

创建实体

package com.bluecoffee.domain;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import java.util.Date;

/**
 * Created by qianlong on 16/10/2.
 */
@Entity
public class Book {

    @Id
    @GeneratedValue
    private Long bookId;

    @Column(nullable = false)
    private String title;

    @Column(nullable = true)
    private String author;

    @Column(name = "create_time" ,nullable = true)
    private Date createTime;

    public Book(){}

    public Book(String title,String author,Date createTime){
        this.title = title;
        this.author = author;
        this.createTime = createTime;
    }

    public Long getBookId() {
        return bookId;
    }

    public void setBookId(Long bookId) {
        this.bookId = bookId;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public String getAuthor() {
        return author;
    }

    public void setAuthor(String author) {
        this.author = author;
    }

    public Date getCreateTime() {
        return createTime;
    }

    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }
}

在上述代码中,注解@GeneratedValue表示bookId我们设置为主键从1开始自增,但是通常我们在实际业务开发过程中,主键需要从自定义序列中获取,我们会建一张表,然后在表中维护不同表主键的自增序列,做法如下:

新建序列号生成表
DROP TABLE IF EXISTS `sequence_generator`;
CREATE TABLE `sequence_generator` (
  `id` decimal(10,0) NOT NULL,
  `sequence_name` varchar(255) NOT NULL,
  `sequence_value` decimal(10,0) NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

对book_store表新建自增序列
INSERT INTO `sequence_generator` VALUES ('1', 'book_store_pk', '10000000');

修改实体类中主键生成规则
 package com.bluecoffee.domain;

import javax.persistence.*;

/**
 * Created by qianlong on 16/10/2.
 */
@Entity
@Table(name="book_store")
public class BookStore {

    @Id
    @GeneratedValue(strategy = GenerationType.TABLE, generator="pk_gen")
    @TableGenerator(name = "pk_gen",
            table="sequence_generator",
            pkColumnName="sequence_name",
            valueColumnName="sequence_value",
            pkColumnValue="book_store_pk",
            allocationSize=1
    )
    private Long bookId;

    @Column(nullable = true)
    private String address;

    @Column(nullable = true)
    private String storeManager;

    public BookStore(){};

    public BookStore(String address,String storeManager){
        this.address = address;
        this.storeManager = storeManager;
    }

    public Long getBookId() {
        return bookId;
    }

    public void setBookId(Long bookId) {
        this.bookId = bookId;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    public String getStoreManager() {
        return storeManager;
    }

    public void setStoreManager(String storeManager) {
        this.storeManager = storeManager;
    }
}

@GeneratedValue(strategy = GenerationType.TABLE, generator="pk_gen") 代表主键生成策略是从表中获取序列,序列号生成器名称为“pk_gen”

@TableGenerator的定义(Java代码)

@Target({TYPE, METHOD, FIELD})   

@Retention(RUNTIME)  

public @interface TableGenerator {  

  String name();  

  String table() default "";  

  String catalog() default "";  

  String schema() default "";  

  String pkColumnName() default "";  

  String valueColumnName() default "";  

  String pkColumnValue() default "";  

  int initialValue() default 0;  

  int allocationSize() default 50;  

  UniqueConstraint[] uniqueConstraints() default {};  

}  

name:序列号生成器名称,与@GeneratedValue中的generator一致
table:表生成策略所持久化的表名,例如,这里表使用的是数据库中的“sequence_generator”。
pkColumnName:表示在持久化表中,该主键生成策略所对应键值的名称。例如在“sequence_generator”中将“sequence_name”作为主键的键值
valueColumnName:表示在持久化表中,该主键当前所生成的值,它的值将会随着每次创建累加。例如,在“sequence_generator”中将“sequence_value”作为主键的值
pkColumnValue:表示在持久化表中,该生成策略所对应的主键。例如在“sequence_generator”表中,将“sequence_name”的值为“book_store_pk”。
allocationSize:表示每次主键值增加的大小,例如设置成1,则表示每次创建新记录后自动加1,默认为50。
initialValue:表示主键初识值,默认为0。
UniqueConstraint:与@Table标记中的用法类似。

创建数据访问dao

package com.bluecoffee.repository;

import com.bluecoffee.domain.Book;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;

/**
 * Created by qianlong on 16/9/27.
 */
public interface BookDao extends JpaRepository<Book,Long> {

    Book findByTitle(String title);

    Book findByTitleAndAuthor(String title,String author);

    @Query("from Book b where b.title=:title")
    Book findBook(@Param("title") String bookTitle);
}

BookDao继承了==JpaRepository==,我们通过查看JpaRepository的API接口文档可以看到该接口已经实现了创建(save)、更新(save)、删除(delete)、查询(findAll、findOne)等基本操作的函数,因此对于这些基础操作的数据访问就不需要我们再开发了。

  • Book findByTitle(String title);
  • Book findByTitleAndAuthor(String title,String author);

findByTitle和findByTitleAndAuthor方法实现了根据书名和作者查询的方法,可以看到我们并没有编写任何SQL就完成了查询功能,这就是spring-data-jpa的一大特色:通过解析方法名创建查询。 但是我们需要注意,查询访问名称必须遵循JPA的规范,查询方法名称必须以find开头,否则就需要自己通过@Query 注解来创建查询,需要编写JPQL语句,并通过类似“:title”来映射@Param指定的参数,如下所示:

    @Query("from Book b where b.title=:title")
    Book findBook(@Param("title") String bookTitle);

分页查询

    @Query("select book from Book book where book.author = :author")
    Page findBookPage(Pageable pageable,@Param("author") String author);

单元测试

最后我们来通过编写单元测试来验证,如下代码所示:

package com.bluecoffee;

import com.bluecoffee.Repository.BookDao;
import com.bluecoffee.Repository.BookStoeDao;
import com.bluecoffee.domain.Book;
import com.bluecoffee.domain.BookStore;
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.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import java.util.Date;
import java.util.Iterator;

/**
 * Created by qianlong on 16/9/27.
 */
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = Application.class)
public class BookDaoTest {

    @Autowired
    private BookDao bookDao;

    @Autowired
    private BookStoeDao bookStoeDao;

    @Test
    public void testBook(){
        try{
            //先清空已有数据
            bookDao.deleteAll();

            //生成10本书
            for(int i=1;i<=10;i++){
                bookDao.save(new Book("book"+i,"author"+i,new Date()));
            }

            // 测试findAll, 查询所有记录
            Assert.assertEquals(10, bookDao.findAll().size());

            // 测试findByTitle, 查询书名为book5的书
            Assert.assertEquals("author5", bookDao.findByTitle("book5").getAuthor());

            // 测试findBook, 查询书名为book7的书
            Assert.assertEquals("author7", bookDao.findBook("book7").getAuthor());

            // 测试findByTitleAndAuthor, 查询书名为book1,作者为author1的书
            Assert.assertEquals("author1", bookDao.findByTitleAndAuthor("book1","author1").getAuthor());

            // 测试findByTitleAndAuthor, 查询书名为book2,作者为author1的书
            Assert.assertEquals(null, bookDao.findByTitleAndAuthor("book2", "author1"));

            //测试删除book3
            bookDao.delete(bookDao.findBook("book3"));

            //测试删除是否成功
            Assert.assertEquals(9, bookDao.findAll().size());

            //分页查询
            bookDao.deleteAll();
            for(int i=1;i<=10;i++){
                bookDao.save(new Book("book"+i,"Alex Qian",new Date()));
            }
            Sort sort = new Sort(Sort.Direction.DESC, "bookId");
            int page = 1;
            int size = 5;
            Pageable pageable = new PageRequest(page, size, sort);
            Page<Book> pages = bookDao.findBookPage(pageable,"Alex Qian");

            Iterator<Book> it= pages.iterator();

            Assert.assertEquals(size,pages.getSize());
            Assert.assertEquals(2,pages.getTotalPages());

            while(it.hasNext()){
                Book book = (Book)it.next();
                System.out.println("title/author/createTime:"+book.getTitle()+"/"+book.getAuthor()+"/"+book.getCreateTime());
            }

        }catch (Exception ex){
            Assert.fail(ex.getMessage());
        }
    }

    @Test
    public void testBookStore(){
        try{
            bookStoeDao.deleteAll();

            //生成5个书店
            for(int i=1;i<=5;i++){
                bookStoeDao.save(new BookStore("address_"+i,"manager_"+i));
            }
            // 测试findAll, 查询所有记录
            Assert.assertEquals(5, bookStoeDao.findAll().size());
            Assert.assertEquals(1, bookStoeDao.getBookStoreByManager("manager_2").size());
            Assert.assertEquals("manager_2", bookStoeDao.getBookStoreByManager("manager_2").get(0).getStoreManager());

        }catch (Exception ex){
            Assert.fail(ex.getMessage());
        }
    }

}


完整代码戳这里: Chapter 4-1-1 - Spring Boot集成JPA访问数据库

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,670评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,928评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,926评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,238评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,112评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,138评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,545评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,232评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,496评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,596评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,369评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,226评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,600评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,906评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,185评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,516评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,721评论 2 335

推荐阅读更多精彩内容

  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,679评论 6 342
  • 2017年8月21日 我原本只想简单记录一下springboot中应用Jpa的简单操作。不想由于hibernate...
    行者N阅读 6,454评论 0 23
  • 要加“m”说明是MB,否则就是KB了. -Xms:初始值 -Xmx:最大值 -Xmn:最小值 java -Xms8...
    dadong0505阅读 4,795评论 0 53
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • 2017年11月27日 星期一 晴 早上看了一则新闻,在印度公交车上,一个青年丢了钱包,怀疑周边四个学生偷的,遂上...
    翔哥山姆阅读 258评论 0 0