【Spring JPA总结】JPA One-To-One实现的4种方式

【参考】
https://hellokoding.com/one-to-one-mapping-in-jpa-and-hibernate/


本文介绍Spring Jpa的One-To-One关联的4种方式,包含:
外键关联下的双向和单向关联
外键关联的意思是两张表都有自己的主键,一张表的主键作为外键存在于另一张表中:

外键关联下的双向和单向关联

共同主键下的双向和单向关联
共同主键的意思是两张表的主键一样,一张表的主键同时又是另一张表的主键+外键:

共同主键下的双向和单向关联


1. 外键关联

【数据模型】学生,和学生证:一个学生只有一张学生证,学生证也只属于某一个学生。


image.png
1.1 使用@OneToOne@JoinColumn来实现双向的外键关联

双向的One-To-One关联,在两个entity中都需要用到注解@OneToOne

  • Student实体中,@OneToOne是主动的一方,@JoinColumn表示外键的列,name属性是用来标识表中所对应的字段的名称。
  • StudentCard中,则是被关联的一方,@OneToOne需要写mappedBy的值,这个的含义是被映射(即这方不用管关联关系),指向Student实体类。
@Entity
@Table(name = "student")
public class Student {

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

    private String name;

    @OneToOne(cascade = CascadeType.ALL, optional = false)
    @JoinColumn(name = "student_card_id")
    private StudentCard studentCard;
}
@Entity
@Table(name = "student_card")
public class StudentCard {

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

    @Column(unique = true, nullable = false)
    private String code = UUID.randomUUID().toString();

    @OneToOne(mappedBy = "studentCard")
    private Student student;
}

【测试a】主entity(student)中findById,能查到两边的数据,sql:

select
student0_.id as id1_3_0_,
student0_.name as name2_3_0_,
student0_.student_card_id as student_3_3_0_,
studentcar1_.id as id1_4_1_,
studentcar1_.code as code2_4_1_
from
student student0_
inner join
student_card studentcar1_
on student0_.student_card_id=studentcar1_.id
where
student0_.id=?

【测试b】子entity(student_card)中findById,能查到两边的数据,sql:

select
studentcar0_.id as id1_4_0_,
studentcar0_.code as code2_4_0_,
student1_.id as id1_3_1_,
student1_.name as name2_3_1_,
student1_.student_card_id as student_3_3_1_
from
student_card studentcar0_
left outer join
student student1_
on studentcar0_.id=student1_.student_card_id
where
studentcar0_.id=?

【测试c】在主entity中测试插入,将会插入数据到两张表中:

    @Test
    public void saveTest() {
        Student student = Student.builder().name("test1").studentCard(new StudentCard()).build();
        studentRepository.save(student);
    }

【测试d】在子entity中无法插入主entity的数据,但能单独创建子entity自己的数据,不过因为它的id并没有关联到主entity的表中,也就没有什么意义(像是一个没有学生的学生证)。

【测试e】在主entity中删除数据,同时会删除关联的子entity数据。

【测试f】在子entity中尝试删除数据,但因为有外键的关联,无法删除数据。(除非它本身的数据并没有被主entity关联到,是个orphan data):

    @Test
    public void delete() {
        studentCardRepository.deleteById(1);
    }
1.2 使用@OneToOne@JoinColumn来实现单向的外键关联

数据模型和#1.1一样。

单向的One-To-One关联,只需要在Student实体中用到注解@OneToOne

  • Student实体中,@OneToOne是主动的一方,@JoinColumn表示外键的列,name属性是用来标识表中所对应的字段的名称。(同双向的关联时配置一样)。
  • StudentCard中,则不需要配置。
@Entity
@Table(name = "student")
public class Student {

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

    private String name;

    @OneToOne(cascade = CascadeType.ALL, optional = false)
    @JoinColumn(name = "student_card_id")
    private StudentCard studentCard;
}
@Entity
@Table(name = "student_card")
public class StudentCard {

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

    @Column(unique = true, nullable = false)
    private String code = UUID.randomUUID().toString();
}

【测试a】主entity中的findById,查询同上述#1.1一样,会inner join子entity,返回两张表的数据。

【测试b】而由于是单向关联,子entity中findById 并不会同上述#1.1一样有left outer join,只会返回自己的数据,sql:

select
studentcar0_.id as id1_4_0_,
studentcar0_.code as code2_4_0_
from
student_card studentcar0_
where
studentcar0_.id=?

【测试c】主entity插入同上述#1.1一样,执行后可以同时插入两张表的数据。

【测试d】子entity插入同上述#1.1一样,可单独插入自己的数据(不过并没有意义)。

【测试e】主entity删除同上述#1.1一样,执行后可以同时删除两张表的数据。

【测试f】子entity能单独删除自己的数据。


2. 共同主键

【数据模型】员工和员工的薪水:一个员工有自己的薪水,每个员工的薪水都不一样,薪水表中的employee_id即为主表employee中的id。
image.png
2.1 使用@OneToOne@MapsId来实现双向的共同主键关联

共同主键表示两个实体共享主键,两个实体类都需要写@OneToOne。在子entity中,主键同时又是外键。

  • 双向的关系,所以需要在主entity也加上@OneToOne。主entity上有mappedBy,因为共同主键关联时,子entity占据了主导地位。
  • 在子entity中,需要再加上@MapsId,表示该列也是主键。而该列同时需要加上@JoinColumn,表示它是外键(即关联到主entity的id)。
@Entity
@Table(name = "employee")
public class Employee {

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

    private String name;

