hibernate5(12)注解映射[4]一对一外键关联

在实际博客站点中,文章内容的数据量非常多,它会影响我们检索文章其他数据的时间,如查询公布时间、标题、类别的等。

这个时候,我们能够尝试将文章内容存在还有一张表中,然后建立起文章——文章内容的一对一映射

一对一关联有两种方式,一种是外键关联。还有一种是复合主键关联。

外键关联

以下我们先看一个一对一单向关联的实例

/*************关联关系维护方************/
@Table(name = "t_article")
@Entity
public class Article {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Integer id;
    private String title;
    @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY,orphanRemoval = true,targetEntity = ArticleContent.class)
    @JoinColumn(name = "article_content_id")
    private ArticleContent articleContent;
    //忽略get和set方法
}

以下是我们的文章内容类

@Table(name = "t_article_content")
@Entity
public class ArticleContent {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Integer id;
    @Lob
    private String content;

    //忽略get和set方法
}

以下是我们的測试类

public class Test3 {
    private ApplicationContext ac;
    private SessionFactory sessionFactory;
    private Session session;
    private Transaction transaction;
    @BeforeClass//在測试类初始化时调用此方法,完毕静态对象的初始化
    public static void before(){
    }
    @Before//每个被注解Test方法在调用前都会调用此方法一次
    public void setup(){//建立针对我们当前測试方法的的会话和事务
        ac = new ClassPathXmlApplicationContext("spring-datasource.xml");
        sessionFactory = (SessionFactory) ac.getBean("sessionFactory");
        session = sessionFactory.openSession();
        transaction = session.beginTransaction();
    }
    //測试级联关系映射注解配置:一对一单向关联
    @Test
    public void test1(){
        //測试级联加入
        Article article = new Article();
        article.setTitle("title");
        ArticleContent articleContent = new ArticleContent();
        articleContent.setContent("content");
        article.setArticleContent(articleContent);//建立映射关系
        session.save(articleContent);
        session.save(article);
        //測试级联删除
//      Article article = (Article) session.get(Article.class,1);
//      session.delete(article);
    @After//每个被注解Test方法在调用后都会调用此方法一次
    public void teardown(){
        if(transaction.isActive()){//假设当前事务尚未提交,则
            transaction.commit();//提交事务,主要为了防止在測试中已提交事务,这里又反复提交
        }
        session.close();
}

调用我们的測试方法test1。

控制台打印:

Hibernate: insert into t_article_content (content) values (?)

Hibernate: insert into t_article (article_content_id, title) values (?, ?)

此时查看数据库:

mysql> show tables; ————————————hibernate帮我们新建的表格

+———————+

| Tables_in_hibernate |

+———————+

| t_article |

| t_article_content |

+———————+

2 rows in set (0.00 sec)

mysql> desc t_article; ————————————单方维护映射关系,通过article_content_id维护

+——————–+————–+——+—–+———+—————-+

| Field | Type | Null | Key | Default | Extra |

+——————–+————–+——+—–+———+—————-+

| id | int(11) | NO | PRI | NULL | auto_increment |

| title | varchar(255) | YES | | NULL | |

| article_content_id | int(11) | YES | MUL | NULL | |

+——————–+————–+——+—–+———+—————-+

3 rows in set (0.00 sec)

mysql> desc t_article_content;

+———+———-+——+—–+———+—————-+

| Field | Type | Null | Key | Default | Extra |

+———+———-+——+—–+———+—————-+

| id | int(11) | NO | PRI | NULL | auto_increment |

| content | longtext | YES | | NULL | |

+———+———-+——+—–+———+—————-+

2 rows in set (0.00 sec)

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 1 | title | 1 |

+—-+——-+——————–+

1 row in set (0.00 sec)

mysql> select * from t_article_content;

+—-+———+

| id | content |

+—-+———+

| 1 | content |

+—-+———+

1 row in set (0.00 sec)

凝视掉測试代码的级联加入部分,执行级联删除部分:

Hibernate: delete from t_article where id=?

Hibernate: delete from t_article_content where id=?

在这里,我们观察到它是先删除文章(维护关系方)。再删除t_article_content的,回忆我们之前的一对多关联測试。都是先删除维护关系方的。这事实上非常好理解,我们肯定要清除掉相应的关联关系(体如今数据库的外键上)才干完毕被关联内容的删除操作

一对一双向关联非常easy,直接在articleContent上加入:

@OneToOne(cascade = CascadeType.ALL,mapperBy = "articleContent")
private Article article;
//忽略getter/setter

使用和上面一样的測试代码。hibernate会帮我们生成表格并插入数据:

mysql> select * from t_article_content;

+—-+———+

| id | content |

+—-+———+

| 1 | content |

+—-+———+

1 row in set (0.00 sec)

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 1 | title | 1 |

+—-+——-+——————–+

1 row in set (0.00 sec)

这时候假设我们尝试在放弃维护的articleContent端进行级联加入:

//測试articleContent级联加入
Article article = new Article();
article.setTitle("title");
ArticleContent articleContent = new ArticleContent();
articleContent.setContent("content");
articleContent.setArticle(article);
session.save(articleContent);

我们的article对象能被成功保存。可是。两者的关联关系建立失败:

mysql> select * from t_article_content;

+—-+———+

| id | content |

+—-+———+

| 1 | content |

| 2 | content |

+—-+———+

2 rows in set (0.00 sec)

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 1 | title | 1 |

| 2 | title | NULL |

+—-+——-+——————–+

2 rows in set (0.00 sec)

这时候我们再尝试从放弃维护端删除:

//这次删除是有级联关系的
ArticleContent articleContent = (ArticleContent) session.get(ArticleContent.class, 1);//注意这里id为1
session.delete(articleContent);

mysql> select * from t_article_content;

+—-+———+

| id | content |

+—-+———+

| 5 | content |

+—-+———+

1 row in set (0.00 sec)

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 6 | title | NULL |

+—-+——-+——————–+

1 row in set (0.00 sec)

会看到我们相应article对象也被删除了!因此,我们须要明白放弃维护关联关系并不代表放弃关联关系,从ArticleContent端,我们一样能进行与关联关系双管的级联加入、删除操作。仅仅是不正确两者关系进行维护。因而在加入时Article端的外键属性article_content_id=null

我们使用mappedBy属性放弃关联。但级联操作依旧有效,因此须要区分开维护关联关系级联操作的差别。

这里须要特别注意的是。在这样的一对一映射中。我们最好选择一个被动方并设定mapperBy属性。即让一方放弃维护关联关系,否则,我们会看到下述现象:

mysql> desc t_article;

+——————–+————–+——+—–+———+—————-+

| Field | Type | Null | Key | Default | Extra |

+——————–+————–+——+—–+———+—————-+

| id | int(11) | NO | PRI | NULL | auto_increment |

| title | varchar(255) | YES | | NULL | |

| article_content_id | int(11) | YES | MUL | NULL | |

+——————–+————–+——+—–+———+—————-+

3 rows in set (0.00 sec)

mysql> desc t_article_content;

+————+———-+——+—–+———+—————-+

| Field | Type | Null | Key | Default | Extra |

+————+———-+——+—–+———+—————-+

| id | int(11) | NO | PRI | NULL | auto_increment |

| content | longtext | YES | | NULL | |

| article_id | int(11) | YES | MUL | NULL | |

+————+———-+——+—–+———+—————-+

3 rows in set (0.00 sec)

两个表中都建立了关于对方的关联映射。

这是全然没有必要的,并且这样会造成的更严重后果,我们来測试级联加入

先调用例如以下測试代码:

//測试article级联加入
Article article = new Article();
article.setTitle("title");
ArticleContent articleContent = new ArticleContent();
articleContent.setContent("content");
article.setArticleContent(articleContent);
session.save(article);

再调用例如以下測试代码:

//測试articleContent级联加入
Article article = new Article();
article.setTitle("title");
ArticleContent articleContent = new ArticleContent();
articleContent.setContent("content");
articleContent.setArticle(article);
session.save(articleContent);

我们会看到数据库相应记录:

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 1 | title | 1 |

| 2 | title | NULL |

+—-+——-+——————–+

2 rows in set (0.00 sec)

mysql> select * from t_article_content;

+—-+———+————+

| id | content | article_id |

+—-+———+————+

| 1 | content | NULL |

| 2 | content | 2 |

+—-+———+————+

2 rows in set (0.00 sec)

即两方各维护各的关联关系。假设这时候我们尝试交换測试级联删除:

Article article = (Article) session.get(Article.class,2);
session.delete(article);

会看到例如以下结果:

mysql> select * from t_article;

+—-+——-+——————–+

| id | title | article_content_id |

+—-+——-+——————–+

| 1 | title | 1 |

+—-+——-+——————–+

1 row in set (0.00 sec)

mysql> select * from t_article_content;

+—-+———+————+

| id | content | article_id |

+—-+———+————+

| 1 | content | NULL |

| 2 | content | 2 |

+—-+———+————+

2 rows in set (0.00 sec)

即级联删除失败了,而这是显然的。由于id为2的文章,相应article_content_id属性为null,在文章方看来,两者都没建立关联关系。这样的时候肯定不是报错就是级联删除失败,而报错是由于假设设置了数据库在t_article_content中设置了对article_id的的外键关联,由于存在记录article_id=2,这时候我们尝试删除article表中id为2的记录,则会由于外键关系约束失败而报错

时间: 2024-08-04 03:01:40

hibernate5(12)注解映射[4]一对一外键关联的相关文章

Hibernate5.2之一对一外键关联(五)

                                                 Hibernate5.2之一对一外键关联(五) 一.简介 上篇文章中笔者介绍了Hibernate关联关系中的一对一外键关联,本篇博客将介绍一对一外键关联.其实我们回过头想一想,外键关联其实就是一对多关联关系中将多的一方简化为一个,就是我们本文所要介绍的一对一的外键关联. 二.外键关联 2.1数据库表的创建 create table people ( id varchar2(255 char) not

Hibernate之关联关系映射(一对一主键映射和一对一外键映射)

1:Hibernate的关联关系映射的一对一外键映射: 1.1:第一首先引包,省略 1.2:第二创建实体类: 这里使用用户信息和身份证信息的关系,用户的主键编号既可以做身份证信息的主键又可以做身份证信息的外键,这里先做外键. 创建User.java: 用户和身份证一对一的关联关系映射       private IdCart idCart; IdCart.java: 身份证和用户,一对一的关系       private User user; 1 package com.bie.bean; 2

Hibernate,关系映射的多对一单向关联、多对一双向关联、一对一主键关联、一对一外键关联、多对多关系关联

2018-11-10  22:27:02开始写 下图内容ORM.Hibernate介绍.hibername.cfg.xml结构: 下图内容hibernate映射文件结构介绍 下图内容hibernate映射文件中主键自增规则.Hibernate实例状态(瞬时状态.持久化状态.托管状态).Hibernate初始化类获取session等方法 下图内容保存数据过程 下面内容保存数据顺序.查询数据方法 get().load()和延迟加载.删除数据 下图内容删除对象顺序.修改数据顺序 下面内容关联关系映射.

hibernate5(13)注解映射[5]一对一共享主键关联

一对一共享主键 下面我们直接通过实例来讲解共享主键配置: 主键主控方:Article package com.zeng2.model; @Table(name = "t_article2") @Entity public class Article { @Id @GeneratedValue(strategy = GenerationType.AUTO) private Integer id; private String title; @OneToOne(cascade = Casc

Hibrenate一对一外键关联

一.一对一(单向):使用外部索引将其中的一个类作为parent,相对应的一个就是子类,并且参照父 类的主键ID来生成数据库表.(比如:可以将husband中设置一个wife_id对应wife中的主键id) 1.Wife 类:生成get.set方法 @Entity//注意使用注解 public class Wife { private int id; private String name; @Id//注意使用注解 @GeneratedValue//注意使用注解 public int getId(

hibernate5(10)注解映射[2]一对多单向关联

在上一篇文章里.我们从端方向一端建立关联关系,完毕了从文章到作者的关联关系建立.但在实际的博客站点中,用户肯定还须要获取自己所写的文章,这时能够建立用户(一)对文章(多)的单向关联映射. 先来看我们的一方配置实例 package com.zeng.model; import java.util.Set; import javax.persistence.CascadeType; import javax.persistence.Entity; import javax.persistence.F

hibernate5(9)注解映射[1]一对多单向关联

一对多单向关联映射 在实际的博客网站中,用户肯定需要获取自己所写的文章,这时可以建立用户(一)对文章(多)的单向关联映射. 先来看我们的一方配置实例 package com.zeng.model; import java.util.Set; import javax.persistence.CascadeType; import javax.persistence.Entity; import javax.persistence.FetchType; import javax.persisten

Hibernate注解:一对多外键关联

情形:两个表,cms_mode是主表,cms_model_field是子表,cms_model_field的model_id字段关联到cms_model的主键. # # Source for table cms_model # DROP TABLE IF EXISTS `cms_model`; CREATE TABLE `cms_model` ( `model_id` int(11) NOT NULL, `name` varchar(50) NOT NULL DEFAULT '', `title

数据库一对一的两种关联 主键关联和外键关联

关联映射:一对一 一对一关系就如球队与球队所在地址之间的关系,一支球队仅有一个地址,而一个地址区也仅有一支球队. 数据表间一对一关系的表现有两种,一种是外键关联,一种是主键关联.图示如下: 一对一外键关联: 一对一主键关联:要求两个表的主键必须完全一致,通过两个表的主键建立关联关系 出自:http://hi.baidu.com/websatyr/item/10e3d83766ffb3f42784f481