Hibernate(八):基于外键映射的1-1关联关系
- 背景:
一个部门只有一个一把手,这在程序开发中就会设计数据映射应该设置为一对一关联。
在hibernate代码开发中,实现这个业务有两种方案:
1)基于外键映射的1-1关联;
2)基于主键映射的1-1关联。
本篇文章主要是用来学习如何使用外键实现1-1关联关系。
- 新建项目hibernate05
新建java project,引入依赖包,在src下添加hibernate.cfg.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<property name="hibernate.connection.username">root</property>
<property name="hibernate.connection.password">123456</property>
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="hibernate.connection.url">jdbc:mysql://localhost/hibernate_01</property> <!-- <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="hibernate.dialect">org.hibernate.dialect.MySQLInnoDBDialect</property> -->
<property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property> <property name="hibernate.show_sql">true</property> <property name="hibernate.format_sql">true</property> <property name="hibernate.hbm2ddl.auto">update</property> <property name="hibernate.current_session_context_class">thread</property> <property name="hibernate.c3p0.max_size">500</property>
<property name="hibernate.c3p0.min_size">20</property>
<property name="hibernate.c3p0.max_statements">10</property>
<property name="hibernate.c3p0.timeout">2000</property>
<property name="hibernate.c3p0.idle_test_period">2000</property>
<property name="hibernate.c3p0.acquire_increment">10</property> <mapping resource="com/dx/hibernate005/onetoonebyforigenkey/Deparment.hbm.xml" />
<mapping class="com.dx.hibernate005.onetoonebyforigenkey.Deparment" />
<mapping resource="com/dx/hibernate005/onetoonebyforigenkey/Manager.hbm.xml" />
<mapping class="com.dx.hibernate005.onetoonebyforigenkey.Manager" />
</session-factory>
</hibernate-configuration>
在src下新建包com.dx.hibernate005.onetoonebyforigenkey
在com.dx.hibernate005.onetoonebyforigenkey包下创建
Deparment.java
package com.dx.hibernate005.onetoonebyforigenkey; public class Deparment {
private Integer departId;
private String departName;
private Manager manager; public Deparment() {
super();
} public Deparment(Integer departId, String departName) {
super();
this.departId = departId;
this.departName = departName;
} public Integer getDepartId() {
return departId;
} public void setDepartId(Integer departId) {
this.departId = departId;
} public String getDepartName() {
return departName;
} public void setDepartName(String departName) {
this.departName = departName;
} public Manager getManager() {
return manager;
} public void setManager(Manager manager) {
this.manager = manager;
} }
Deparment.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-6-2 10:54:26 by Hibernate Tools 3.5.0.Final -->
<hibernate-mapping>
<class name="com.dx.hibernate005.onetoonebyforigenkey.Deparment" table="DEPARMENTS">
<id name="departId" type="java.lang.Integer">
<column name="DEPART_ID" />
<generator class="native" />
</id>
<property name="departName" type="java.lang.String">
<column name="DEPART_NAME" />
</property> <!-- 在Deparment中定义一个Many-to-one就可以实现在Deparments表中多一个外键列 -->
<many-to-one name="manager" class="com.dx.hibernate005.onetoonebyforigenkey.Manager" column="MANAGER_ID" unique="true"></many-to-one>
</class>
</hibernate-mapping>
Manager.java
package com.dx.hibernate005.onetoonebyforigenkey; public class Manager {
private Integer managerId;
private String managerName;
private Deparment deparment; public Manager() {
super();
} public Manager(Integer managerId, String managerName) {
super();
this.managerId = managerId;
this.managerName = managerName;
} public Integer getManagerId() {
return managerId;
} public void setManagerId(Integer managerId) {
this.managerId = managerId;
} public String getManagerName() {
return managerName;
} public void setManagerName(String managerName) {
this.managerName = managerName;
} public Deparment getDeparment() {
return deparment;
} public void setDeparment(Deparment deparment) {
this.deparment = deparment;
} }
Deparment.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-6-2 10:54:26 by Hibernate Tools 3.5.0.Final -->
<hibernate-mapping>
<class name="com.dx.hibernate005.onetoonebyforigenkey.Manager" table="MANAGERS">
<id name="managerId" type="java.lang.Integer">
<column name="MANAGER_ID" />
<generator class="native" />
</id>
<property name="managerName" type="java.lang.String">
<column name="MANAGER_NAME" />
</property> <!-- 在对应的数据表中已经有了外键,当前持久化类使用one-to-one进行映射 -->
<one-to-one name="deparment" class="com.dx.hibernate005.onetoonebyforigenkey.Deparment" property-ref="manager"></one-to-one>
</class>
</hibernate-mapping>
TestMain.java测试类,并引入JUnit4所需的依赖包。
package com.dx.hibernate005.onetoonebyforigenkey; import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.boot.Metadata;
import org.hibernate.boot.MetadataSources;
import org.hibernate.boot.model.naming.ImplicitNamingStrategyComponentPathImpl;
import org.hibernate.boot.registry.StandardServiceRegistry;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test; public class TestMain {
private SessionFactory sessionFactory = null;
private Session session = null;
private Transaction transaction = null; @Before
public void init() {
StandardServiceRegistry standardRegistry = new StandardServiceRegistryBuilder().configure().build();
Metadata metadata = new MetadataSources(standardRegistry).getMetadataBuilder().applyImplicitNamingStrategy(ImplicitNamingStrategyComponentPathImpl.INSTANCE).build(); sessionFactory = metadata.getSessionFactoryBuilder().build();
session = sessionFactory.getCurrentSession();
transaction = session.beginTransaction();
} @After
public void destory() {
transaction.commit();
session.close();
sessionFactory.close();
}
}
运行空测试类,初始化sql:
Hibernate: create table DEPARMENTS (
DEPART_ID integer not null auto_increment,
DEPART_NAME varchar(255),
MANAGER_ID integer,
primary key (DEPART_ID)
) engine=InnoDB
Hibernate: create table MANAGERS (
MANAGER_ID integer not null auto_increment,
MANAGER_NAME varchar(255),
primary key (MANAGER_ID)
) engine=InnoDB
Hibernate: alter table DEPARMENTS
drop index UK_j9bdvgylyeboig73ntlfcuyf6
Hibernate: alter table DEPARMENTS
add constraint UK_j9bdvgylyeboig73ntlfcuyf6 unique (MANAGER_ID)
Hibernate: alter table DEPARMENTS
add constraint FKnlmf88ii3maigw7hgc182cqi4
foreign key (MANAGER_ID)
references MANAGERS (MANAGER_ID)
- 测试:
1)测试insert:
@Test
public void testInsert(){
Deparment deparment=new Deparment();
deparment.setDepartName("Depart1"); Manager manager=new Manager();
manager.setManagerName("manager1"); deparment.setManager(manager);
manager.setDeparment(deparment); session.save(manager);
session.save(deparment);
}
测试通过,测试打印sql
Hibernate:
insert
into
MANAGERS
(MANAGER_NAME)
values
(?)
Hibernate:
insert
into
DEPARMENTS
(DEPART_NAME, MANAGER_ID)
values
(?, ?)
2)测试select
@Test
public void testSelect() {
// 1.查询时,采用懒加载。
Deparment deparment = (Deparment) session.get(Deparment.class, 1);
System.out.println(deparment.getDepartName()); // // 2.会出现懒加载异常问题
// session.close();
// Manager manager = deparment.getManager();
// System.out.println(manager.getClass());
// System.out.println(manager.getManagerName());
}
3)测试Select Manager
@Test
public void testSelectManager() {
// 1.查询时,没采用懒加载。
Manager manager = session.get(Manager.class, 1);
System.out.println(manager.getManagerName()); // 2.不会出现懒加载异常问题
session.close();
System.out.println(manager.getDeparment().getDepartName());
}
4)测试修改
@Test
public void testUpdate() {
// // 1) session关闭时,修改执行
// Manager manager = session.get(Manager.class, 1);
// manager.setManagerName("Mgr11111");
// // session.save(manager);
//
// Deparment deparment = session.get(Deparment.class, 1);
// deparment.setDepartName("Depart111111");
// // session.save(deparment); // // 2)通过manager 关联修改deparment
// Manager manager = session.get(Manager.class, 1);
// manager.setManagerName("Mgr222");
// manager.getDeparment().setDepartName("DEPT22"); // 3)通过department关联修改manager
Deparment deparment = session.get(Deparment.class, 1);
deparment.setDepartName("DPT333");
deparment.getManager().setManagerName("Mgr333");
}
5)测试删除
@Test
public void testDelete() {
// 1)在department表中有记录与之关联时,删除失败
// Manager manager = session.get(Manager.class, 1);
// session.delete(manager); // 2)删除成功,只删除了deparment表中的记录,manger表中的记录并没有删除,之后再删除manager就不会出现错误。
Deparment deparment = session.get(Deparment.class, 1);
session.delete(deparment);
}
Hibernate(八):基于外键映射的1-1关联关系的更多相关文章
- Hibernate之基于外键映射的一对一(1-1)关联关系
1.对于基于外键的1-1关联,其外键可以存放在任意一边,在需要存放外键一端,增加many-to-one元素.为many-to-one元素增加unique="true"属性来表示为1 ...
- hibernate中基于主键映射1-1关联关系和基于外键映射1-1关联关系的不同
基于主键映射1-1关联关系和基于外键映射1-1关联关系的不同,主要区别是在配置映射文件上会有区别 两个持久化类为Manager和Department 1:基于主键映射1-1关联关系 1)使用其他持久化 ...
- hibernate笔记--基于外键的单(双)向的一对一映射关系
假设我们有两张表,人员信息表Person,和身份信息表IdCard,我们知道每个人只有一个身份证号,所以这里的Person和IdCard表是一一对应的,也就是一对一的映射关系,基于外键的单向一对一映射 ...
- java之hibernate之基于外键的双向一对一关联映射
这篇讲解 基于外键的双向一对一关联映射 1.考察如下信息,人和身份证之间是一个一对一的关系.表的设计 2.类结构 Person.java public class Person implements ...
- java之hibernate之基于外键的一对一单向关联映射
这篇讲解基于外键的一对一单向关联映射 1.考察如下信息,人和身份证之间是一个一对一的关系.表的设计 注意:基于外键的一对一关联的表结构和多对一的表结构是一致的,但是,外键是唯一的. 2.类的结构 Pe ...
- Hibernate 基于外键映射的一对一关联关系随手记
//有外键的一端默认使用懒加载. //没有外键的一端不使用懒加载,而是直接将它引用的对象也一并查询出来. //没有外键列不仅有外键约束还有唯一约束,即没有外键列一端的对象不能被有外键列一端的两个对象同 ...
- Hibernate之基于主键映射的一对一关联关系
1. 基于主键的映射策略:指一端的主键生成器使用foreign策略,表明根据"对方"的主键来生成自己的主键,自己并不能独立生成主键.并用<param> 子元素指定使用当 ...
- Hibernate学习(二)关系映射----基于外键的单向一对一
事实上,单向1-1与N-1的实质是相同的,1-1是N-1的特例,单向1-1与N-1的映射配置也非常相似.只需要将原来的many-to-one元素增加unique="true"属性, ...
- Hibernate关系映射(二) 基于外键的双向一对一
基于外键的双向一对一关联映射 需要在一端添加<one-to-one>标签,用property-ref来指定反向属性引用. 还是通过刚才用户和地址来演示双向一对一关联. 代码演示 一.实体类 ...
随机推荐
- Netty实现客户端和服务端通信简单例子
Netty是建立在NIO基础之上,Netty在NIO之上又提供了更高层次的抽象. 在Netty里面,Accept连接可以使用单独的线程池去处理,读写操作又是另外的线程池来处理. Accept连接和读写 ...
- 在使用document.getElementById('xxx').files[0]时,关于计算图片大小
在使用文件上传属性时,一直好奇图片上传的大小时如何计算的,最近在使用中认识到的计算方式: 首先,图片大小的存储基本单位是字节(byte).每个字节是由8个比特(bit)组成.所以,一个字节在十进制中 ...
- docker部署PMA(LNMP架构)
上篇文章中已经部署好php环境,测试也已经OK centos 7 docker 启动了一个web服务 但是启动时 报 WARNING: IPv4 forwarding is disabled. Net ...
- 设计模式 --> (17)状态模式
状态模式 允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它的类.它有两种使用情况: (1)一个对象的行为取决于它的状态, 并且它必须在运行时刻根据状态改变它的行为. (2)一个操作中 ...
- [转]C++ 初始化列表的初始化顺序
构造函数初始化列表仅用于初始化成员的值,并不指定这些初始化执行的次序.成员被初始化的次序就是定义成员的次序.第一个被定义的成员先被初始化,依次类推.一般,初始化的顺序无关紧要,然而,如果一个成员是根据 ...
- python全栈学习--day9(函数初始)
Python 函数 函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段. 函数能提高应用的模块性,和代码的重复利用率.你已经知道Python提供了许多内建函数,比如print().但你也 ...
- c# 动态实例化一个泛型类
动态实例化一个类,比较常见,代码如下 namespace ConsoleApp2 { public class MyClass { } } Type classType = Type.GetType( ...
- c语言-第零次作业
1.你认为大学的学习生活.同学关系.师生应该是怎样?请一个个展开描述. 我很荣幸能考进集美大学.集美大学历史悠久.师资力量雄厚.教师与学生素质高.并且集美大学的学习生活和我理想中的一样!首先老师认真负 ...
- TensorFlow实现Softmax Regression识别手写数字中"TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败”问题
出现问题: 在使用TensorFlow实现MNIST手写数字识别时,出现"TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应 ...
- 大神都在看的RxSwift 的完全入坑手册
大神都在看的RxSwift 的完全入坑手册 2015-09-24 18:25 CallMeWhy callmewhy 字号:T | T 我主要是通过项目里的 Rx.playground 进行学习和了解 ...