Spring JPA 关系映射系列教程:OneToOne 关系映射详解

这是JPA 关系映射 系列教程的第一篇:JPA One-To-One 外键关系映射

JPA 关系映射系列(SPring Boot, Postgresql):

  1. JPA One-To-One 外键 关系映射
  2. JPA One-To-Many 关系映射
  3. JPA Many-To-Many 关系映射

为了完成这边教程你所需要的工具如下:

Spring Data JPA
Spring Boot
Postgresql 数据库

如何在ubuntu下安装Postgresql以及图形界面客户端PgAdmin3,请戳我

项目依赖

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>demo</name>
    <description>Demo project for Spring Boot</description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.5.6.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
    </properties>

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

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.postgresql</groupId>
            <artifactId>postgresql</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>


</project>

One To One 关系:
在这里 我们使用 book 以及 book_detail 来描述一对一的关系。

Selection_054.png

下面我们来看看如果使用 Spring Boot 来定义实体映射:

package com.example.demo;

import javax.persistence.*;

@Entity
public class Book {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private long id;
    private String name;

    @OneToOne(cascade = CascadeType.ALL)
    @JoinColumn(name = "book_detail")
    private BookDetail bookDetail;

    public Book() {
    }

    public Book(String name, BookDetail bookDetail) {
        this.name = name;
        this.bookDetail = bookDetail;
    }

    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 BookDetail getBookDetail() {
        return bookDetail;
    }

    public void setBookDetail(BookDetail bookDetail) {
        this.bookDetail = bookDetail;
    }
}

package com.example.demo;

import javax.persistence.*;

@Entity
public class BookDetail {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private long id;


    private long numberOfPages;


   @OneToOne(cascade = CascadeType.ALL,mappedBy = "bookDetail")
    private Book book;

    public BookDetail() {
    }

    public BookDetail(long numberOfPages, Book book) {
        this.numberOfPages = numberOfPages;
     
    }

    public long getId() {
        return id;
    }

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

    public long getNumberOfPages() {
        return numberOfPages;
    }

    public void setNumberOfPages(long numberOfPages) {
        this.numberOfPages = numberOfPages;
    }

    public Book getBook() {
        return book;
    }

    public void setBook(Book book) {
        this.book = book;
    }
}

Spring Boot 中所有的实体类都需要用注解 @Entity 来标记,被这个注解标记的实体类将会在数据库中生成一个对应的 Table。表名通常情况下以类名作为默认表名, 当然你也可以指定一个已经存在的表来对应这个实体类。比如:

            @Entity
            @Table(name ="book_specific")

@Id 标记一个字段为 主键(Primary Key). 当一个主键字段被定义的时候,主键的值将会被 ObjectDB 自动注入到这个字段中。

@GeneratedValue 通常情况下,关系数据库会为每一个数据库维护一个特殊的全局数字生成器。这个数字生成器会自动为每一个没有定义主键字段的实体对象生成一个 ID。

    @Id
    @GeneratedValue
    private long id;

commit期间, AUTO策略使用全局数字生成器为每个新的实体对象生成主键。这些生成的值在数据库级别是唯一的,不会被
回收,这些主键值被多个表共享 。

简单的来解释一下这里所谓的主键值被多个表共享是什么意思,
    例如我们现在一个数据库,里面有两张表 A1 跟 A2。
A1里面有三条数据 A11 A12 A13
A2里面有三条数据 A21 A22 A23,
每一个表的每一条数据的主键id通常情况是自增加的,
所以理论上的A11 的主键id=1,A12的主键id=2,A13的主键id=3
所以理论上的A21 的主键id=1,A22的主键id=2,A23的主键id=3

但是如果你使用AUTO的策略的话,主键的增加是根据你将数值插入数据库的顺序来决定的。
如果你先在A1表中插入 A11 A12 A13三条数据,然后在A2表中插入A21 A22 A23三条数据
那们的主键的id就会变成 
A11 的主键id=1,A12的主键id=2,A13的主键id=3
A21 的主键id=4,A22的主键id=5,A23的主键id=6
如果你两个表是相互交替的插入数据,那么主键的生成也是相互的交替的,
    这就导致了每一张表的主键ID都不一定是从1开始增加的
    这取决你插入数据的顺序

