Hibernate笔记③--集合映射、组合映射、联合主键、查询案例
lazy 懒加载 默认为proxy
继承映射
discriminant column="type" type="string"
集合映射
生成表的语句:
|
public public Configuration cfg=new Configuration().configure("/hibernate.cfg.xml"); SchemaExport exprot=new exprot.create(true, true); } } |
集合的实体
|
private String id; private String name; private private private String[] arrayValue; private |
该配置文件
|
<hibernate-mapping> <class <id <generator </id>
<property <column </property>
<set <key <element </set>
<list <key <list-index <element </list> <array <key <list-index <element </array>
<map <key <map-key <element </map>
</class> </hibernate-mapping> |
生成的表结构

自己写的crud
|
package com.pcx.dao;
import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; import java.util.List;
import org.hibernate.Hibernate; import org.hibernate.Session; import org.hibernate.Transaction;
import com.pcx.model.Collection; import com.pcx.util.HibernateSessionFactory;
public class CollTest { public void testQueryColl(String id){ Session session=HibernateSessionFactory.getSession(); Collection c=(Collection) session.get(Collection.class, id); List list=c.getListValue(); for (Object o : list) { System.out.println(o); } HibernateSessionFactory.closeSession(); }
public void testColl(){ Collection c=new Collection(); c.setName("测试");
String[]ss={"a","b","c","d"}; ArrayList<String>list=new ArrayList<String>(); list.add("a"); list.add("c"); list.add("e");
HashSet<String>set=new HashSet<String>(); set.add("1"); set.add("3"); set.add("aaaa");
HashMap<String, String>map=new HashMap<String, String>(); map.put("1", "11"); map.put("2", "22"); map.put("3", "33");
c.setArrayValue(ss); c.setListValue(list); c.setSetValue(set); c.setMapValue(map);
Session session=HibernateSessionFactory.getSession(); Transaction tx=session.beginTransaction(); session.save(c); tx.commit(); HibernateSessionFactory.closeSession(); }
public static void main(String[] args) { CollTest test=new CollTest(); // test.testColl(); test.testQueryColl("4028d0814ecd7aa4014ecd7aa66c0001"); } } |
组合映射
两个实体类对应生成一张表格
Emp类:
|
private String id; private String name; private Address address; |
对应的地址类:address类
|
private String homeAddress; private String comAddress; |
对应的配置文件
|
<hibernate-mapping> <class <id <generator </id>
<property <column </property> <component <property <property </component> </class> </hibernate-mapping> |
对应生成的表结构为

自己写的crud
|
package com.pcx.dao;
import org.hibernate.Session; import org.hibernate.Transaction;
import com.pcx.model.Address; import com.pcx.model.Emp; import
public public Session session=HibernateSessionFactory.getSession(); Transaction tx=session.beginTransaction(); Address ad=new Address(); ad.setComAddress("济南长清"); ad.setHomeAddress("山东泰安"); Emp e=new Emp(); e.setName("程志"); e.setAddress(ad); session.save(e); tx.commit(); HibernateSessionFactory.closeSession(); }
public Session session=HibernateSessionFactory.getSession(); Emp e=(Emp) session.get(Emp.class,id ); System.out.println(e.getId()); System.out.println(e.getName()); System.out.println(e.getAddress().getHomeAddress()); }
public ComponentTest com=new ComponentTest(); // com.testCom(); com.testQueryCom("4028d0814ece0647014ece0648f20001"); } } |
联合主键
如果一个表没主键,那么由表生成类的时候会单独生成一个 id类