    @OneToOne(cascade = CascadeType.ALL, mappedBy = "employee")
    private EmployeeSalary employeeSalary;
}
@Entity
@Table(name = "employee_salary")
public class EmployeeSalary {

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

    private double salary;

    @OneToOne(cascade = CascadeType.ALL, optional = false)
    @JoinColumn(name = "employee_id")
    @MapsId
    private Employee employee;
}

【测试a】主entity中findById,能查到两边的数据,sql:

select
employee0_.id as id1_1_0_,
employee0_.name as name2_1_0_,
employeesa1_.employee_id as employee1_2_1_,
employeesa1_.salary as salary2_2_1_
from
employee employee0_
left outer join
employee_salary employeesa1_
on employee0_.id=employeesa1_.employee_id
where
employee0_.id=?

【测试b】子entity中findById,能查到两边的数据,这里的sql会查两遍,第一遍是子entity自己的表,即employee_salary,第二遍是自己的表+left outer join关联到主entity表。这样看来效率比较低下。具体sql:

select
employeesa0_.employee_id as employee1_2_0_,
employeesa0_.salary as salary2_2_0_
from
employee_salary employeesa0_
where
employeesa0_.employee_id=?

select
employee0_.id as id1_1_0_,
employee0_.name as name2_1_0_,
employeesa1_.employee_id as employee1_2_1_,
employeesa1_.salary as salary2_2_1_
from
relation_study_employee employee0_
left outer join
employee_salary employeesa1_
on employee0_.id=employeesa1_.employee_id
where
employee0_.id=?

【测试c】在#2.1一开始解释了,虽然employee是主entity,但其实共同主键子entity(即employee_salary)占据了主导地位,所以在主entity中想要插入两张表的数据,需要将employee entity set回salary中才可以,如:

    @Test
    public void save() {
        Employee employee = Employee.builder().name("mark").build();

        EmployeeSalary employeeSalary = new EmployeeSalary();
        employeeSalary.setSalary(9000d);
        employee.setEmployeeSalary(employeeSalary);
        employeeSalary.setEmployee(employee);

        employeeRepository.save(employee);
    }

【测试d】尝试在employee_salary中插入两张表的数据,成功:

    @Test
    public void saveTest() {
        EmployeeSalary employeeSalary = new EmployeeSalary();
        employeeSalary.setSalary(9000d);
        employeeSalary.setEmployee(Employee.builder().name("mark").build());

        employeeSalaryRepository.save(employeeSalary);
    }

【测试e】在主entity中删除数据,同时会删除关联的子entity数据。

【测试e】在子entity中删除数据,同时会删除关联的主entity数据。

2.2 使用@OneToOne@PrimaryKeyJoinColumn来实现双向的共同主键

数据模型和#2.1一样。

共同主键表示两个实体共享主键,在子entity中,主键同时又是外键:

  • 因为是单向的关联,主entity不需要配置。
  • 子entity需要配置@OneToOne@PrimaryKeyJoinColumn,在@PrimaryKeyJoinColumn中需要指定本表的列(employee_id),以及要关联到另一张表的列名(id)。
@Entity
@Table(name = "employee")
public class Employee {

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

    private String name;
}
@Entity
@Table(name = "employee_salary")
public class EmployeeSalary {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private int employeeId;

    private double salary;

    @OneToOne(cascade = CascadeType.ALL, optional = false)
    @PrimaryKeyJoinColumn(name = "employee_id", referencedColumnName = "id")
    private Employee employee;
}

【测试a】由于是单向关联,主entity(employee)中findById只会返回自己的数据,sql:

select
employee0_.id as id1_1_0_,
employee0_.name as name2_1_0_
from
relation_study_employee employee0_
where
employee0_.id=?

【测试b】子entity(employee_salary)中findById,和#2.1一样,会返回两张表数据。唯一不同的是,这次只会查询一遍,即使用left outer join返回所有数据,并不会查两遍,具体sql:

select
employeesa0_.employee_id as employee1_2_0_,
employeesa0_.salary as salary2_2_0_
from
relation_study_employee_salary employeesa0_
where
employeesa0_.employee_id=?
Hibernate:
select
employee0_.id as id1_1_0_,
employee0_.name as name2_1_0_

from
    relation_study_employee employee0_ 
where
    employee0_.id=?

【测试c】在主entity中尝试插入数据,由于是单向关联,只会插入本表数据。

【测试d】在子entity中尝试插入数据,需要分别调用各自的repository才能插入两张表的数据:

    @Test
    public void saveTest() {
        Employee employee = Employee.builder().name("mark").build();
        employeeRepository.save(employee);

        EmployeeSalary employeeSalary = new EmployeeSalary();
        employeeSalary.setSalary(9000d);
        employeeSalary.setEmployee(employee);
        employeeSalary.setEmployeeId(employee.getId());

        employeeSalaryRepository.save(employeeSalary);
    }

【测试e】在主entity中尝试删除数据,只能删除自己表的数据。

【测试f】在子entity中尝试删除数据,能同时删除两张表数据。

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,631评论 18 399
  • 一. Java基础部分.................................................
    wy_sure阅读 3,811评论 0 11
  • 总结精炼于http://www.yiibai.com/jpa/jpa_architecture.html其他不错的...
    静心安分读书阅读 860评论 0 0
  • Hibernate 序 创建 hibernate 工程示例: 创建工程,引入 jar 包 创建配置文件 hiber...
    WJunF阅读 968评论 0 3
  • 一、重点知识 git 监视的是文件内容的修改 $ git checkout -- abc.txt : 其实是用版本...
    一花一世界yu阅读 891评论 0 2