这就是所谓的主键值被多个表共享

所以通常情况下我们为了保证每一个表的主键ID都从一开始增加,我们需要设置
@GeneratedValue(strategy=GenerationType.IDENTITY

关于JPA中主键的生成策略,一共有四种模式:
@GeneratedValue(strategy=GenerationType.AUTO)
@GeneratedValue(strategy=GenerationType.IDENTITY)
@GeneratedValue(strategy=GenerationType.SEQUENCE)
@GeneratedValue(strategy=GenerationType.TABLE)

如果大家对这方面感兴趣的话,请点击关注楼主,或者给楼主留言,楼主会考虑写一篇专门的文章来讲解JPA中主键的生成策略。

@OneToOne 在两个实体之间定义了一对一的关系,这里Book与BookDetail是一对一的关系。
mappedBy=”bookDetail” 定义了两张表之间由谁来维护彼此的关系。这里表示由Book这张表来维护两者之间的关系。这里
BookDetail类中的mappeBy的值 bookDetail 是实体类Book中对BookDetail的引用字段名。当我们定义了mappedBy属性之后, JPA就会在Book表中新增加一列bookDetail,它的值就是BookDetail表中的主键值。
@JoinColumn 通常情况下,如果不指定这个属性,那么JPA会默认帮你在Book类中新增一列以bookDetail为默认名字,如果你指定了这个属性,那么JPA会按照你给的属性命为新增的一列命名。

当然了关于@JoinColumn以及mappendBy这两个属性的用法不仅仅与此,事实上他们是JPA中关系映射非常重要的注解方法。
当然本文不在讨论范围之类。在之后的OneToMany以及 ManyToMany 教程中中我会详细的讨论这两个属性

当然了如果你这么定义:

class Book(){
    
@OneToOne(cascade = CascadeType.ALL,mappedBy = "book")

private BookDetail bookDetail; 

}

class BookDetail(){

@OneToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "book_id")
private Book book;

}

也是可以的,只不过这回在BookDetail表中新增加的一列,他的值就是Book表中的主键值。 这一次有BookDetail来维护两个表之间的关系。

OneToOne关系中,谁来维护关系并不是特别重要,因为两者是等价的,你可以引用我的主键,同样我也可以引用你的主键。 因此这跟OneToManyManyToMany是不一样的。 例如在OneToMany中,关系的维护方永远是Many的那一方。比如班级跟学生就是一对多的关系,这个时候就必须要在学生表中引用班级的主键作为一列。 因为我们的表是不能保存集合的关系。 在班级表中,每一个班级都有很多学生,这是一个集合,这没有办法用数据库在保存。 但是在学生表中,每一个学生都对应唯一一个班级。 所以我们可以引用班级的主键作为学生表的外键。 这就是为什么在一对多一方,关系的维护者永远是Many的那一方了。

对于OneToManyManyToMany的关系映射,是这一系列教程的后两篇文章。 稍后我会详细解释这两种映射关系

下一步我们需要给Spring Boot配置数据库,以便它能够连接到我们的数据库:
在Spring Boot的application.properties中添加以下语句:

# ===============================
# = DATA SOURCE
# ===============================
# Set here configurations for the database connection
spring.datasource.url=jdbc:postgresql://localhost:5432/jpa1
spring.datasource.username=postgres
spring.datasource.password=9145190618
spring.datasource.driver-class-name=org.postgresql.Driver
# Keep the connection alive if idle for a long time (needed in production)
spring.datasource.testWhileIdle=true
spring.datasource.validationQuery=SELECT 1
# ===============================
# = JPA / HIBERNATE
# ===============================
# Show or not log for each sql query
spring.jpa.show-sql=true
# Hibernate ddl auto (create, create-drop, update): with "create-drop" the database
# schema will be automatically created afresh for every start of application
spring.jpa.hibernate.ddl-auto=create-drop

# Naming strategy
spring.jpa.hibernate.naming.implicit-strategy=org.hibernate.boot.model.naming.ImplicitNamingStrategyLegacyHbmImpl
spring.jpa.hibernate.naming.physical-strategy=org.springframework.boot.orm.jpa.hibernate.SpringPhysicalNamingStrategy

