hibernate关联对象的增删改查------增
本文可作为,北京尚学堂马士兵hibernate课程的学习笔记。
这一节,我们看看hibernate关联关系的增删改查
就关联关系而已,咱们在上一节已经提了很多了,一对多,多对一,单向,双向...
其实咱们可以简单的说就是A与B,有关系。
至于他们到底是一对多,多对一,暂且不论。
咱们要讨论的是,如果我存储A,那么数据库里是否会有B;如果我删除A,那么与之相关的B是否也会删除;如果我更新了A,那么B是否会被更新;如果我查询出A,那么B是否也会被查询出来。
首先,咱们看一对多,多对一双向的例子。
还是我们上一节的例子,dream与person。
一个person有多个dream,而每一个dream只能属于一个person。
我们在配置xml时,让hibernate自动生成建表语句,并且每次都新生数据库。
<property name="hbm2ddl.auto">create</property>
实体类情况
@Entity public class Person { private int id; private String name; private Set<Dream> dreams=new HashSet<>(); @Id @GeneratedValue public int getId() { return id; } @Column(name ="myname") public String getName() { return name; } @OneToMany(mappedBy="person") public Set<Dream> getDreams() { return dreams; }//省略部分代码 } @Entity public class Dream { private int id; private String description; private Person person; @Id @GeneratedValue public int getId() { return id; } @ManyToOne @JoinColumn(name="personId") public Person getPerson() { return person; } //省略部分代码 }
OK,两个类的关系已经很清楚了,双向多对一/一对多。
如果我们的测试代码如下:
//代码片1
public static void main(String[] args) { Person p=new Person(); p.setName("dlf"); Dream d=new Dream(); d.setDescription("marry glt"); d.setPerson(p); SessionFactory sessionFactory = new AnnotationConfiguration().configure().buildSessionFactory(); Session session = sessionFactory.getCurrentSession(); session.beginTransaction(); session.save(p); session.save(d); session.getTransaction().commit(); }
上面的代码是没有问题的,而且即使先save dream也OK。
那么现在问题来了
d.setPerson(p);
这句代码,已经说明了dream里有一个引用指向了person。
那么我可以在代码层次不save person么?
即只有一个
session.save(d);
答案是否定的,报下面的错误。
object references an unsaved transient instance - save the transient instance before flushing: com.bjsxt.hibernate.Person
因为在数据库保存dream的时候,person还没有保存,所以出错了。
上面的例子告诉我们,默认情况下:hibernate不会替我们存储"一"的那一方。
当我写到这块的时候,我忽然脑子一抽,又写下了下面的测试代码:
//代码片2 public static void main(String[] args) { Dream d=new Dream(); d.setDescription("marry glt"); Person p=new Person(); p.setName("dlf"); p.getDreams().add(d); SessionFactory sessionFactory = new AnnotationConfiguration().configure().buildSessionFactory(); Session session = sessionFactory.getCurrentSession(); session.beginTransaction(); session.save(p); session.getTransaction().commit(); }
如果,我以"一"的那一方为主,上面的代码会是个什么情况?
答案是,不报错,代码能正常执行,但是数据库里只有p的信息而没有d的信息。
上面的代码确实有点奇葩了,在上一节的时候,我们就已经达成了共识,在处理关联关系时,数据库层面总是在多的那一方维护关系。
所以上面的代码,并不能保存dream。
上面也说了,是在默认情况下,那特殊情况呢?
我们看api文档
上面是manytoone这个元标签可以带的属性,cascade翻译成中文就是级联。他是caschdetype型的
而Caschdetype是enum型的。可以取六个值
我们最经常用的就是,all,persist与remove。
all的意思是,对这个对象的增删改四中操作都会影响到与之关联的那个对象。(关于查询的级联,由另一个属性管理)
persist的意思是,对这个对象的增加(save操作)会连带把与之关联的那个对象也sava了。
remove的意思就是,如果删除这个对象也会删除与之关联的那个对象。
OK看下面的改动
@ManyToOne(cascade={CascadeType.ALL}) @JoinColumn(name="personId") public Person getPerson() { return person; }
此时,我们下面的代码,就能正常运行了,而且数据库里,两个对象都存储了,且外键关联也是OK的。
//代码片3 Person p=new Person(); p.setName("dlf"); Dream d=new Dream(); d.setDescription("marry glt"); d.setPerson(p); session.save(d);
写到这,我的脑子又抽了,如果我给onetomany的那一方加上cascade会如何?
@OneToMany(mappedBy="person",cascade={CascadeType.ALL}) public Set<Dream> getDreams() { return dreams; } 测试代码: Dream d=new Dream(); d.setDescription("marry glt"); Person p=new Person(); p.setName("dlf"); p.getDreams().add(d); session.save(p);
cascade确实有用,此时数据库里已经有了dream,但是dream的外键是null。
为什么?因为dream里并没有关联person。
其实,cascade这个东西也不是绝对的,如果关联关系比较麻烦,我们发现数据库里总是少存了一个对象,那就按照代码1的办法,直接存两次对象不就OK了。
谈点规律:
1 我们发现如果存储对象的时候,我们从多的一方操作是比较简单的。所以以后,尽量在多的一方操作。
2 为了符合逻辑,同时也不出错,一点两个对象之间的关系是双向的,那么在代码层次,就把两个的引用都设好,这样就不会出问题了。
3 之前说的,如果是双向的,设定mappedby。
hibernate关联对象的增删改查------增的更多相关文章
- 【基础篇】js对本地文件增删改查--增
前置条件: 1. 本地有安装node,点击传送门 项目目录: 1. msg.json内容 { "data": [ { "id": 1, "name&q ...
- Sqlserver 增删改查----增
注意我说的常见查询,可不是简单到一个表得增删改查,做过实际开发得人都知道,在实际开发中,真正牵扯到一个表得增删改查只能说占很小得一部分,大多都是好几个表的关联操作的. 下面我就说一下我在实际开发中经常 ...
- Ajax增删改查-----------增
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- php 数据库内容增删改查----增
首先,建立一个主页面(crud.php) <!DOCTYPE html> <html lang="en"> <head> <meta ch ...
- django基础之day04,必知必会13条,双下划线查询,字段增删改查,对象的跨表查询,双下划线的跨表查询
from django.test import TestCase # Create your tests here. import os import sys if __name__ == " ...
- MongoDB的ObjectId和基本操作增删改查(3)
ObjectId 基本操作增删改查 增: insert 介绍: mongodb存储的是文档,. 文档是json格式的对象. 语法: db.collectionName.insert(document) ...
- oracle初试、函数、增删改查、多表查询
安装oracle后的测试以及解锁账户 安装后打开命令行,输入 sqlplus 回车后会提示输入用户名,输入 sys或者system 回车后输入密码,密码为安装or ...
- SQL学习之简单增删改查
SQL最常用的语句,就是增删改查: 增删改查的对象,分别是库(文件夹),表(文件),表的内容(表的记录): 一.创建一个基本的表 #create table Student_Info (Name VA ...
- Django——6 模型基础ORM 数据库连接配置 模型的创建与映射 数据的增删改查
Django Django的ORM简介 数据库连接配置 模型的创建与映射 数据库的增删改查 增数据 查数据及补充 改数据 删数据 Django的ORM系统分析 ORM概念:对象关系映射(Objec ...
随机推荐
- Java内存泄漏分析系列之四:jstack生成的Thread Dump日志线程状态
原文地址:http://www.javatang.com Thread Dump日志的线程信息 以下面的日志为例: "resin-22129" daemon prio=10 tid ...
- 安卓高级9 用原生intent分享
大家都用过安卓app时发现有个分享按钮如下: 所以今天特此分享用用原生完成: package qianfeng.com.simplesharedemo; import android.content. ...
- JBOSS EAP实战(1)
JBOSS的诞生 1998年,在硅谷SUN公司的SAP实验室,一个年轻人正坐在电脑前面思考,然后写着什么东西.不,他没有在写程序,他在写辞呈.他正在做出人生的一个重大决定:他要辞掉在SUN的这份工作, ...
- intel-hadoop/HiBench流程分析----以贝叶斯算法为例
1.HiBench算法简介 Hibench 包含9个典型的hadoop负载(micro benchmarks,hdfs benchmarks,web search bench marks,machin ...
- ROS(indigo) 语音工具 科大讯飞 百度 pocketsphinx julius rospeex 16.11.22更新 ROS中文语音
ROS语音工具汇总,目前先给出链接,只用过一些简单的命令. 中文语音: 参考链接:使用科大讯飞库 1 http://www.ncnynl.com/archives/201611/1069.html 2 ...
- 详解EBS接口开发之应收款处理
参考实例参考:杜春阳 R12应收模块收款API研究 (一)应收款常用标准表简介 1.1 常用标准表 如下表中列出了与应收款处理相关的表和说明: 表名 说明 其他信息 AR_BATCHES_ALL ...
- Android Studio中Git的配置及协同开发
转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51595096 本文出自:[openXu的博客] 目录: 一 Android Stutio配置 ...
- 使用DWR实现自动补全 类似百度搜索框的自动显示效果
使用DWR实现自动补全 自动补全:是指用户在文本框中输入前几个字母或汉字的时候,自动在存放数据的文件或数据库中将所有以这些字母或汉字开头的数据提示给用户供用户选择 在日常上网过程中,我们经常使用搜索引 ...
- NDK在windows下的开发环境搭建及开发过程
在Android应用的开发工程中,不管是游戏还是普通应用,都时常会用到.so即动态链接库,关于.so是什么玩意儿,有什么好处,这个大家可以在网上查一下,本人不做过多解释..so本是linux下的文件类 ...
- TortoiseSVN文件夹图标不显示
伴随着十二月的脚步,小编带领的市委组织部项目有条不紊的进行着,在最近的项目中遇到一个问题TortoiseSVN文件夹的图标不显示,为什么小编已经安装好TortoiseSVN了,发现文件夹的图标还是系统 ...