zl程序教程

您现在的位置是:首页 >  后端

当前栏目

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

映射 12 注解 关联 外键 一对一
2023-09-14 08:57:15 时间
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方法 }

下面是我们的文章内容类


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的,回想我们之前的一对多关联测试,都是先删除维护关系方的,这其实很好理解,我们肯定要清除掉相应的关联关系(体现在数据库的外键上)才能完成被关联内容的删除操作


@OneToOne(cascade = CascadeType.ALL,mapperBy = "articleContent")

private Article article;

//忽略getter/setter

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


ArticleContent articleContent = new ArticleContent(); articleContent.setContent("content"); articleContent.setArticle(article); session.save(articleContent);

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


//这次删除是有级联关系的

ArticleContent articleContent = (ArticleContent) session.get(ArticleContent.class, 1);//注意这里id为1

session.delete(articleContent);

会看到我们相应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)


两个表中都建立了关于对方的关联映射。这是完全没有必要的,而且这样会造成的更严重后果,我们来测试级联添加
先调用如下测试代码:


ArticleContent articleContent = new ArticleContent(); articleContent.setContent("content"); article.setArticleContent(articleContent); session.save(article);

再调用如下测试代码:


ArticleContent articleContent = new ArticleContent(); articleContent.setContent("content"); articleContent.setArticle(article); session.save(articleContent);

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


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

session.delete(article);

会看到如下结果:


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


Mybatis当中的多对多映射关系(非注解版) 学习mybatis时,不用和spring集成, 不用注解,可以了解得多一些, 在这之上,使用注解和spring集成, 乃至到spring boot, 其间的传递路径才不会迷路。
Hibernate之关联关系映射(一对一主键映射和一对一外键映射) 1:Hibernate的关联关系映射的一对一外键映射: 1.1:第一首先引包,省略 1.2:第二创建实体类: 这里使用用户信息和身份证信息的关系,用户的主键编号既可以做身份证信息的主键又可以做身份证信息的外键,这里先做外键。