hibernate 数据关联一对多
一对多,多对一 (在多的一端存放一的外键)
但是在实体类中不需要创建这个外键
// 在一的一方创建Set集合
public class User { private Integer id;
private String username;
private String password;
private Set<Address> addressSet;
}
//多的一方创建User的对象
public class Address { private Integer id;
private String address;
private User user;
}
在映射文件中也要配置
一对多的一:User.hbm.xml中
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.kaishengit.pojo"> <class name="User" table="user">
<id name="id">
<generator class="native"/>
</id> <property name="username"/>
<property name="password"/>
<!-- name是类中属性的名字,站在User的角度这是一对多,所以是one-to-many
然后指定对应的多指的是哪个类.
<key column="userid"/>指维持这种关系的多的这一端(Address)类对应的表中的外键
inverse="true"表示放弃关系维护-->
<set name="addressSet" cascade="delete" inverse="true">
<key column="userid"/>
<one-to-many class="Address"/>
</set> </class>
</hibernate-mapping>
一对多的多:Address.hbm.xml中
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.kaishengit.pojo"> <class name="Address" table="address">
<id name="id">
<generator class="native"/>
</id>
<property name="address"/>
<!-- 站在这个角度是多对一many-to-one
name写对应的属性名,class是对应的类 然后指定外键-->
<many-to-one name="user" class="User" column="userid"/> </class> </hibernate-mapping>
--------------------------------------------------------------------------
--------------------------------------------------------------------------
程序的执行
1.添加
// 方式1
User u1 = new User();
u1.setUsername("u1");
u1.setPassword("aaa"); Address a1 = new Address();
a1.setAddress("a1");
a1.setUser(u1); Address a2 = new Address();
a2.setAddress("a2");
a2.setUser(u1); /* 如果是先save的u1,就能拿到u1的id
然后再save a1,a2的时候加入外键中,等于执行了3条sql 如果是先save a1,a2 那么save的结果是没有userid这个外键的
在save了u1之后会再去update他们(因为在address关联User的时候,user是自由态
,当user保存的时候就变成了持久态,User状态发生变化会同步到数据库中,所以address会更新
),等于执行了3条sql还有两条update*/
session.save(u1);
session.save(a1);
session.save(a2); // 所以对于一对多的这种情况,先存1,再存多 // 方式2 User u1 = new User();
u1.setUsername("u1");
u1.setPassword("aaa"); Address a1 = new Address();
a1.setAddress("a1"); Address a2 = new Address();
a2.setAddress("a2"); Set<Address> set = new HashSet<Address>();
set.add(a1);
set.add(a2);
user.setAddressSet(set); // 这样即使是先存1再存多还是5条sql语句 session.save(u1);
session.save(a1);
session.save(a2);
所以在一对多的时候,关系要让多来维护,save的时候先保存一的再保存多的
可以在一的配置中inverse="true"放弃关系维护
2.查询
// 单表查询user
User user = (User) session.get(User.class, 25);
System.out.println(user.getUsername());
/* 要使用address了,单表查询address,这个是延迟查询,当使用address的时候再去查
,但是这个查询就必须放在session中才能有作用,给xxx.hbm.xml配置
lazy="false" 就能取消这个延迟查询,不管你用不用address,但是只要你用user我都会查关联user
的一切比如(address,这个时候就可以放在session外边了因为查询过后已经放入了内存中*/
Set<Address> set = user.getAddressSet();
for(Address a : set) {
System.out.println(a.getAddress());
} // 或者反过来用address查询,也都是两条单表查询
Address a1 = (Address) session.get(Address.class, 17);
System.out.println(a1.getAddress());
System.out.println(a1.getUser().getUsername());
hibernate默认就是这样的单表查询,但是我们可以改成联接查询 -->
<!-- 添加fetch="join"表示用一条sql语句查询到所有包括关联的..所以只要不是
你能承受这么多查询,都要使用以前的延迟查询
<set name="addressSet" cascade="delete" inverse="true" fetch="join">
<key column="userid"/>
<one-to-many class="Address"/>
</set>
3.级联删除
配置属性cascade="delete"属性后,默认删除一的时候把关联的多也删除掉
<set name="addressSet" cascade="delete" inverse="true">
<key column="userid"/>
<one-to-many class="Address"/>
</set>
也删除了address
User user = (User) session.get(User.class, 26);
session.delete(user);
注:
我们在user中配置了address的Set,在address中配置了user,这叫
双向配置,实际上我们都是从user找address,很少从address找user,所以address的可以不配置
hibernate 数据关联一对多的更多相关文章
- hibernate 数据关联一对多 3.1
一对多,多对一 (在多的一端存放一的外键) 但是在实体类中不需要创建这个外键 // 在一的一方创建Set集合 public class User { private Integer id; priva ...
- hibernate 数据关联多对多 4.1
多对多,必须有一张关系表来维持关系 数据库student,teacher student_teacher 三张表 但是在pojo中只需要建立student和teacher两个类,除非关系表也代表某种业 ...
- hibernate 数据关联一对一 3.2
第一种一对一 person和card,card的id即作为主键,又作为外键 // 各村对方的一个对象 public class Person { private Integer id; privat ...
- hibernate 数据关联多对多
多对多,必须有一张关系表来维持关系 数据库student,teacher student_teacher 三张表 但是在pojo中只需要建立student和teacher两个类,除非关系表也代表某种业 ...
- hibernate 数据关联一对一
第一种一对一 person和card,card的id即作为主键,又作为外键 // 各村对方的一个对象 public class Person { private Integer id; privat ...
- Hibernate之关联映射(一对多和多对一映射,多对多映射)
~~~接着之前的Hibernate框架接着学习(上篇面试过后发现真的需要学习以下框架了,不然又被忽悠让去培训.)~~~ 1:Hibernate的关联映射,存在一对多和多对一映射,多对多映射: 1.1: ...
- 初识Hibernate之关联映射(二)
上篇我们介绍了关联映射的几种形式,有单向多对一,单向一对多,还有双向一对多.本篇接着介绍有关关联映射的其他几种映射方式,主要有以下几种: 基于外键的单向一对一关联映射 基于主键的单向一对一关联映射 单 ...
- Hibernate 关联关系(一对多)
Hibernate 关联关系(一对多) 1. 什么是关联(association) 1.1 关联指的是类之间的引用关系.如果类A与类B关联,那么被引用的类B将被定义为类A的属性.例如: class B ...
- Hibernate之关联关系(一对多)
今日分享hibernate框架的简单关联关系 一:关联关系简介 1.1 什么是关联关系 关联指的是类之间的引用关系.如果类A与类B关联,那么被引用的类B将被定义为类A的属性. 例如: class B{ ...
随机推荐
- sap ftp 处理
[转] SAP FTP Function 本文示例如何使用SAP FTP Function将文件从应用服务器传输到另外一个FTP服务器上. DATA: BEGIN OF ig_ftp_result O ...
- crontab定时任务写法记录
基本格式 : * * * * * command 分 时 日 月 周 命令 第1列表示分钟1-59 每分钟用*或者 */1表示 第2列表示小时1-23(0表示0点) 第3列表示日期1-31 第4列表示 ...
- linux里的drwxr-xr-x代表的意思
权限的计算是除去第一位字母开始,权限都是三个符号为一组合,其中-表示没有这个权限 d:第一位表示文件类型.d是目录文件,l是链接文件,-是普通文件,p是管道 rwx:第2-4位表示这个文件的属主拥有的 ...
- Oracle数据库体系结构(4)oracle控制文件
控制文件的概述 1.控制文件是oracle数据库非常重要的物理文件,描述了整个数据库的物理结构信息,包括数据库名称.数据文件与重做日志文件的名称与位置,日志序列号等信息.数据库实例根据初始化参数CON ...
- zabbix实现mysql数据库的监控(三)
上面一章“zabbix实现mysql数据库的监控(二)”使用MPM来监控mysql,但是遇到安装问题始终解决不了,这里改用percona-monitoring-plugins进行zabbxi上监控my ...
- poj 3278 Catch That Cow-搜索进阶-暑假集训
Catch That Cow Time Limit:2000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Subm ...
- Thrift文件加载
一.简述 通过前面的分析,我们知道无论是创建一个客户端还是服务器,第一步要做的就是调用thriftpy.load对thrift文件进行解析,并在内存中构建相应的module,本文将对load方法进行一 ...
- Linux课程---5、常用文件命令和目录命令(创建文件命令)
Linux课程---5.常用文件命令和目录命令(创建文件命令) 一.总结 一句话总结: touch file1 1.管道符|有什么用? 将前一个命令的结果作为后一个命令的输入:比如查看文件前3行:ca ...
- 图像处理检测方法 — SIFT和SURF
0.特征与匹配方法总结汇总对比 参考网址:http://simtalk.cn/2017/08/18/%E7%89%B9%E5%BE%81%E4%B8%8E%E5%8C%B9%E9%85%8D/#ORB ...
- define的用法与注意事项
------------------------------------------------- 在编程使用宏替换时,当字符串中不只一个符号时,加上括号表现出优先级, 如果是带参数的宏定义,则要给宏 ...