Hibernate入门之one to one关系映射详解

前言

本节开始我们正式步入关系映射的讲解,对于关系映射,在.NET或NET Core对于相关实体大多我们称之为主体和依赖体,而在java中称之为源实体和目标实体,只是叫法不同,本质上大同小异,文中若有错误之处,还望指正。

one to one关系映射

一对一关联可以是单向或双向的,在单向关联中,源实体具有一个关系字段,该关系字段引用目标实体,并且源实体的表包含外键,在双向关联中,每个实体(即源实体和目标实体)都有一个相互引用的关系字段,并且目标实体的表包含外键,源实体必须使用mappedBy属性来定义双向一对一映射,接下来我们分别来看单向关联和双向关联。

单向关联

这里我们分别给出所用到的两个对象Post和PostDetail,一篇博客发表只对应此篇博客详情,如下:

复制代码
@Entity
public class Post {

    @Id
    @GeneratedValue
    private Long id;

    private String title;

    @OneToOne
    @JoinColumn(name = "post_detail_id")
    private PostDetail postDetail;

}
复制代码
复制代码
@Entity
public class PostDetail {

    @Id
    @GeneratedValue
    private Long id;

    @Column(name = "created_on")
    private Date createdOn;

    @Column(name = "created_by")
    private String createdBy;

    public PostDetail() {
    }

    public PostDetail(String createdBy) {
        createdOn = new Date();
        this.createdBy = createdBy;
    }
}
复制代码

如上单向关联中,通过源实体(Post)中的关系字段post_detail_id来引用目标实体,且此字段作目标实体(PostDetail)的外键,很明显像这种情况不符合实际情况,因为postdetail依赖于post。

双向关联

即便是一对一关系也会存在所属关系即目标实体依赖于源实体,所以上述post和postdetail关系是双向的,如果关系是双向的,则非拥有方(目标实体)必须使用属性mappedBy来指定拥有方的关系字段或属性。此时我们将post实体和postdetail实体修改成如下,而postdetail添加源实体。

复制代码
@Entity
public class Post {

    @OneToOne(mappedBy = "post")
    private PostDetail postDetail;

}

@Entity
public class PostDetail {

    @OneToOne
    private Post post;
}
复制代码

 

 

上述我们在Post通过mappedBy添加了对目标实体PostDetail的所属关系,同时在PostDetail添加了对Post的关联,此时我们可以看到默认生成的外键为post_id即(源实体名称 + '_' + id),若需要修改外键名称,直接通过注解@JoinColumn显式指定外键名称,比如如下指定为custom_post_id。

分析到这里为止,其实对于一对一关系映射配置还是非常简单,我们只需要根据业务搞清楚所属关系,然后通过mappedBy指定,最终可以默认或显式指定外键名称,有的童鞋可能就问了,要是我想将实体PostDetail中的外键post_id也作为主键呢,也就是说将post_id作为共享主键,此时我们应该怎么办呢?此时实体PostDetail中的主键id将不再是自动生成,去除自动生成注解,然后通过注解@MapsId来指定源实体,改造后的实体PostDetail如下

复制代码
@Entity
public class PostDetail {

    @Id
    private Long id;

    @OneToOne
    @MapsId
    private Post post;
}
复制代码

虽然我们达到了共享主键的目的,但是在PostDetail表中其主键名称为post_id,如果感觉不太友好或者不太好理解,那就继续通过注解@JoinColumn指定名称为id,如下:

复制代码
@Entity
public class PostDetail {

    @Id
    private Long id;

    @OneToOne
    @MapsId
    @JoinColumn(name = "id")
    private Post post;
}
复制代码

总结

本节我们详细讲解了一对一关系映射,无论是单向关联还是双向关联根据实际业务而定,大部分情况下需要通过注解@mappedBy来进行双向关联,同时呢,我们也可以通过注解@MapsId来配置共享主键。好了,本节到此结束,下一节我们进入一对多关系映射的讲解。


为了方便大家在移动端也能看到我分享的博文,现已注册个人公众号,扫描上方左边二维码即可,欢迎大家关注,有时间会及时分享相关技术博文。

感谢花时间阅读此篇文章,如果您觉得这篇文章你学到了东西也是为了犒劳下博主的码字不易不妨打赏一下吧,让楼主能喝上一杯咖啡,在此谢过了!
如果您觉得阅读本文对您有帮助,请点一下“推荐”按钮,您的“推荐”将是我最大的写作动力!
本文版权归作者和博客园共有,来源网址:http://www.cnblogs.com/CreateMyself)/欢迎各位转载,但是未经作者本人同意,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责任的权利。
posted @   Jeffcky  阅读(1247)  评论(0编辑  收藏  举报
编辑推荐:
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
点击右上角即可分享
微信分享提示