hibernate5(11)注解映射[3]一对多多对一双向关联

简介: <div class="markdown_views"><p>在上两篇文章里,我们详细地分别讲解了一对多和多对一的单向关联配置的具体属性含义,在这一篇文章里,我们完成两者的的整合建立双向关联。 <br>在实际的博客网站中,我们可能需要根据文章读取作者(用户)信息,但肯定也要让用户能获取自己的文章信息,针对这种需求,我们可以建立文章(多)对用户(一)的双向关联映射。 <br>

在上两篇文章里,我们详细地分别讲解了一对多和多对一的单向关联配置的具体属性含义,在这一篇文章里,我们完成两者的的整合建立双向关联。
在实际的博客网站中,我们可能需要根据文章读取作者(用户)信息,但肯定也要让用户能获取自己的文章信息,针对这种需求,我们可以建立文章(多)对用户(一)的双向关联映射。
下面先看实例映射配置文件:

/********************一方配置User********************/
@Entity//声明当前类为hibernate映射到数据库中的实体类
@Table(name = "t_user1")//声明在数据库中自动生成的表名为t_user
public class User {
    @Id//声明此列为主键,作为映射对象的标识符
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Integer id;
    private String name;
    @OneToMany(cascade = CascadeType.ALL,fetch = FetchType.LAZY,targetEntity = Article.class,orphanRemoval = true,mappedBy = "user")//用户作为一方使用OneToMany注解
    @JoinColumn(name = "user_id")
//  @JoinTable(name = "t_user_articles",inverseJoinColumns = {@JoinColumn(name = "article_id")},joinColumns = {@JoinColumn(name = "user_id")})
    private Set<Article> articles;//文章作为多方,我们使用Set集合来存储,同时还能防止存放相同的文章

}
/*****************多方配置****************/
@Table(name = "t_article1")
@Entity
public class Article {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Integer id;
    private String content;
    /**
     * @ManyToOne 使用此标签建立多对一关联,此属性在“多”方使用注解在我们的“一”方属性上
     * @cascade 指定级联操作,以数组方式指定,如果只有一个,可以省略“{}”
     * @fetch 定义抓取策略
     * @optional 定义是否为必需属性,如果为必需(false),但在持久化时user = null,则会持久化失败
     * @targetEntity 目标关联对象,默认为被注解属性所在类
     */
    @ManyToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY)
    private User user;

映射关系确立好,开始编写我们的测试文件:


public class Test2 {
    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 test3(){
        User user = new User();
        user.setName("oneObject");
        Set<Article> articles = new HashSet<Article>();
        for(int i = 0 ; i < 3;i ++){
            Article article = new Article();
            article.setContent("moreContent" + i) ;
            articles.add(article);
        }
        user.setArticles(articles);//建立关联关系
        session.save(user);
    }

    @After//每一个被注解Test方法在调用后都会调用此方法一次
    public void teardown(){
        if(transaction.isActive()){//如果当前事务尚未提交,则
            transaction.commit();//提交事务,主要为了防止在测试中已提交事务,这里又重复提交
        }

        session.clear();
        session.close();
        sessionFactory.close();
    }
    @After//在类销毁时调用一次
    public void after(){
    }

}

这个时候,我们运行测试方法test3会发现报错:

org.hibernate.AnnotationException: Associations marked as mappedBy must not define database mappings like @JoinTable or @JoinColumn: com.zeng.model.User.articles
意思是,一旦被注解@mapperBy,即放弃了维护关联关系,而@JoinColumn注解的都是在“主控方”,因而我们需要注解在Article类中

@ManyToOne(cascade = CascadeType.ALL,fetch = FetchType.LAZY)
@JoinColumn(name = "user_id",unique = false,updatable = true)
private User user;

然后我们再运行测试方法:会看到:

Hibernate: drop table if exists t_article1
Hibernate: drop table if exists t_user1
Hibernate: create table t_article1 (id integer not null auto_increment, content varchar(255), user_id integer, primary key (id))
Hibernate: create table t_user1 (id integer not null auto_increment, name varchar(255), primary key (id))
Hibernate: alter table t_article1 add index FK6D6D45665B90FD3C (user_id), add constraint FK6D6D45665B90FD3C foreign key (user_id) references t_user1 (id)——————在这里,我们添加了外键约束,这也是hibernate对象关联在数据库的重要体现
上面是我们的表创建工作,下面是记录创建工作
Hibernate: insert into t_user1 (name) values (?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)

DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent0, id=3, user=null}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.User{id=1, articles=[com.zeng.model.Article#1, com.zeng.model.Article#2, com.zeng.model.Article#3], name=oneObject}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent2, id=1, user=null}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent1, id=2, user=null}

参考前面一篇文章的测试结果,在一对多单向配置中,因为关联关系是有一方维护,所以在最后总有三句额外的update语句,来完成article表到user表的映射关系,但在user放弃维护权后,如果我们再尝试通过保存用户通过建立起两表的映射关系,是不成功的。 从蓝色粗体部分,似乎User和article建立了关联关系。事实上,这是一种伪关联,它看似让我们通过session.save(user)。就完成了4者的关联创建,但在数据库层次,他们的关联关系是没有建立的,这从蓝色记录Article记录中user=null可以说明这一点。

此外,我们可以通过测试尝试从user中获取article对象来进一步验证:

User user = (User) session.get(User.class, 1);
System.out.println("获取用户对应的文章数据:"+user.getArticles());

打印结果:获取用户对应的文章数据:[]
这是因为我们的关联信息是由多方维护的(user_id),我们想要真正完成两者,必须从主维护方:article下手
运行以下测试代码:

User user = new User();
user.setName("oneObject1");
for(int i = 0 ; i < 3;i ++){
    Article article = new Article();
    article.setContent("moreContent1" + i) ;
    article.setUser(user);//有article来建立关联关系
    session.save(article);//持久化
}

得到打印结果:

Hibernate: insert into t_user1 (name) values (?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)
Hibernate: insert into t_article1 (content, user_id) values (?, ?)

DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent10, id=4, user=com.zeng.model.User#2}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent12, id=6, user=com.zeng.model.User#2}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.Article{content=moreContent11, id=5, user=com.zeng.model.User#2}
DEBUG: org.hibernate.internal.util.EntityPrinter - com.zeng.model.User{id=2, articles=null, name=oneObject1}

再查看数据库:
mysql> select * from t_article1;
+—-+—————+———+
| id | content | user_id |
+—-+—————+———+
| 1 | moreContent2 | NULL |——————上次操作遗留
| 2 | moreContent1 | NULL |——————上次操作遗留
| 3 | moreContent0 | NULL |——————上次操作遗留
| 4 | moreContent10 | 2 |
| 5 | moreContent11 | 2 |
| 6 | moreContent12 | 2 |
+—-+—————+———+
6 rows in set (0.00 sec)

从sql语句和蓝色DEBUG、数据库记录我们能够看出,这才是最优雅的添加关联操作,既没有多余的update语句,同时完成了数据库关联关系的建立。

目录
相关文章
|
8月前
|
Java 数据库连接
hibernate注解实体类(Dept.java)
hibernate注解实体类(Dept.java)
|
4月前
|
Java 数据库连接 API
解锁高效开发秘籍:深入探究 Hibernate 如何优雅处理一对多与多对多关系,让数据映射再无烦恼!
【9月更文挑战第3天】Hibernate 是 Java 领域中最流行的 ORM 框架之一,广泛用于处理实体对象与数据库表之间的映射。尤其在处理复杂关系如一对多和多对多时,Hibernate 提供了丰富的 API 和配置选项。本文通过具体代码示例,展示如何使用 `@OneToMany`、`@JoinColumn`、`@ManyToMany` 和 `@JoinTable` 等注解优雅地实现这些关系,帮助开发者保持代码简洁的同时确保数据一致性。
71 4
|
5月前
|
XML JSON Java
使用IDEA+Maven搭建整合一个Struts2+Spring4+Hibernate4项目,混合使用传统Xml与@注解,返回JSP视图或JSON数据,快来给你的SSH老项目翻新一下吧
本文介绍了如何使用IntelliJ IDEA和Maven搭建一个整合了Struts2、Spring4、Hibernate4的J2EE项目,并配置了项目目录结构、web.xml、welcome.jsp以及多个JSP页面,用于刷新和学习传统的SSH框架。
162 0
使用IDEA+Maven搭建整合一个Struts2+Spring4+Hibernate4项目,混合使用传统Xml与@注解,返回JSP视图或JSON数据,快来给你的SSH老项目翻新一下吧
|
5月前
|
Java 数据库连接 数据库
AI 时代风起云涌,Hibernate 实体映射引领数据库高效之路,最佳实践与陷阱全解析!
【8月更文挑战第31天】Hibernate 是一款强大的 Java 持久化框架,可将 Java 对象映射到关系数据库表中。本文通过代码示例详细介绍了 Hibernate 实体映射的最佳实践,包括合理使用关联映射(如 `@OneToMany` 和 `@ManyToOne`)以及正确处理继承关系(如单表继承)。此外,还探讨了常见陷阱,例如循环依赖可能导致的无限递归问题,并提供了使用 `@JsonIgnore` 等注解来避免此类问题的方法。通过遵循这些最佳实践,可以显著提升开发效率和数据库操作性能。
98 0
|
5月前
|
数据库 开发者 Java
Hibernate映射注解的魔力:实体类配置的革命,让你的代码量瞬间蒸发!
【8月更文挑战第31天】Hibernate 是一款出色的对象关系映射框架,简化了 Java 应用与数据库的交互。其映射注解让实体类配置变得直观简洁。本文深入剖析核心概念与使用技巧,通过示例展示如何简化配置。
60 0
|
7月前
|
JSON Java 数据库连接
Hibernate中使用@Lob 注解保存String[] 问题
Hibernate中使用@Lob 注解保存String[] 问题
47 2
|
8月前
|
Java 数据库连接
Hibernate中使用Criteria查询及注解——(Emp.hbm.xml)
Hibernate中使用Criteria查询及注解——(Emp.hbm.xml)
|
8月前
|
Java 数据库连接
hibernate注解实体类(Emp.java)
hibernate注解实体类(Emp.java)
|
8月前
|
Java 数据库连接
hibernate注解的测试
hibernate注解的测试
|
8月前
|
Java 数据库连接
Hibernate中使用Criteria查询及注解——( EmpCondition)
Hibernate中使用Criteria查询及注解——( EmpCondition)