Student 类中就单独一个StudentId
StudentId中则包含 id name
生成的配置文件为:
|
<?xml version="1.0" encoding="utf-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <!-- Mapping file autogenerated by MyEclipse Persistence Tools --> <hibernate-mapping> <class name="com.pcx.model.Student" table="student" catalog="hibernate"> <composite-id name="id" class="com.pcx.model.StudentId"> <key-property name="id" type="java.lang.Integer"> <column name="id" /> </key-property> <key-property name="name" type="java.lang.String"> <column name="name" /> </key-property> </composite-id> </class> </hibernate-mapping> |
在一个例子
Emp
|
private PKId pkid; private String name; |
PKId
|
private Integer fristId; private Integer lastId; |
配置文件:
|
<?xml <!DOCTYPE hibernate-mapping "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class <composite-id <key-property <key-property </composite-id>
<property <column </property> </class> </hibernate-mapping> |
配置文件字段配置错了可能会出现
类的变量没配对
提示错误找不到配置文件。
多对多关系
多个学生对应多个老师
学生
private String id;
private String name;
private Set<Teacher> teachers;
老师
private String id;
private String name;
private Set<Student> students;
配置文件
学生
|
<hibernate-mapping> <class <id <generator </id> <property <column </property>
<set <key <many-to-many </set> </class> </hibernate-mapping> |
老师
|
<hibernate-mapping> <class <id <generator </id> <property <column </property>
<set <key <many-to-many </set> </class> </hibernate-mapping> |
对应的crud
|
package com.pcx.dao;
import java.util.HashSet; import java.util.Set;
import org.hibernate.Session; import org.hibernate.Transaction;
import com.pcx.entity.Student; import com.pcx.entity.Teacher; import
public private Session session=HibernateSessionFactory.getSession(); Transaction tx=session.beginTransaction();
Teacher t1=new Teacher(); t1.setName("彭"); Teacher t2=new Teacher(); t2.setName("李"); Student s1=new Student(); Student s2=new Student(); Student s3=new Student(); Student s4=new Student();
s1.setName("aaa"); s2.setName("aaa"); s3.setName("bbb"); s4.setName("aaa");
Set<Teacher>set=new HashSet<Teacher>(); set.add(t1);set.add(t2); s1.setTeachers(set);
session.save(s1);
tx.commit(); HibernateSessionFactory.closeSession(); } public Session session=HibernateSessionFactory.getSession(); Teacher t=(Teacher) session.get(Teacher.class, id); System.out.println(t.getName()); Set<Student>ss=t.getStudents(); for (Student student : ss) { System.out.println(student.getName()); } HibernateSessionFactory.closeSession(); } public TestManyToMany test=new TestManyToMany(); // test.testManyToMany(); test.testQuery("4028d0814ece5103014ece51053b0002"); } } |
一对一
查询
Hql
懒加载的问题
一对多的表关系中 默认是proxy
true 不加载关联的 老师查出来学生不被加载了
false 加载关联的
enchor代理
load验证id在表中是否存在, 不加载其他的属性,若不关闭Session 可以取出name 因为被代理掉了
同Query类的iterator
查询一个单体的时候
根据ID来查询 get load
get直接发送sql语句,直接将数据从数据库中加载到当前程序中,并且置入对象属性。
load不发送sql语句,只是查询当前给定的id在数据库中是否有这样一条记录对应存在
协助维护数据关系的
可以通过forward --过滤器 用于开启或关闭Session
unionQueresult()
分页
Criteria类
gt 大于1000的
lt 小于1000的
复合条件
.add().add()
排序 addorder
ReStrictions
结合离线查询
查询的案例代码
|
package com.pcx.dao;
import java.util.List;
import org.hibernate.Criteria; import org.hibernate.Query; import org.hibernate.Session; import org.hibernate.Transaction; import org.hibernate.criterion.DetachedCriteria; import org.hibernate.criterion.Order; import org.hibernate.criterion.Restrictions;
import com.pcx.model1.Emp; import com.pcx.util.HibernateSessionFactory;
public class QueryTest { public static void initDate(){ Session session=HibernateSessionFactory.getSession(); Transaction tx=session.beginTransaction(); for (int i = 0; i < 100; i++) { Emp e=new Emp(); e.setName("张三"+i); session.save(e); } tx.commit(); HibernateSessionFactory.closeSession(); } public static List<Emp> hql1(){ Session session=HibernateSessionFactory.getSession(); String hql=" from Emp e where e.sal>1000"; Query query=session.createQuery(hql); List<Emp>emps=query.list(); for (Emp emp : emps) { System.out.println(emp); } HibernateSessionFactory.closeSession(); return emps; } /** * 查询的第二种情况 * @param args */ public static void hql2(){ Session session=HibernateSessionFactory.getSession(); String hql="select name,age from Emp "; Query query=session.createQuery(hql); List emps=query.list(); Object[]obj=(Object[]) emps.get(0); String name=(String) obj[0]; Integer age=(Integer) obj[1]; System.out.println("name:"+name+" age:"+age); HibernateSessionFactory.closeSession(); } /** * 绑定变量 * @param args */ public static void bindVar(){ Session session=HibernateSessionFactory.getSession(); String hql="from Emp where name=:name"; Query q=session.createQuery(hql); q.setParameter("name", "张三"); List<Emp>list=q.list(); System.out.println( list.get(0));; HibernateSessionFactory.closeSession(); } // 使用Criteria public static void load1(){ Session s=HibernateSessionFactory.getSession(); Criteria c=s.createCriteria(Emp.class); //这个Criteria绝大多数的应用是增加条件 // c.add(Restrictions.eq("name", "李四")); // c.add(Restrictions.between("sal", 1000.00, 5000.00)); // c.add(Restrictions.gt("age", 20));//大于 // c.add(Restrictions.lt("age", 20));//小于 // c.add(null).add(null);//可以不停的去add 节省了拼sql的时间 // c.add(Restrictions.like("name", "李%")); // c.setFirstResult(0); // c.setMaxResults(20); c.addOrder(Order.asc("sal")); List<Emp>list=c.list(); for (Emp emp : list) { System.out.println(emp); } HibernateSessionFactory.closeSession(); } //离线查询 public static void load2(){ Session session=HibernateSessionFactory.getSession(); DetachedCriteria dc=DetachedCriteria.forClass(Emp.class); dc.add(Restrictions.eq("name", "李四")); List<Emp>list=dc.getExecutableCriteria(session).list(); for (Emp emp : list) { System.out.println(emp); } HibernateSessionFactory.closeSession(); } //分页查询 public static void load3(int firstRow,int maxResult){ Session session=HibernateSessionFactory.getSession(); Query query=session.createQuery("from Emp"); query.setFirstResult(firstRow); query.setMaxResults(maxResult); List<Emp>list=query.list(); for (Emp emp : list) { System.out.println(emp); } HibernateSessionFactory.closeSession(); } //分页查询二 public static void load4(int currentPage){ int pageSize=10; int totalPageSize=(int)getTotalSize(); int pageCount=totalPageSize/pageSize; if (totalPageSize%pageSize!=0) { pageCount++; } System.out.println("共"+pageCount+"页"); System.out.println("共"+totalPageSize+"条记录"); //如果客户给出的 if (currentPage<1) { currentPage=1; } //如果客户给出的当前页大于总页数。则给当前页赋值为最大页数 if(currentPage>pageCount){ currentPage = pageCount; } Session session = HibernateSessionFactory.getSession(); String hql = "from Emp order by id"; Query query = session.createQuery(hql); query.setFirstResult((currentPage-1)*pageSize); query.setMaxResults(pageSize); List<Emp> list = query.list(); for (Emp emp : list) { System.out.println(emp.getId()); System.out.println(emp.getName()); } HibernateSessionFactory.closeSession();
} private static long getTotalSize() { Session session = HibernateSessionFactory.getSession(); String hql ="select count(*) from Emp"; Query query = session.createQuery(hql); List list = query.list(); Object obj = list.get(0); return (Long) obj;
} public static void main(String[] args) { // List<Emp>emps=hql1(); //Session已经关闭,加载不了对象 // for (Emp emp : emps) { // System.out.println(emp); // } // hql2(); // bindVar(); // load2(); initDate(); } } |
Hibernate笔记③--集合映射、组合映射、联合主键、查询案例的更多相关文章
- Hibernate主配置文件、映射配置文件以及复合主键查询
Hibernate.cfg.xml主配置文件 主配置文件中主要配置:数据库连接信息.其他参数.映射信息! 常用配置查看源码: hibernate-distribution-3.6.0.Final\pr ...
- hibernate 联合主键 composite-id
如果表使用联合主键(一个表有两个以上的主键),你可以映射类的多个属性为标识符属性.如:<composite-id>元素接受<key-property> 属性映射(单表映射)和& ...
- Hibernate(5)—— 联合主键 、一对一关联关系映射(xml和注解) 和 领域驱动设计
俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及的知识点总结如下: One to One 映射关系 一对一单向外键(XML/Annotation) 一对一双向外键关联(XML/A ...
- Hibernate联合主键映射
1.联合主键的映射规则 1) 类中的每个主键属性都对应到数据表中的每个主键列. Hibernate要求具有联合主键的实体类实现Serializable接口,并且重写hashCode与equals方法, ...
- hibernate 注解 联合主键映射
联合主键用Hibernate注解映射方式主要有三种: 第一.将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并重写equals和hascode,再将 该类注解 ...
- Hibernate 中 联合主键映射 组合关系映射 大对象映射(或者说文本大对象,二进制数据大对象)
Clob:文本大对象,最长4G Blob:二进制数据大对象,最长4G util: public class HibUtil { private static SessionFactory sessio ...
- hibernate里联合主键composite-id映射,查询单个主键的问题
今天项目中遇到这个问题,搞了大半天,现在记录下来hibernate里联合主键配置(多个字段一起作为主键) <class name="com.cskj.hibernate.map.BbW ...
- Hibernate注解映射联合主键的三种主要方式
今天在做项目的时候,一个中间表没有主键,所有在创建实体的时候也未加组件,结果报以下错误: org.springframework.beans.factory.BeanCreationException ...
- 联合主键用Hibernate注解映射的三种方式
第一.将联合主键的字段单独放在一个类中,该类需要实现java.io.Serializable接口并重写equals和hascode,再将该类注解为@Embeddable,最后在主类中(该类不包含联合主 ...
随机推荐
- swoft orm中的坑(针对实体类的属性名称和数据库字段不相等)
最近在用swoft的orm,发现了一些问题: 首先看下实体类的定义 它的属性名称和所映射的数据库字段名不一致,这个就会导致蛋疼的问题,首先,在我们使用orm的时候,应该使用哪个字段? 我直接说结论,在 ...
- 2.Built-in types-基本数据类型(Dart中文文档)
初次翻译,部分内容并非按字面翻译,是按本人理解进行了内容重组.如有错误望指正. Dart语言内置如下数据类型: numbers strings booleans lists (所谓的数组) maps ...
- CBrother异或加密与C++异或加密函数
CBrother脚本异或加密与C++异或加密函数 异或对于数据加密来说是最简单的方式,在一般的安全性要求不是非常高的地方,异或加密是最好的选择. C++异或加密代码 int g_PWD = 0xffe ...
- 10-[协程] greenlet模块、 gevent模块
1.greenlet模块:实现20个任务切换 如果我们在单个线程内有20个任务,要想实现在多个任务之间切换,使用greenlet模块可以非常简单地实现这20个任务直接的切换 使用yield生成器的方式 ...
- gabor变换人脸识别的python实现,att_faces数据集平均识别率99%
大家都说gabor做人脸识别是传统方法中效果最好的,这几天就折腾实现了下,网上的python实现实在太少,github上的某个版本还误导了我好几天,后来采用将C++代码封装成dll供python调用的 ...
- jquery ajax 上传文件和传递参数到一个接口的实现方法
参考:https://blog.csdn.net/qq_15674631/article/details/81095284 参考:https://www.jianshu.com/p/46e6e03a0 ...
- Spring学习(十四)----- Spring Auto Scanning Components —— 自动扫描组件
一. Spring Auto Scanning Components —— 自动扫描组件 1. Declares Components Manually——手动配置componen ...
- Nginx应用场景
1. Nginx应用场景 1)http服务器.Nginx可以独立的提供http服务,可以做网页静态服务器(也就是将静态文件放到nginx目录下,通过nginx来访问就ok) 2)虚拟主机,可以在一 ...
- Linux下的文件系统结构
文章链接:https://blog.csdn.net/qq_38646470/article/details/80159630
- ABORT: Can't find command 'my_print_defaults'.
解决办法是输入如下命令: export PATH=/usr/local/mysql/bin:$PATH 将/usr/local/mysql/bin加到$PATH环境变量里