8.2 Hibernate:多对一单向关联(unidirectional)

2023-12-16 04:48

本文主要是介绍8.2 Hibernate:多对一单向关联(unidirectional),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 建立多对一应用场景

以人与手机的关系为例,同一时间点上,一部手机只能由一个人使用,而一个人可以同时使用一部或多部手机。手机与人之间是多对一的关系。
使用 MySQL 数据库:

CREATE TABLE `test`.`person` (`id` INT UNSIGNED NOT NULL AUTO_INCREMENT,`name` VARCHAR(16) NOT NULL,`birth` TIMESTAMP NOT NULL,PRIMARY KEY (`id`),UNIQUE INDEX `id_UNIQUE` (`id` ASC))
ENGINE = InnoDB
DEFAULT CHARACTER SET = utf8
COMMENT = '人';
CREATE TABLE `test`.`phone` (`id` INT UNSIGNED NOT NULL AUTO_INCREMENT,`imei` VARCHAR(15) NOT NULL,`number` VARCHAR(11) NOT NULL,`owner_id` INT UNSIGNED NULL,PRIMARY KEY (`id`),UNIQUE INDEX `id_UNIQUE` (`id` ASC),UNIQUE INDEX `imei_UNIQUE` (`imei` ASC),UNIQUE INDEX `number_UNIQUE` (`number` ASC),INDEX `FK_PHONE_idx` (`owner_id` ASC),CONSTRAINT `FK_PHONE`FOREIGN KEY (`owner_id`)REFERENCES `test`.`person` (`id`)ON DELETE NO ACTIONON UPDATE NO ACTION)
ENGINE = InnoDB
DEFAULT CHARACTER SET = utf8
COMMENT = '手机';

2 定义映射类
2.1 表 person 的映射类定义:

package hibernate;import java.util.Date;import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;@Entity
@Table(name = "person")
public class Person {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)@Column(name = "id")private int id;@Column(name = "name")private String name;@Column(name = "birth")private Date birth;public Person() {}public Person(String name, Date birth) {this.name = name;this.birth = birth;}// 省略 Getters 和 Setters ...}

2.2 表 phone 的映射类定义:

package hibernate;import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.ForeignKey;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.Table;@Entity
@Table(name = "phone")
public class Phone {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)@Column(name = "id")private int id;@Column(name = "imei")private String imei;@Column(name = "number")private String number;@ManyToOne@JoinColumn(name = "owner_id", foreignKey = @ForeignKey(name = "FK_PHONE"))private Person person;public Phone() {}public Phone(String imei, String number) {this.imei = imei;this.number = number;}// 省略 Getters 和 Setters ...}

3 测试
3.1 新增

@Test
public void test() {Configuration configuration = new Configuration().configure("hibernate.cfg.xml");SessionFactory sessionFactory = configuration.buildSessionFactory();Session session = sessionFactory.openSession();Transaction transaction = session.beginTransaction();Person person = new Person("Jack", new Date());session.save(person);Phone phone1 = new Phone("866661234567890", "18012345678");phone1.setPerson(person);session.save(phone1);Phone phone2 = new Phone("866660987654321", "18087654321");phone2.setPerson(person);session.save(phone2);transaction.commit();session.close();sessionFactory.close();
}

运行测试,对象 personphone1phone2 都被持久化存储到对应的 personphone 数据表中,且 phone1phone2 对应记录的外键都是新插入的 person 对应记录的主键(日志省略)。

如果将单元测试代码中 session.save(phone1);session.save(phone2); 屏蔽掉,则只会持久化存储 person 对象。

如果将单元测试代码中 session.save(person); 屏蔽掉,默认情况下单元测试报异常,不能成功持久化存储任一对象数据。

java.lang.IllegalStateException: org.hibernate.TransientObjectException: object references an unsaved transient instance - save the transient instance beforeQuery flushing

由此可见:新增多对一关系数据时,默认首先需要保存“一”方的数据,只能成功保存了一”方的数据后才有可能保存“多”方的数据,除非修改 @ManyToOnecascade 属性,才能实现级联插入和更新(后续会详细阐述 cascade 属性的用途)

3.2 删除
基于以上测试新增的数据进行删除测试。

@Test
public void test() {Configuration configuration = new Configuration().configure("hibernate.cfg.xml");SessionFactory sessionFactory = configuration.buildSessionFactory();Session session = sessionFactory.openSession();Transaction transaction = session.beginTransaction();List<Phone> phones = session.createQuery(" FROM Phone").list();for (Phone phone : phones) {session.delete(phone);}Person person = (Person) session.createQuery(" FROM Person").list().get(0);session.delete(person);transaction.commit();session.close();sessionFactory.close();
}

注意:
(1) 可以只删除 phone,默认不会对其关联的 person 产生影响,除非修改 @ManyToOnecascade 属性(后续会详细阐述 cascade 属性的用途)
(2) 删除 person 前必须首先删除外键关联的 phone,否则出现以下异常。

Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException: Cannot delete or update a parent row: a foreign key constraint fails

3.3 更新

@Test
public void test() {Configuration configuration = new Configuration().configure("hibernate.cfg.xml");SessionFactory sessionFactory = configuration.buildSessionFactory();Session session = sessionFactory.openSession();Transaction transaction = session.beginTransaction();Phone phone = (Phone) session.createQuery(" FROM Phone").list().get(0);phone.setImei("123456789012345");phone.setNumber("15912345678");phone.getPerson().setName("Bruce");phone.getPerson().setBirth(new Date());transaction.commit();session.close();sessionFactory.close();
}