# Allows Hibernate to generate SQL optimized for a particular DBMS
spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.PostgreSQLDialect


#There is also a /shutdown endpoint, but its only visible by default via JMX. To enable it as an HTTP endpoint, add endpoints.shutdown.enabled=true to your application.properties file.
endpoints.shutdown.enabled=true

如果你没有实现创建数据库,那么你需要手动用图形GUI或者在命令行中创建数据库

pring.datasource.url=jdbc:postgresql://localhost:5432/jpa 指明了数据库为jpa,监听端口为5432,这是Postgresql数据库的默认监听端口。

这里pring.jpa.hibernate.ddl-auto可以是none,update,create,create-drop,有关详细信息,请参考Hibernate文档。

        none: 这的默认值,不改变数据库结构。(一般在部署阶段,可以使用这个模式)
        update: Hibernate根据给定的实体结构更改数据库。
        create: 每次创建数据库,但不要在关闭时丢弃。
        create-drop: 创建数据库,然后在SessionFactory关闭时将其删除(在开发阶段,可以使用这个方式,因为每次程序启动时候   都会删除已有的数据,在重新创建)。我们这里从创建开始,因为我们还没有数据库结构。第一次运行后,我们可以根据程序要求将其切换为更新或无更新。当您想对数据库结构进行一些更改时,请使用更新。    

我们运行程序的时候,我们可以看到:数据库中已经自动创建了两张表了:

Book 表
BookDetail 表

最右边是外建,其值为book表中主键的id值

Spring Data JPA 包含了一些内置的Repository实现了一些常用的操作数据库的功能, 比如:findOnefindAllsave 等等。下面根据上面两个实体类,我们在通过JPA提供给我们的Repository来简单的操作我们的数据库。

我们所需要做的事情就是为我们的实体类 Book 和 BookDetail 建立对应的Repository接口:

package com.example.demo;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.stereotype.Repository;

@Repository
public interface BookRepository extends JpaRepository<Book, Long> {
}

package com.example.demo;

import org.springframework.data.jpa.repository.JpaRepository;

public interface BookDetailRepository extends JpaRepository<BookDetail, Long> {
}

下面我们运行APP来使用Repository新建数据库数据:

package com.example.demo;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

import java.util.ArrayList;
import java.util.List;

@SpringBootApplication
public class DemoApplication implements CommandLineRunner {

    @Autowired
    BookRepository bookRepository;

    @Autowired
    BookDetailRepository bookDetailRepository;

    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }

    @Override
    public void run(String... strings) throws Exception {


        List<Book> books = new ArrayList<>();
        books.add(new Book("Book A", new BookDetail(49)));
        books.add(new Book("Book B", new BookDetail(59)));
        books.add(new Book("Book C", new BookDetail(69)));
        bookRepository.save(books);

    }
}

运行结果:


Selection_060.png
Selection_061.png

如果喜欢这篇文章,请给楼主点个赞或者加个关注。下次给大家带来的是JPA One-To-Many 关系映射 以 及JPA Many-To-Many 关系映射

源码已经分享在Github

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

推荐阅读更多精彩内容

  • 转自:jpa-内心求法 JPA概要摘要:JPA定义了Java ORM及实体操作API的标准。本文摘录了JPA的一些...
    谁在烽烟彼岸阅读 1,167评论 0 4
  • 本文中我们介绍并比较两种最流行的开源持久框架:iBATIS和Hibernate,我们还会讨论到Java Persi...
    大同若鱼阅读 4,305评论 4 27
  • 作为一种轻量级的关系映射工具,Hibernate支持各种关系映射,例如:多对一、一对多和一对一的数据库表关系,通过...
    Ystrator阅读 525评论 0 1
  • 终于到了数据库操作部分了,通常我们对于数据库的操作无非是增删改查,对于单表操作而言,SQL语句大都是类似的,同时时...
    蓝色的咖啡阅读 3,622评论 0 9
  • 并发包 ConcurrentHashMap(类HashMap的并发类) CopyOnWriteArrayList(...
    小鱼游儿阅读 289评论 0 1