Hibernate关联关系的CRUD
本文以Group和User(一对多、多对一)双向关联为例,介绍关联关系的CRUD
下面先介绍两个属性
cascade:只影响CRUD中的CUD,即存储(save)、更新(update)、删除(delete) fetch:只影响CRUD中的R,即读取(get、load)
cascade(级联): 此属性仅仅帮助我们简化编程,不是必选项 如果A和B为单向关联,则在主导方设置cascade属性 如果A和B为双向关联,则在双方都要设置cascade属性
如果两个对象之间有关联关系,比如User和Group多对一关联 如果想要保存User的时候自动保存Group,可在User类的@ManyToOne注解中设置cascade(级联)属性 反过来,如果想要保存Group的时候自动保存User,可在Group类的@OneToMany注解中设置cascade(级联)属性
对于User(多方)和Group(一方) 执行如下语句 User u = (User) session.get(User.class, 1); 在取出多方对象u的同时也会把一方对象对应属性取出来,这是@ManyToOne的默认设置.相当于设置@ManyToOne(fetch=FetchType.EAGER) 反过来,取一方对象的时候,则不会把多方对象取出 若想要在取出一方对象g的同时取出多方对象,则应在Group类的@OneToMany注解中设置fetch属性
@OneToMany注解中fetch的默认值为FetchType.LAZY, 相当于设置@OneToMany(fetch=FetchType.LAZY) fetch的取值有两个 fetch=FetchType.EAGER :执行语句就取出 fetch=FetchType.LAZY :用的时候才取出 两者的区别类似于get和load的区别 注意:对于双向关联,不要两边都设EAGER(会有多余的查询语句发出),可以都设LAZY 一般是多对一设置EAGER,一对多设置LAZY
建Group实体类和User实体类,以及Junit测试类
@Entity
@Table(name="_group")
public class Group {
private int id;
private String name;
private Set<User> users = new HashSet<User>();
@Id
@GeneratedValue
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@OneToMany(mappedBy="group")
public Set<User> getUsers() {
return users;
}
public void setUsers(Set<User> users) {
this.users = users;
}
}
@Entity
@Table(name="_user")
public class User {
private int id;
private String name;
private Group group;
@Id
@GeneratedValue
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@ManyToOne
public Group getGroup() {
return group;
}
public void setGroup(Group group) {
this.group = group;
}
}
public class ORMappingTest { private static SessionFactory sf = null; @BeforeClass
public static void beforeClass(){
sf = new Configuration().configure().buildSessionFactory();
} @AfterClass
public static void afterClass(){
sf.close();
} @Test
public void test() { //下面将会完善此test()方法 }
}
接下来开始CRUD的练习
首先在Group类和User类的@OneToMany和@ManyToOne注解中添加cascade属性
//Group类 此处fetch属性中,LAZY为默认值,故也可不写
@OneToMany(mappedBy="group",cascade={CascadeType.ALL},fetch=FetchType.LAZY)
public Set<User> getUsers() {
return users;
}
//User类 此处fetch属性中,EAGER为默认值,故也可不写
@ManyToOne(cascade={CascadeType.ALL},fetch=FetchType.EAGER)
public Group getGroup() {
return group;
}
然后,完善Junit测试类的test()方法
CRUD----C 增 session.save()
1.保存User对象的同时自动保存Group对象
@Test
public void test() {
Session session = sf.getCurrentSession();
session.beginTransaction(); Group g = new Group();
g.setName("lisi"); User u = new User();
u.setName("user1");
u.setGroup(g); session.save(u); //因为在User类的@ManyToOne注解中设置了cascade(级联)属性,故只需保存User对象即可,Group对象会自动保存 session.getTransaction().commit();
}
2.保存Group对象的同时自动保存User对象
@Test
public void test() {
Session session = sf.getCurrentSession();
session.beginTransaction(); User u1 = new User();
User u2 = new User(); Group g = new Group(); u1.setName("user1");
u1.setGroup(g);
u2.setName("user2");
u2.setGroup(g); g.setName("group1");
g.getUsers().add(u1);
g.getUsers().add(u2); session.save(g); //因为在Group类的@OneToMany注解中设置了cascade(级联)属性,故只需保存Group对象即可,User对象会自动保存 session.getTransaction().commit();
}
CRUD----R 查 session.get()和session.load()
1.取出User对象的同时取出Group对象
@Test
public void test() { Session session = sf.getCurrentSession();
session.beginTransaction(); User u = (User) session.get(User.class, 1); //在取出多方对象u的同时也会把一方对象对应属性取出来,这是@ManyToOne的默认设置
System.out.println(u.getGroup().getName()); //打印输出_group表中对应的name值 session.getTransaction().commit();
}
2.取出Group对象的同时取出User对象
@Test
public void test() { Session session = sf.getCurrentSession();
session.beginTransaction(); Group g = (Group) session.get(Group.class, 2); //在取出一方对象g的同时取出多方对象,此时应在Group类的@OneToMany注解中设置fetch属性
for(User u : g.getUsers()){
System.out.println(u.getId()+","+u.getName()); //打印输出_user表中的相关数据
} session.getTransaction().commit();
}
CRUD----U 改 session.update()
通过取出的User对象,既可以更改User对象中的属性名字,也可以更改与其级联的Group对象中的属性名字
@Test
public void test { Session session = sf.getCurrentSession();
session.beginTransaction(); User u = (User) session.get(User.class, 3); //在取出多方对象u的同时也会把一方对象对应属性取出来,这是@ManyToOne的默认设置
u.setName("name"); //更改User对象中的属性名字
u.getGroup().setName("Groupname"); //也可以更改与其级联的Group对象中的属性名字
session.update(u); session.getTransaction().commit();
}
CRUD----D 删 session.delete()
删除User表中的某一条记录
两种方式,如下:
@Test
public void testDelete() { Session session = sf.getCurrentSession();
session.beginTransaction(); //删除User表中id值为1的记录
User u = (User) session.get(User.class, 1); //在取出多方对象u的同时也会把一方对象对应属性取出来,这是@ManyToOne的默认设置
u.setGroup(null); //应该先消除关联关系,再删除对应记录
session.delete(u); //如果直接删除,由于设置了cascade属性,则会删除Group和User中相关的所有数据 //删除User表中id值为2的记录
session.createQuery("delete from User u where u.id=2").executeUpdate(); session.getTransaction().commit();
}
Hibernate关联关系的CRUD的更多相关文章
- hibernate关联关系的crud之级联
cascade级联,只会影响CRUD的CUD,不会影响读取.不设置级联,从多的一方能读出一的一方,设了级联,从一的一方,默认也不能读出多的一方. 如果两个对象之间有关联,不管是一对多,多对一,单向还是 ...
- hibernate关联关系的crud2
hibernate关联关系的CRUD操作,解释都在注释里了,讲了fetchType.cascade. User类: package com.oracle.hibernate; import javax ...
- hibernate关联关系笔记
Hibernate关联关系笔记 单向N:1 * 有连接表:在N方使用<join>/<many-to-one>.1方无需配置与之关联的持久化类. * 没有连接表:在N方使用& ...
- Hibernate关联关系映射
1. Hibernate关联关系映射 1.1. one to one <class name="Person"> <id name="id" ...
- Hibernate 关联关系(一对多)
Hibernate 关联关系(一对多) 1. 什么是关联(association) 1.1 关联指的是类之间的引用关系.如果类A与类B关联,那么被引用的类B将被定义为类A的属性.例如: class B ...
- hibernate(十)双向关联关系的CRUD
本文链接:http://www.orlion.ml/28/ 一.保存 1. 假设一个group有多个user,一个user只属于一个group,当保存user对象到数据库中时可以 User u = n ...
- 关联关系的CRUD
关联关系中的CRUD_Cascade_Fetch 1. hibernate_1700_one2many_many2one_bi_crud 2. 设定 cascade 可以设定在持久化时对于关联对象的操 ...
- Hibernate关联关系配置(一对多、一对一和多对多)
第一种关联关系:一对多(多对一) "一对多"是最普遍的映射关系,简单来讲就如消费者与订单的关系. 一对多:从消费者角的度来说一个消费者可以有多个订单,即为一对多. 多对一:从订单的 ...
- Hibernate关联关系之双向1—n
•双向 1-n 与双向 n-1 是完全相同的两种情形 •双向 1-n 需要在1的一端可以访问n的一端,反之依然. 测试实例代码: 实体类: package com.elgin.hibernate.nt ...
随机推荐
- 开源码应用之Eclipse篇
开写这篇的时候,恰逢Eclpse Mars(4.5)正式公布,最终由日蚀变登火星了,也离我開始基于Eclipse开发产品已经过去10年,这10年间,经历了Eclipse由私有核心框架到拥抱OSGi, ...
- Eclipse 创建 Java 包
打开新建 Java 包向导 你可以使用新建 Java 包向导来创建 Java 包.Java 包向导打开方式有: 通过点击 "File" 菜单并选择 New > Package ...
- 我的消灭复杂password之行
近期几天.网易一直提示邮箱账号异常.特意去查看了一下,发现须要改动password.可是经常使用的password又不让反复使用.于是无奈之下.就想办法消灭这些复杂password,由于实在是太难(g ...
- NetCore 中 EFcore的DbFirst和CodeFirst混合 使用注意
NetCore 最近很火热.笔者想把自己以前的旧项目迁移到NetCore平台. 先用EFcore的DBFirst根据数据库创建实体类,然后加入数据库版本控制功能也就是EFcore的CodeFirst部 ...
- scikit-learn:3.4. Model persistence
參考:http://scikit-learn.org/stable/modules/model_persistence.html 训练了模型之后,我们希望能够保存下来,遇到新样本时直接使用已经训练好的 ...
- lucene学习-创建索引
本文的lucene是基于lucene3.5版本. 使用lucene实现搜索引擎开发,核心的部分是建立索引和搜索.本节主要是记录创建索引部分的内容. 创建的索引结构如图所示. 创建索引的步骤分为以下几个 ...
- poj 2112(二分+多重匹配)
题目链接:http://poj.org/problem?id=2112 思路:由于要求奶牛走的最远距离的最短路程,显然我们可以二分距离,如果奶牛与挤奶器的距离小于等于limit的情况下,能够满足,则在 ...
- 读取文件之<绝对路径>与<相对路径>
前言:字符流.字节流读取文件,下面的代码是在网上找到的一个各种文件读取方式,还算比较详细,分享给大家. public class ReadFromFile { /** * 以字节为单位读取文件,常用于 ...
- node.js调用函数
首先EditPlus编辑器,打开新建的文本文档,另存为副本 调用函数分为调用本地函数,和其他文件的函数 1.调用本地函数 var http = require('http'); http.create ...
- 网络流——SAP模板
//网络流SAP模板,复杂度O(N^2*M) //使用前调用init(源点,汇点,图中点的个数),然后调用add_edge()加边 //调用getflow得出最大流 #define N 55 #def ...