运行测试,phone 关联的 person 级联更新。

3.4 查询

@Test
public void test() {Configuration configuration = new Configuration().configure("hibernate.cfg.xml");SessionFactory sessionFactory = configuration.buildSessionFactory();Session session = sessionFactory.openSession();Transaction transaction = session.beginTransaction();List<Phone> phones = session.createQuery(" FROM Phone").list();for (int i = 1; i <= phones.size(); i++) {System.out.println("---------- Phone " + i + "----------");System.out.println("IMEI : " + phones.get(i - 1).getImei());System.out.println("Number : " + phones.get(i - 1).getNumber());System.out.println("Owner Name : " + phones.get(i - 1).getPerson().getName());System.out.println("Owner Birth : " + phones.get(i - 1).getPerson().getBirth());}transaction.commit();session.close();sessionFactory.close();
}

运行测试,打印结果:

---------- Phone 1----------
IMEI : 866661234567890
Number : 18012345678
Owner Name : Jack
Owner Birth : 2017-07-09 21:07:49.0
---------- Phone 2----------
IMEI : 866660987654321
Number : 18087654321
Owner Name : Jack
Owner Birth : 2017-07-09 21:07:49.0

这篇关于8.2 Hibernate:多对一单向关联(unidirectional)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/499153

相关文章

java面试常见问题之Hibernate总结

1  Hibernate的检索方式 Ø  导航对象图检索(根据已经加载的对象,导航到其他对象。) Ø  OID检索(按照对象的OID来检索对象。) Ø  HQL检索(使用面向对象的HQL查询语言。) Ø  QBC检索(使用QBC(Qurey By Criteria)API来检索对象。 QBC/QBE离线/在线) Ø  本地SQL检索(使用本地数据库的SQL查询语句。) 包括Hibern

org.hibernate.hql.ast.QuerySyntaxException:is not mapped 异常总结

org.hibernate.hql.ast.QuerySyntaxException: User is not mapped [select u from User u where u.userName=:userName and u.password=:password] 上面的异常的抛出主要有几个方面:1、最容易想到的,就是你的from是实体类而不是表名,这个应该大家都知道,注意

Caused by: org.hibernate.MappingException: Could not determine type for: org.cgh.ssh.pojo.GoodsType,

MappingException:这个主要是类映射上的异常,Could not determine type for: org.cgh.ssh.pojo.GoodsType,这句话表示GoodsType这个类没有被映射到

Hibernate框架中,使用JDBC语法

/*** 调用存储过程* * @param PRONAME* @return*/public CallableStatement citePro(final String PRONAME){Session session = getCurrentSession();CallableStatement pro = session.doReturningWork(new ReturningWork<C

hibernate修改数据库已有的对象【简化操作】

陈科肇 直接上代码: /*** 更新新的数据并并未修改旧的数据* @param oldEntity 数据库存在的实体* @param newEntity 更改后的实体* @throws IllegalAccessException * @throws IllegalArgumentException */public void updateNew(T oldEntity,T newEntity

C++ STL关联容器Set与集合论入门

1. 简介 Set(集合)属于关联式容器,也是STL中最实用的容器,关联式容器依据特定的排序准则,自动为其元素排序。Set集合的底层使用一颗红黑树,其属于一种非线性的数据结构,每一次插入数据都会自动进行排序,注意,不是需要排序时再排序,而是每一次插入数据的时候其都会自动进行排序。因此,Set中的元素总是顺序的。 Set的性质有:数据自动进行排序且数据唯一,是一种集合元素,允许进行数学上的集合相

关联规则(一)Apriori算法

此篇文章转自 http://blog.sina.com.cn/s/blog_6a17628d0100v83b.html 个人觉得比课本上讲的更通俗易懂! 1.  挖掘关联规则 1.1   什么是关联规则 一言蔽之,关联规则是形如X→Y的蕴涵式,表示通过X可以推导“得到”Y,其中X和Y分别称为关联规则的先导(antecedent或left-hand-side, LHS)和后

Hibernate插入数据时,报错:org.springframework.dao.DataIntegrityViolationException: could not insert: [cn.itc

在用junit测试:插入数据时,报一下错误: 错误原因: package junit;import org.junit.Test;import cn.itcast.crm.container.ServiceProvinder;import cn.itcast.crm.dao.ISysUserDao;import cn.itcast.crm.domain.SysRole;

Hibernate中自带的连接池!!!

<span style="font-size:18px; font-family: Arial, Helvetica, sans-serif;"><?xml version="1.0" encoding="UTF-8"?></span> <span style="font-size:18px;"><!DOCTYPE hibernate-configuration PUBLIC"-//Hibern

【数据库实战】1_Oracle_命中关联人或黑名单或反洗钱客户

一、字段名称 1、CST_ID :客户编号 2、IDV_LGL_NM :客户姓名 3、关联方标志 RELPARTY_IND,0-否 未命中,1-是 命中 4、TBPC1010表,RSRV_FLD1_INF(备用字段)中的 第6位:黑名单标志,0无,1是。 第10位:反洗钱风险等级1-5。 反洗钱风险等级5级: 1级-低风险客户 2级-较低风险客户 3级-中风险客户 4级-较高风险客户 5级-高风