第1课 课程内容. 6

第2课Hibernate UML图. 6

第3课 风格. 7

第4课 资源. 7

第5课 环境准备. 7

第6课 第一个示例HibernateHelloWorld 7

第7课 建立Annotation版本的HellWorld 9

第8课 什么是O/RMapping 11

一、     定义:. 11

二、     Hibernate的创始人:. 11

三、     Hibernate做什么:. 12

四、     Hibernate存在的原因:. 12

五、     Hibernate的优缺点:. 12

六、     Hibernate使用范围:. 12

第9课Hibernate的重点学习:Hibernate的对象关系映射. 12

一、对象---关系映射模式. 12

二、常用的O/R映射框架:. 13

第10课 模拟Hibernate原理(OR模拟) 13

一、     项目名称. 13

二、     原代码. 13

第11课Hibernate基础配置. 15

一、     提纲. 15

二、     介绍MYSQL的图形化客户端. 16

三、     Hibernate.cfg.xml:hbm2ddl.auto 16

四、     搭建日志环境并配置显示DDL语句. 16

五、     搭建Junit环境. 16

六、     ehibernate.cfg.xml : show_sql 17

七、     hibernate.cfg.xml :format_sql 17

八、     表名和类名不同,对表名进行配置. 17

九、     字段名和属性相同. 17

十、     字段名和属性名不同. 17

十一、      不需要(持久化)psersistence的字段. 18

十二、      映射日期与时间类型,指定时间精度. 18

十三、      映射枚举类型. 19

第12课 使用hibernate工具类将对象模型生成关系模型. 19

第13课ID主键生成策略. 20

一、     Xml方式. 20

<generator>元素(主键生成策略) 20

二、     annotateon方式. 21

1、AUTO默认. 21

2、IDENTITY 22

3、SEQUENCE 22

4、为Oracle指定定义的Sequence 22

5、TABLE - 使用表保存id值. 23

三、     联合主键. 24

1、xml方式. 24

2、annotation方式. 27

第14课Hibernate核心开发接口(重点) 29

一、     Configuration(AnnotationConfiguration) 29

二、     SessionFactory 29

三、     Session 29

1、    管理一个数据库的任务单元. 29

2、    save(); 29

3、    delete() 29

4、    load() 29

5、    Get() 30

6、    load()与get()区别. 31

7、    update() 31

8、    saveOrUpdate() 32

9、    clear() 32

10、     flush() 33

11、     evict() 33

第15课 持久化对象的三种状态. 35

一、     瞬时对象(TransientObject):. 35

二、     持久化对象(PersistentObject):. 35

三、     离线对象(DetachedObject):. 35

四、     三种状态的区分:. 35

五、     总结:. 35

第16课 关系映射(重点) 36

一、     一对一 关联映射. 36

(一)     唯一外键关联-单向(unilateralism) 37

(二)     唯一外键关联-双向. 40

(三)     主键关联-单向(不重要) 41

(四)     主键关联-双向(不重要) 44

(五)     联合主键关联(Annotation方式) 44

二、     component(组件)关联映射. 45

(一)     Component关联映射:. 45

(二)     User实体类:. 45

(三)     Contact值对象:. 46

(四)     xml--User映射文件(组件映射):. 46

(五)     annotateon注解. 46

(六)     导出数据库输出SQL语句:. 47

(七)     数据表结构:. 47

(八)     组件映射数据保存:. 47

三、     多对一– 单向. 48

(一)     对象模型图:. 48

(二)     关系模型:. 48

(三)     关联映射的本质:. 48

(四)     实体类. 48

(五)     xml方式:映射文件:. 49

(六)     annotation 50

(七)     多对一 存储(先存储group(对象持久化状态后,再保存user)):. 50

(八)     重要属性-cascade(级联):. 51

(九)     多对一  加载数据. 51

四、     一对多- 单向. 51

(一)     对象模型:. 52

(二)     关系模型:. 52

(三)     多对一、一对多的区别:. 52

(四)     实体类. 52

(五)     xml方式:映射. 52

(六)     annotateon注解. 53

(七)     导出至数据库(hbmàddl)生成的SQL语句:. 53

(八)     一对多 单向存储实例:. 53

(九)     生成的SQL语句:. 54

(十)     一对多,在一的一端维护关系的缺点:. 54

(十一)    一对多 单向数据加载:. 54

(十二)    加载生成SQL语句:. 54

五、     一对多- 双向. 54

(一)     xml方式:映射. 55

(二)     annotateon方式注解. 55

(三)     数据保存:. 56

(四)     关于inverse属性:. 56

(五)     Inverse和cascade区别:. 56

(六)     一对多双向关联映射总结:. 57

六、     多对多- 单向. 57

(一)     实例场景:. 57

(二)     对象模型:. 57

(三)     关系模型:. 57

(四)     实体类. 57

(五)     xml方式:映射. 58

(六)     annotation注解方式. 58

(七)     生成SQL语句. 59

(八)     数据库表及结构:. 59

(九)     多对多关联映射 单向数据存储:. 59

(十)     多对多关联映射 单向数据加载:. 61

七、     多对多- 双向. 61

(一)     xml方式:映射. 61

(二)     annotation注解方式. 62

八、     关联关系中的CRUD_Cascade_Fetch 63

九、     集合映射. 63

十、     继承关联映射. 64

(一)     继承关联映射的分类:. 64

(二)     对象模型:. 64

(三)     单表继承SINGLE_TABLE:. 64

(四)     具体表继承JOINED:. 70

(五)     类表继承TABLE_PER_CLASS 72

(六)     三种继承关联映射的区别:. 74

第17课hibernate树形结构(重点) 75

一、     节点实体类:. 75

二、     xml方式:映射文件:. 75

三、     annotation注解. 76

四、     测试代码:. 76

五、     相应的类代码:. 76

第18课 作业-学生、课程、分数的映射关系. 79

一、     设计. 79

二、     代码:. 79

三、     注意. 80

第19课Hibernate查询(Query Language) 80

一、     Hibernate可以使用的查询语言. 80

二、     实例一. 80

三、     实体一测试代码:. 82

四、     实例二. 86

五、     实例二测试代码. 87

第20课Query by Criteria(QBC) 89

一、     实体代码:. 89

二、     Restrictions用法. 90

三、     工具类Order提供设置排序方式. 91

四、     工具类Projections提供对查询结果进行统计与分组操作. 91

五、     QBC分页查询. 92

六、     QBC复合查询. 92

七、     QBC离线查询. 92

第21课Query By Example(QBE) 92

一、     实例代码. 92

第22课Query.list与query.iterate(不太重要) 93

一、     query.iterate查询数据. 93

二、     query.list()和query.iterate()的区别. 94

三、     两次query.list() 94

第23课 性能优化策略. 95

第24课hibernate缓存. 95

一、     Session级缓存(一级缓存) 95

二、     二级缓存. 95

1、    二级缓存的配置和使用:. 96

2、    二级缓存的开启:. 96

3、    指定二级缓存产品提供商:. 96

4、    使用二级缓存. 97

5、    应用范围. 99

6、    二级缓存的管理:. 99

7、    二级缓存的交互. 100

8、    总结. 102

三、     查询缓存. 102

四、     缓存算法. 103

第25课 事务并发处理. 104

一、     数据库的隔离级别:并发性作用。. 104

1、    Mysql查看数据库隔离级别:. 104

2、    Mysql数据库修改隔离级别:. 104

二、     事务概念(ACID) 104

三、     事务并发时可能出现问题. 104

第26课hibernate悲观锁、乐观锁. 105

一、     悲观锁. 105

1、    悲观锁的实现. 105

2、    悲观锁的适用场景:. 105

3、    实例:. 105

4、    悲观锁的使用. 106

5、    执行输出SQL语句:. 106

二、     乐观锁. 107


第1课 课程内容

1、  HelloWorld

a)        Xml

b)        Annotction

2、  Hibernate原理模拟-什么是O/RMapping以及为什么要有O/RMapping

3、  常风的O/R框架

4、  Hibernate基础配置

5、  Hibernate核心接口介绍

6、  对象的三种状态

7、  ID生成策略

8、  关系映射

9、  Hibernate查询(HQL)

10、    在Struts基础上继续完美BBS2009

11、    性能优化

12、    补充话题


第2课 Hibernate UML图 Hibernate 学习参考_Hibernate 第3课 风格

1、  先脉络,后细节

2、  先操作、后原理

3、  重Annotation,轻xml配置文件

a)        JPA (可以认为EJB3的一部分)

b)        Hibernate– extension

第4课 资源

1、  http://www.hibernate.org

a)        hibernate-distribution-3.3.2.GA-dist.zip

b)        hibernate-annotations-3.4.0.GA.zip

c)         slf4j-1.5.10.zip    (hibernate内部日志用)

2、  hibernatezh_CN文档

3、  hibernateannotateon references

第5课 环境准备

1、  下载hibernate3.3.2

2、  下载hibernate-annotations-3.4.0

3、  注意阅读hibernate compatibility matrix

4、  下载slf4j 1.5.8

第6课 第一个示例Hibernate HelloWorld

1、  建立新的java项目,名为hibernate_0100_HelloWorld

2、  学习建立User-liberary-hibernate,并加入相应的jar包

a)        项目右键-build path-configure build path-add library

b)        选择User-library ,在其中新建library,命名为hibernate

c)         在该library中加入hibernate所需的jar名

                        i.             Hibernatecore

                       ii.             /lib/required

                     iii.             Slf-nopjar

3、  引入mysql的JDBC驱动名

4、  在mysql中建立对应的数据库以及表

a)        Create databasehibernate;

b)        Usehibernate;

c)         Createtable Student (id int primary key, name varchar(20),age int);

5、  建立hibernate配置文件hibernate.cfg.xml

a)        从参考文档中copy

b)        修改对应的数据库连接

c)         注释提暂时不需要的内容

6、  建立Student类

7、  建立Student映射文件Student.hbm.xml

a)        参考文档

8、  将映射文件加入到hibernate.cfg.xml

a)        参考文档

9、  写测试类Main,在Main中对Student对象进行直接的存储测试


  1. public staticvoid main(String[] args) { 
  2.          
  3.         Configuration cfg = null
  4.         SessionFactory sf = null
  5.         Session session = null
  6.          
  7.         Student s = new Student(); 
  8.         s.setId(2); 
  9.         s.setName("s1"); 
  10.         s.setAge(1); 
  11.          
  12.         /*
  13.          * org.hibernate.cfg.Configuration类的作用:
  14.          * 读取hibernate配置文件(hibernate.cfg.xml或hiberante.properties)的.
  15.          * new Configuration()默认是读取hibernate.properties
  16.          * 所以使用new Configuration().configure();来读取hibernate.cfg.xml配置文件
  17.          */ 
  18.         cfg = new Configuration().configure(); 
  19.          
  20.         /*
  21.          * 创建SessionFactory
  22.          * 一个数据库对应一个SessionFactory
  23.          * SessionFactory是线线程安全的。
  24.          */ 
  25.         sf = cfg.buildSessionFactory(); 
  26.          
  27.         try
  28.             //创建session 
  29.             //此处的session并不是web中的session 
  30.             //session只有在用时,才建立concation,session还管理缓存。 
  31.             //session用完后,必须关闭。 
  32.             //session是非线程安全,一般是一个请求一个session. 
  33.             session = sf.openSession(); 
  34.              
  35.             //手动开启事务(可以在hibernate.cfg.xml配置文件中配置自动开启事务) 
  36.             session.beginTransaction(); 
  37.             /*
  38.              * 保存数据,此处的数据是保存对象,这就是hibernate操作对象的好处,
  39.              * 我们不用写那么多的JDBC代码,只要利用session操作对象,至于hibernat如何存在对象,这不需要我们去关心它,
  40.              * 这些都有hibernate来完成。我们只要将对象创建完后,交给hibernate就可以了。
  41.              */ 
  42.             session.save(s); 
  43.             session.getTransaction().commit(); 
  44.         } catch (HibernateException e) { 
  45.             e.printStackTrace(); 
  46.             //回滚事务 
  47.             session.getTransaction().rollback(); 
  48.         } finally
  49.             //关闭session 
  50.             session.close(); 
  51.             sf.close(); 
  52.         }    
  53.     } 
  1. public static void main(String[] args) {  
  2.           
  3.         Configuration cfg = null;  
  4.         SessionFactory sf = null;  
  5.         Session session = null;  
  6.           
  7.         Student s = new Student();  
  8.         s.setId(2);  
  9.         s.setName("s1");  
  10.         s.setAge(1);  
  11.           
  12.         /* 
  13.          * org.hibernate.cfg.Configuration类的作用: 
  14.          * 读取hibernate配置文件(hibernate.cfg.xml或hiberante.properties)的. 
  15.          * new Configuration()默认是读取hibernate.properties 
  16.          * 所以使用new Configuration().configure();来读取hibernate.cfg.xml配置文件 
  17.          */  
  18.         cfg = new Configuration().configure();  
  19.           
  20.         /* 
  21.          * 创建SessionFactory 
  22.          * 一个数据库对应一个SessionFactory  
  23.          * SessionFactory是线线程安全的。 
  24.          */  
  25.         sf = cfg.buildSessionFactory();  
  26.           
  27.         try {  
  28.             //创建session  
  29.             //此处的session并不是web中的session  
  30.             //session只有在用时,才建立concation,session还管理缓存。  
  31.             //session用完后,必须关闭。  
  32.             //session是非线程安全,一般是一个请求一个session.  
  33.             session = sf.openSession();  
  34.               
  35.             //手动开启事务(可以在hibernate.cfg.xml配置文件中配置自动开启事务)  
  36.             session.beginTransaction();  
  37.             /* 
  38.              * 保存数据,此处的数据是保存对象,这就是hibernate操作对象的好处, 
  39.              * 我们不用写那么多的JDBC代码,只要利用session操作对象,至于hibernat如何存在对象,这不需要我们去关心它, 
  40.              * 这些都有hibernate来完成。我们只要将对象创建完后,交给hibernate就可以了。 
  41.              */  
  42.             session.save(s);  
  43.             session.getTransaction().commit();  
  44.         } catch (HibernateException e) {  
  45.             e.printStackTrace();  
  46.             //回滚事务  
  47.             session.getTransaction().rollback();  
  48.         } finally {  
  49.             //关闭session  
  50.             session.close();  
  51.             sf.close();  
  52.         }     
  53.     }  

1、  FAQ:

a)        要调用newConfiguration().configure().buildSessionFactory(),而不是省略

*       org.hibernate.cfg.Configuration类的作用:

         *读取hibernate配置文件(hibernate.cfg.xml或hiberante.properties)的.

         *new Configuration()默认是读取hibernate.properties

         * 所以使用new Configuration().configure();来读取hibernate.cfg.xml配置文件

注意:在hibernate里的操作都应该放在事务里

第7课 建立Annotation版本的HellWorld

         注意:要求hibernate3.0版本以后支持

1、  创建teacher表,create table teacher(id int primary key,namevarchar(20),title varchar(10));

2、  创建Teacher类

  1. public class Teacher { 
  2.     private int id;  
  3.     private String name;     
  4.     private String title; 
  5.     //设置主键使用@Id 
  6.     public int getId() { 
  7.         return id; 
  8.     } 
  9.     public void setId(int id) { 
  10.         this.id = id; 
  11.     } 
  12.     public String getName() { 
  13.         return name; 
  14.     } 
  15.     public void setName(String name) { 
  16.         this.name = name; 
  17.     } 
  18.     public String getTitle() { 
  19.         return title; 
  20.     } 
  21.     public void setTitle(String title) { 
  22.         this.title = title; 
  23.     } 
  1. public class Teacher {  
  2.     private int id;   
  3.     private String name;      
  4.     private String title;  
  5.     //设置主键使用@Id  
  6.     public int getId() {  
  7.         return id;  
  8.     }  
  9.     public void setId(int id) {  
  10.         this.id = id;  
  11.     }  
  12.     public String getName() {  
  13.         return name;  
  14.     }  
  15.     public void setName(String name) {  
  16.         this.name = name;  
  17.     }  
  18.     public String getTitle() {  
  19.         return title;  
  20.     }  
  21.     public void setTitle(String title) {  
  22.         this.title = title;  
  23.     }  
  24. }  

1、  在hibernate library中加入annotation的jar包

a)        Hibernateannotations jar

b)        Ejb3persistence jar

c)         Hibernatecommon annotations jar

d)        注意文档中没有提到hibernate-common-annotations.jar文件

2、  参考Annotation文档建立对应的注解

  1. import javax.persistence.Entity; 
  2. import javax.persistence.Id; 
  3. /** @Entity 表示下面的这个Teacher是一个实体类
  4. * @Id 表示主键Id*/ 
  5. @Entity //*** 
  6. public class Teacher { 
  7.     private int id;  
  8.     private String name;     
  9.     private String title; 
  10.     //设置主键使用@Id 
  11.     @Id  //*** 
  12.     public int getId() { 
  13.         return id; 
  14.     } 
  15.     public void setId(int id) { 
  16.         this.id = id; 
  17.     } 
  18.     public String getName() { 
  19.         return name; 
  20.     } 
  21.     public void setName(String name) { 
  22.         this.name = name; 
  23.     } 
  24.     public String getTitle() { 
  25.         return title; 
  26.     } 
  27.     public void setTitle(String title) { 
  28.         this.title = title; 
  29.     }} 
  1. import javax.persistence.Entity;  
  2. import javax.persistence.Id;  
  3. /** @Entity 表示下面的这个Teacher是一个实体类 
  4.  * @Id 表示主键Id*/  
  5. @Entity //***  
  6. public class Teacher {  
  7.     private int id;   
  8.     private String name;      
  9.     private String title;  
  10.     //设置主键使用@Id  
  11.     @Id  //***  
  12.     public int getId() {  
  13.         return id;  
  14.     }  
  15.     public void setId(int id) {  
  16.         this.id = id;  
  17.     }  
  18.     public String getName() {  
  19.         return name;  
  20.     }  
  21.     public void setName(String name) {  
  22.         this.name = name;  
  23.     }  
  24.     public String getTitle() {  
  25.         return title;  
  26.     }  
  27.     public void setTitle(String title) {  
  28.         this.title = title;  
  29.     }}  

1、  在hibernate.cfg.xml中建立映射<maping class=…/>

<mapping class="com.wjt276.hibernate.model.Teacher"/>

注意:<mapping>标签中使用的是class属性,而不是resource属性,并且使用小数点(.)导航,而不是”/”

2、  参考文档进行测试


  1. public staticvoid main(String[] args) {         
  2.         Teacher t = new Teacher(); 
  3.         t.setId(1); 
  4.         t.setName("s1"); 
  5.         t.setTitle("中级"); 
  6.         //注此处并不是使用org.hibernate.cfg.Configuration来创建Configuration 
  7. //而使用org.hibernate.cfg.AnnotationConfiguration来创建Configuration,这样就可以使用Annotation功能 
  8.         Configuration cfg = new AnnotationConfiguration(); 
  9.          
  10.         SessionFactory sf = cfg.configure().buildSessionFactory(); 
  11.         Session session = sf.openSession(); 
  12.          
  13.         session.beginTransaction(); 
  14.         session.save(t); 
  15.         session.getTransaction().commit(); 
  16.          
  17.         session.close(); 
  18.         sf.close(); 
  19.     } 
  1. public static void main(String[] args) {          
  2.         Teacher t = new Teacher();  
  3.         t.setId(1);  
  4.         t.setName("s1");  
  5.         t.setTitle("中级");  
  6.         //注此处并不是使用org.hibernate.cfg.Configuration来创建Configuration  
  7. //而使用org.hibernate.cfg.AnnotationConfiguration来创建Configuration,这样就可以使用Annotation功能  
  8.         Configuration cfg = new AnnotationConfiguration();  
  9.           
  10.         SessionFactory sf = cfg.configure().buildSessionFactory();  
  11.         Session session = sf.openSession();  
  12.           
  13.         session.beginTransaction();  
  14.         session.save(t);  
  15.         session.getTransaction().commit();  
  16.           
  17.         session.close();  
  18.         sf.close();  
  19.     }  

1、  FAQ:@ 后不给提示

解决方法:windows→Proferences→seach “Content Assist”设置Auto-Activation如下:

Hibernate 学习参考_Hibernate_02

——————————————————————————————————————————————————————————————————————————————————————————————

第8课 什么是O/R Mapping

一、             定义:

ORM(ObjectRelational Mapping)---是一种为了解决面向对象与关系型数据库存在的互不匹配的现象的技术。简单说:ORM是通过使用描述对象和数据库之间映射的元数据,将Java程序中的对象自动持久化到关系数据中。本质上就是将数据从一种形式转换到另外一种形式。

Hibernate 学习参考_Hibernate_03

分层后,上层不需要知道下层是如何做了。

分层后,不可以循环依赖,一般是单向依赖。

一、             Hibernate的创始人:

Gavin King

二、             Hibernate做什么:

1、    就是将对象模型(实体类)的东西存入关系模型中,

2、    实体中类对应关系型库中的一个表,

3、    实体类中的一个属性会对应关系型数据库表中的一个列

4、    实体类的一个实例会对应关系型数据库表中的一条记录。

%%将对象数据保存到数据库、将数据库数据读入到对象中%%

OOA---面向对象的分析、面向对象的设计

OOD---设计对象化

OOP---面向对象的开发

阻抗不匹配---例JAVA类中有继承关系,但关系型数据库中不存在这个概念这就是阻抗不匹配。Hibernate可以解决这个问题

三、             Hibernate存在的原因:

1、  解决阻抗不匹配的问题;

2、  目前不存在完整的面向对象的数据库(目前都是关系型数据库);

3、  JDBC操作数据库很繁琐

4、  SQL语句编写并不是面向对象

5、  可以在对象和关系表之间建立关联来简化编程

6、  O/RMapping简化编程

7、  O/RMapping跨越数据库平台

8、  hibernate_0200_OR_Mapping_Simulation

四、             Hibernate的优缺点:

1、  不需要编写的SQL语句(不需要编辑JDBC),只需要操作相应的对象就可以了,就可以能够存储、更新、删除、加载对象,可以提高生产效;

2、  因为使用Hibernate只需要操作对象就可以了,所以我们的开发更对象化了;

3、  使用Hibernate,移植性好(只要使用Hibernate标准开发,更换数据库时,只需要配置相应的配置文件就可以了,不需要做其它任务的操作);

4、  Hibernate实现了透明持久化:当保存一个对象时,这个对象不需要继承Hibernate中的任何类、实现任何接口,只是个纯粹的单纯对象—称为POJO对象(最纯粹的对象—这个对象没有继承第三方框架的任何类和实现它的任何接口)

5、  Hibernate是一个没有侵入性的框架,没有侵入性的框架我们一般称为轻量级框架

6、  Hibernate代码测试方便。

五、             Hibernate使用范围:

1.       针对某一个对象,简单的将它加载、编辑、修改,且修改只是对单个对象(而不是批量的进行修改),这种情况比较适用;

2.       对象之间有着很清晰的关系(例:多个用户属于一个组(多对一)、一个组有多个用户(一对多));

3.       聚集性操作:批量性添加、修改时,不适合使用Hibernate(O/映射框架都不适合使用);

4.       要求使用数据库中特定的功能时不适合使用,因为Hibernate不使用SQL语句;

第9课 Hibernate的重点学习:Hibernate的对象关系映射

一、对象---关系映射模式

l         属性映射;

l         类映射:

l         关联映射:

n         一对一;

n         一对多;

n         多对多。

二、常用的O/R映射框架:

1、  Hibernate

2、  ApacheOJB

3、  JDO(是SUN提出的一套标准—Java数据对象)

4、  Toplink(Orocle公司的)

5、  EJB(2.0X中有CMP;3.0X提出了一套“Java持久化API”---JPA)

6、  IBatis(非常的轻量级,对JDBC做了一个非常非常轻量级的包装,严格说不是O/R映射框架,而是基于SQL的映射(提供了一套配置文件,把SQL语句配置到文件中,再配置一个对象进去,只要访问配置文件时,就可得到对象))

7、  JAP(是SUN公司的一套标准)

a)        意愿统一天下

第10课 模拟Hibernate原理(OR模拟)

         我们使用一个项目来完成

         功能:有一个配置文件,文件中完成表名与类名对象,字段与类属性对应起来。

测试驱动开发

一、                      项目名称

         hibernate_0200_OR_Mapping_Simulation

二、             原代码

  1. Test类: 
  2.     public staticvoid main(String[] args) throws Exception{ 
  3.          
  4.         Student s = new Student(); 
  5.         s.setId(10); 
  6.         s.setName("s1"); 
  7.         s.setAge(1); 
  8.          
  9.         Session session = new Session();//此Session是我们自己定义的Session 
  10.          
  11.         session.save(s); 
  1. Test类:  
  2.     public static void main(String[] args) throws Exception{  
  3.           
  4.         Student s = new Student();  
  5.         s.setId(10);  
  6.         s.setName("s1");  
  7.         s.setAge(1);  
  8.           
  9.         Session session = new Session();//此Session是我们自己定义的Session  
  10.           
  11.         session.save(s);  
  12. }  

  1. Session类 
  2. import java.lang.reflect.Method; 
  3. import java.sql.Connection; 
  4. import java.sql.DriverManager; 
  5. import java.sql.PreparedStatement; 
  6. import java.util.HashMap; 
  7. import java.util.Map; 
  8. import com.wjt276.hibernate.model.Student; 
  9. public class Session { 
  10.     String tableName = "_Student"
  11.     Map<String,String> cfs = new HashMap<String,String>(); 
  12.     String[] methodNames;//用于存入实体类中的get方法数组 
  13.     public Session(){ 
  14.         cfs.put("_id", "id"); 
  15.         cfs.put("_name", "name"); 
  16.         cfs.put("_age", "age"); 
  17.         methodNames = new String[cfs.size()]; 
  18.     } 
  19.     public void save(Student s)throws Exception{ 
  20.         String sql = createSQL();//创建SQL串 
  21.         Class.forName("com.mysql.jdbc.Driver"); 
  22.         Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/hibernate","root","root"); 
  23.         PreparedStatement ps = conn.prepareStatement(sql); 
  24.         // 
  25.         for(int i =0; i < methodNames.length; i++){ 
  26.             Method m = s.getClass().getMethod(methodNames[i]);//返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法 
  27.             Class r = m.getReturnType();//返回一个 Class 对象,该对象描述了此 Method 对象所表示的方法的正式返回类型 
  28.             if(r.getName().equals("java.lang.String")) { 
  29.                 //对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。 
  30.                 //个别参数被自动解包,以便与基本形参相匹配,基本参数和引用参数都随需服从方法调用转换 
  31.                 String returnValue = (String)m.invoke(s); 
  32.                 ps.setString(i + 1, returnValue); 
  33.             } 
  34.             if(r.getName().equals("int")) { 
  35.                 Integer returnValue = (Integer)m.invoke(s); 
  36.                 ps.setInt(i + 1, returnValue); 
  37.             } 
  38.             if(r.getName().equals("java.lang.String")) { 
  39.                 String returnValue = (String)m.invoke(s); 
  40.                 ps.setString(i + 1, returnValue); 
  41.             } 
  42.             System.out.println(m.getName() + "|" + r.getName()); 
  43.         }        
  44.         ps.executeUpdate(); 
  45.         ps.close(); 
  46.         conn.close(); 
  47.     } 
  48.     private String createSQL() { 
  49.         String str1 = ""
  50.         int index = 0
  51.          
  52.         for(String s : cfs.keySet()){ 
  53.             String v = cfs.get(s);//取出实体类成员属性 
  54.             v = Character.toUpperCase(v.charAt(0)) + v.substring(1);//将成员属性第一个字符大写 
  55.             methodNames[index] = "get" + v;//拼实体类成员属性的getter方法 
  56.             str1 += s + ",";//根据表中字段名拼成字段串 
  57.             index ++; 
  58.         } 
  59.         str1 = str1.substring(0,str1.length() -1); 
  60.         String str2 = ""
  61.         //根据表中字段数,拼成?串 
  62.         for (int i =0; i < cfs.size(); i++){    str2 += "?,";} 
  63.         str2 = str2.substring(0,str2.length() -1); 
  64.         String sql = "insert into " + tableName +"(" + str1 + ")" +" values (" + str2 + ")"
  65.         System.out.println(sql); 
  66.         return sql; 
  67.     }} 
  1. Session类  
  2. import java.lang.reflect.Method;  
  3. import java.sql.Connection;  
  4. import java.sql.DriverManager;  
  5. import java.sql.PreparedStatement;  
  6. import java.util.HashMap;  
  7. import java.util.Map;  
  8. import com.wjt276.hibernate.model.Student;  
  9. public class Session {  
  10.     String tableName = "_Student";  
  11.     Map<String,String> cfs = new HashMap<String,String>();  
  12.     String[] methodNames;//用于存入实体类中的get方法数组  
  13.     public Session(){  
  14.         cfs.put("_id""id");  
  15.         cfs.put("_name""name");  
  16.         cfs.put("_age""age");  
  17.         methodNames = new String[cfs.size()];  
  18.     }  
  19.     public void save(Student s) throws Exception{  
  20.         String sql = createSQL();//创建SQL串  
  21.         Class.forName("com.mysql.jdbc.Driver");  
  22.         Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/hibernate","root","root");  
  23.         PreparedStatement ps = conn.prepareStatement(sql);  
  24.         //  
  25.         for(int i = 0; i < methodNames.length; i++){  
  26.             Method m = s.getClass().getMethod(methodNames[i]);//返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法  
  27.             Class r = m.getReturnType();//返回一个 Class 对象,该对象描述了此 Method 对象所表示的方法的正式返回类型  
  28.             if(r.getName().equals("java.lang.String")) {  
  29.                 //对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。  
  30.                 //个别参数被自动解包,以便与基本形参相匹配,基本参数和引用参数都随需服从方法调用转换  
  31.                 String returnValue = (String)m.invoke(s);  
  32.                 ps.setString(i + 1, returnValue);  
  33.             }  
  34.             if(r.getName().equals("int")) {  
  35.                 Integer returnValue = (Integer)m.invoke(s);  
  36.                 ps.setInt(i + 1, returnValue);  
  37.             }  
  38.             if(r.getName().equals("java.lang.String")) {  
  39.                 String returnValue = (String)m.invoke(s);  
  40.                 ps.setString(i + 1, returnValue);  
  41.             }  
  42.             System.out.println(m.getName() + "|" + r.getName());  
  43.         }         
  44.         ps.executeUpdate();  
  45.         ps.close();  
  46.         conn.close();  
  47.     }  
  48.     private String createSQL() {  
  49.         String str1 = "";  
  50.         int index = 0;  
  51.           
  52.         for(String s : cfs.keySet()){  
  53.             String v = cfs.get(s);//取出实体类成员属性  
  54.             v = Character.toUpperCase(v.charAt(0)) + v.substring(1);//将成员属性第一个字符大写  
  55.             methodNames[index] = "get" + v;//拼实体类成员属性的getter方法  
  56.             str1 += s + ",";//根据表中字段名拼成字段串  
  57.             index ++;  
  58.         }  
  59.         str1 = str1.substring(0,str1.length() -1);  
  60.         String str2 = "";  
  61.         //根据表中字段数,拼成?串  
  62.         for (int i = 0; i < cfs.size(); i++){    str2 += "?,";}  
  63.         str2 = str2.substring(0,str2.length() -1);  
  64.         String sql = "insert into " + tableName + "(" + str1 + ")" + " values (" + str2 + ")";  
  65.         System.out.println(sql);  
  66.         return sql;  
  67.     }}  
第11课 Hibernate基础配置

一、 提纲

1、  对应项目:hibernate_0300_BasicConfiguration

2、  介绍MYSQL的图形化客户端

3、  Hibernate.cfg.xml:hbm2ddl.auto

a) 先建表还是先建实体类

4、  搭建日志环境并配置显示DDL语句

5、  搭建Junit环境

a) 需要注意Junit的Bug

6、  ehibernate.cfg.xml: show_sql

7、  hibernate.cfg.xml:format_sql

8、  表名和类名不同,对表名进行配置

a) Annotation:@Table

b) Xml:自己查询

9、  字段名和属性相同

a) 默认为@Basic

b) Xml中不用写column

10、    字段名和属性名不同

a) Annotation:@Column

b) Xml:自己查询

11、    不需要psersistence的字段

a) Annotation:@Transient

b) Xml:不写

12、    映射日期与时间类型,指定时间精度

a) Annotation:@Temporal

b) Xml:指定type

13、    映射枚举类型

a) Annotation:@Enumerated

b) Xml:麻烦

14、    字段映射的位置(field或者get方法)

a) Best practice:保持field和get/set方法的一致

15、    @Lob

16、    课外:CLOB BLOB类型的数据存取

17、    课外:Hibernate自定义数据类型

18、    Hibernate类型

二、 介绍MYSQL的图形化客户端

这样的软件网络很多,主要自己动手做

三、 Hibernate.cfg.xml:hbm2ddl.auto

在SessionFactory创建时,自动检查数据库结构,或者将数据库schema的DDL导出到数据库. 使用 create-drop时,在显式关闭SessionFactory时,将drop掉数据库schema.取值 validate | update | create | create-drop

四、 搭建日志环境并配置显示DDL语句

我们使用slf接口,然后使用log4j的实现。

1、  首先引入log4j的jar包(log4j-1.2.14.jar),

2、  然后再引入slf4j实现LOG4J和适配器jar包(slf4j-log4j12-1.5.8.jar)

3、  最后创建log4j的配置文件(log4j.properties),并加以修改,只要保留

log4j.logger.org.hibernate.tool.hbm2ddl=debug

五、 搭建Junit环境

1、首先引入Junit 类库 jar包 (junit-4.8.1.jar)

2、在项目名上右键→new→Source Folder→输入名称→finish

3、注意,你对哪个包进行测试,你就在测试下建立和那个包相同的包

4、建立测试类,需要在测试的方法前面加入”@Test”


  1. public class TeacherTest { 
  2.      
  3.     private static SessionFactory sf =null
  4.      
  5.     @BeforeClass//表示Junit此类被加载到内存中就执行这个方法 
  6.     public staticvoid beforClass(){ 
  7.         sf = new AnnotationConfiguration().configure().buildSessionFactory(); 
  8.     } 
  9.     @Test//表示下面的方法是测试用的。 
  10.     public void testTeacherSave(){ 
  11.         Teacher t = new Teacher(); 
  12.         t.setId(6); 
  13.         t.setName("s1"); 
  14.         t.setTitle("中级"); 
  15.         Session session = sf.openSession(); 
  16.         session.beginTransaction(); 
  17.         session.save(t); 
  18.         session.getTransaction().commit(); 
  19.          
  20.         session.close(); 
  21.     } 
  22.      
  23.     @AfterClass//Junit在类结果时,自动关闭 
  24.     public staticvoid afterClass(){ 
  25.         sf.close(); 
  26.     }} 
  1. public class TeacherTest {  
  2.       
  3.     private static SessionFactory sf = null;  
  4.       
  5.     @BeforeClass//表示Junit此类被加载到内存中就执行这个方法  
  6.     public static void beforClass(){  
  7.         sf = new AnnotationConfiguration().configure().buildSessionFactory();  
  8.     }  
  9.     @Test//表示下面的方法是测试用的。  
  10.     public void testTeacherSave(){  
  11.         Teacher t = new Teacher();  
  12.         t.setId(6);  
  13.         t.setName("s1");  
  14.         t.setTitle("中级");  
  15.         Session session = sf.openSession();  
  16.         session.beginTransaction();  
  17.         session.save(t);  
  18.         session.getTransaction().commit();  
  19.           
  20.         session.close();  
  21.     }  
  22.       
  23.     @AfterClass//Junit在类结果时,自动关闭  
  24.     public static void afterClass(){  
  25.         sf.close();  
  26.     }}  

六、 ehibernate.cfg.xml : show_sql

输出所有SQL语句到控制台. 有一个另外的选择是把org.hibernate.SQL这个log category设为debug。

取值: true | false

七、 hibernate.cfg.xml :format_sql

在log和console中打印出更漂亮的SQL。


取值: true | false

True样式:

  1. 16:32:39,750 DEBUG SchemaExport:377 -  
  2.     create table Teacher ( 
  3.         id integer not null
  4.         name varchar(255), 
  5.         title varchar(255), 
  6.         primary key (id) 
  1. 16:32:39,750 DEBUG SchemaExport:377 -   
  2.     create table Teacher (  
  3.         id integer not null,  
  4.         name varchar(255),  
  5.         title varchar(255),  
  6.         primary key (id)  
  7.  )  

False样式:

  1. 16:33:40,484 DEBUG SchemaExport:377 - create table Teacher (id integer not null, name varchar(255), title varchar(255), primary key (id)) 
  1. 16:33:40,484 DEBUG SchemaExport:377 - create table Teacher (id integer not null, name varchar(255), title varchar(255), primary key (id))  

八、 表名和类名不同,对表名进行配置


Annotation:使用 @Table(name=”tableName”) 进行注解

         例如:

  1.     /**
  2. * @Entity 表示下面的这个Teacher是一个实体类
  3. * @Table 表示映射到数据表中的表名,其中的name参数表示"表名称"
  4. * @Id 表示主键Id,一般放在getXXX前面
  5. */ 
  6. @Entity 
  7. @Table(name="_teacher"
  8. public class Teacher { 
  9. [……] 
  1.     /** 
  2.  * @Entity 表示下面的这个Teacher是一个实体类 
  3.  * @Table 表示映射到数据表中的表名,其中的name参数表示"表名称" 
  4.  * @Id 表示主键Id,一般放在getXXX前面 
  5.  */  
  6. @Entity  
  7. @Table(name="_teacher")  
  8. public class Teacher {  
  9. [……]  
  10. }  

Xml:

  1. <class name="Student" table="_student"
  1. <class name="Student" table="_student">  

九、 字段名和属性相同

Annotation:默认为@Basic

注意:如果在成员属性没有加入任何注解,则默认在前面加入了@Basic

Xml中不用写column

十、 字段名和属性名不同

Annotation:使用@Column(name=”columnName”)进行注解

         例如:

  1. /**
  2. * @Entity 表示下面的这个Teacher是一个实体类
  3. * @Table 表示映射到数据表中的表名,其中的name参数表示"表名称"
  4. * @Column 表示实体类成员属性映射数据表中的字段名,其中name参数指定一个新的字段名
  5. * @Id 表示主键Id
  6. */ 
  7. @Entity 
  8. @Table(name="_teacher"
  9. public class Teacher { 
  10.  
  11.     private int id;  
  12.     private String name;     
  13.     private String title; 
  14.  
  15.     //设置主键使用@Id 
  16.     @Id 
  17.     public int getId() { 
  18.         return id; 
  19.     } 
  20.  
  21.     @Column(name="_name")//字段名与属性不同时 
  22.     public String getName() { 
  23.         return name; 
  24.     } 
  25. …… 
  1. /** 
  2.  * @Entity 表示下面的这个Teacher是一个实体类 
  3.  * @Table 表示映射到数据表中的表名,其中的name参数表示"表名称" 
  4.  * @Column 表示实体类成员属性映射数据表中的字段名,其中name参数指定一个新的字段名 
  5.  * @Id 表示主键Id 
  6.  */  
  7. @Entity  
  8. @Table(name="_teacher")  
  9. public class Teacher {  
  10.   
  11.     private int id;   
  12.     private String name;      
  13.     private String title;  
  14.   
  15.     //设置主键使用@Id  
  16.     @Id  
  17.     public int getId() {  
  18.         return id;  
  19.     }  
  20.   
  21.     @Column(name="_name")//字段名与属性不同时  
  22.     public String getName() {  
  23.         return name;  
  24.     }  
  25. ……  

Xml:

  1. <property name="name" column="_name"/> 
  1. <property name="name" column="_name"/>  


十一、 不需要(持久化)psersistence的字段

就是不实体类的某个成员属性不需要存入数据库中

Annotation:使用@Transient 进行注解就可以了。

例如:

  1. @Transient 
  2.     public String getTitle() { 
  3.         return title; 
  1. @Transient  
  2.     public String getTitle() {  
  3.         return title;  
  4. }  

Xml:不写(就是不需要对这个成员属性进行映射)

十二、 映射日期与时间类型,指定时间精度

Annotation:使用@Temporal(value=TemporalType)来注解表示日期和时间的注解

            其中TemporalType有三个值:TemporalType.TIMESTAMP 表示yyyy-MM-dd HH:mm:ss

                                     TemporalType.DATE      表示yyyy-MM-dd

                                     TemporalType.TIME      表示HH:mm:ss


  1. @Temporal(value=TemporalType.DATE) 
  2.     public Date getBirthDate() { 
  3.         return birthDate; 
  1. @Temporal(value=TemporalType.DATE)  
  2.     public Date getBirthDate() {  
  3.         return birthDate;  
  4. }  

注意:当使用注解时,属性为value时,则这个属性名可以省略,例如:@Temporal(TemporalType)

Xml:使用type属性指定hibernate类型

  1. <property name="birthDate" type="date"/> 
  1. <property name="birthDate" type="date"/>  

    注意:hibernate日期时间类型有:date, time, timestamp,当然您也可以使用Java包装类

        

十二、 映射枚举类型

Annotation:使用@Enumerated(value=EnumType)来注解表示此成员属性为枚举映射到数据库

        其中EnumType有二个值:①EnumType.STRING  表示直接将枚举名称存入数据库

                             ②EnumType.ORDINAL 表示将枚举所对应的数值存入数据库

Xml:映射非常的麻烦,先要定义自定义类型,然后再使用这个定义的类型……

第12课 使用hibernate工具类将对象模型生成关系模型

(也就是实体类生成数据库中的表),完整代码如下:

package com.wjt276.hibernate;

import org.hibernate.cfg.AnnotationConfiguration;

import org.hibernate.cfg.Configuration;

import org.hibernate.tool.hbm2ddl.SchemaExport;

/**

* Hibernate工具<br/>

* 将对象模型生成关系模型(将对象生成数据库中的表)

* 把hbm映射文件(或Annotation注解)转换成DDL

* 生成数据表之前要求已经存在数据库

* 注:这个工具类建立好后,以后就不用建立了。以后直接Copy来用。

* @author wjt276

* @version 1.0 2009/10/16

*/

public classExportDB {

    public static voidmain(String[] args){

        /*org.hibernate.cfg.Configuration类的作用:

         *读取hibernate配置文件(hibernate.cfg.xml或hiberante.properties)的.

         *new Configuration()默认是读取hibernate.properties

         *所以使用newConfiguration().configure();来读取hibernate.cfg.xml配置文件

         */

        Configuration cfg = new AnnotationConfiguration().configure();

        /*org.hibernate.tool.hbm2ddl.SchemaExport工具类:

         *需要传入Configuration参数

         *此工具类可以将类导出生成数据库表

         */

        SchemaExport export = newSchemaExport(cfg);

        /** 开始导出

         *第一个参数:script是否打印DDL信息

         *第二个参数:export是否导出到数据库中生成表

         */

        export.create(true, true);

    }}

运行刚刚建立的ExportDB类中的main()方法,进行实际的导出类。

第13课 ID主键生成策略

一、 Xml方式

<id>标签必须配置在<class>标签内第一个位置。由一个字段构成主键,如果是复杂主键<composite-id>标签

被映射的类必须定义对应数据库表主键字段。大多数类有一个JavaBeans风格的属性, 为每一个实例包含唯一的标识。<id> 元素定义了该属性到数据库表主键字段的映射。

<id

       name="propertyName"                                          (1)

       type="typename"                                              (2)

       column="column_name"                                         (3)

       unsaved-value="null|any|none|undefined|id_value"             (4)

       access="field|property|ClassName"                            (5)

       node="element-name|@attribute-name|element/@attribute|.">

        <generatorclass="generatorClass"/>

</id>

(1) name (可选): 标识属性的名字(实体类的属性)。

(2) type (可选): 标识Hibernate类型的名字(省略则使用hibernate默认类型),也可以自己配置其它hbernate类型(integer, long, short, float,double, character, byte, boolean, yes_no, true_false)

(2) length(可选):当type为varchar时,设置字段长度

(3) column (可选 - 默认为属性名): 主键字段的名字(省略则取name为字段名)。

(4) unsaved-value (可选 - 默认为一个切合实际(sensible)的值): 一个特定的标识属性值,用来标志该实例是刚刚创建的,尚未保存。 这可以把这种实例和从以前的session中装载过(可能又做过修改--译者注) 但未再次持久化的实例区分开来。

(5) access (可选 - 默认为property): Hibernate用来访问属性值的策略。

如果 name属性不存在,会认为这个类没有标识属性。

unsaved-value 属性在Hibernate3中几乎不再需要。

还有一个另外的<composite-id>定义可以访问旧式的多主键数据。 我们强烈不建议使用这种方式。

<generator>元素(主键生成策略)

主键生成策略是必须配置

用来为该持久化类的实例生成唯一的标识。如果这个生成器实例需要某些配置值或者初始化参数, 用<param>元素来传递。

<id name="id"type="long" column="cat_id">

       <generator class="org.hibernate.id.TableHiLoGenerator">

               <param name="table">uid_table</param>

               <param name="column">next_hi_value_column</param>

       </generator>

</id>

所有的生成器都实现org.hibernate.id.IdentifierGenerator接口。 这是一个非常简单的接口;某些应用程序可以选择提供他们自己特定的实现。当然, Hibernate提供了很多内置的实现。下面是一些内置生成器的快捷名字:

increment

用于为long, short或者int类型生成 唯一标识。只有在没有其他进程往同一张表中插入数据时才能使用。 在集群下不要使用。

identity

对DB2,MySQL, MS SQL Server,Sybase和HypersonicSQL的内置标识字段提供支持。 返回的标识符是long, short 或者int类型的。 (数据库自增)

sequence

在DB2,PostgreSQL, Oracle, SAPDB, McKoi中使用序列(sequence), 而在Interbase中使用生成器(generator)。返回的标识符是long, short或者 int类型的。(数据库自增)

hilo

使用一个高/低位算法高效的生成long, short 或者 int类型的标识符。给定一个表和字段(默认分别是 hibernate_unique_key和next_hi)作为高位值的来源。 高/低位算法生成的标识符只在一个特定的数据库中是唯一的。

seqhilo

使用一个高/低位算法来高效的生成long, short 或者 int类型的标识符,给定一个数据库序列(sequence)的名字。

uuid

用一个128-bit的UUID算法生成字符串类型的标识符, 这在一个网络中是唯一的(使用了IP地址)。UUID被编码为一个32位16进制数字的字符串,它的生成是由hibernate生成,一般不会重复。

UUID包含:IP地址,JVM的启动时间(精确到1/4秒),系统时间和一个计数器值(在JVM中唯一)。 在Java代码中不可能获得MAC地址或者内存地址,所以这已经是我们在不使用JNI的前提下的能做的最好实现了

guid

在MS SQL Server 和 MySQL 中使用数据库生成的GUID字符串。

native

根据底层数据库的能力选择identity,sequence 或者hilo中的一个。(数据库自增)

assigned

让应用程序在save()之前为对象分配一个标示符。这是 <generator>元素没有指定时的默认生成策略。(如果是手动分配,则需要设置此配置)

select

通过数据库触发器选择一些唯一主键的行并返回主键值来分配一个主键。

foreign

使用另外一个相关联的对象的标识符。通常和<one-to-one>联合起来使用。

二、 annotateon方式

使用@GeneratedValue(strategy=GenerationType)注解可以定义该标识符的生成策略

Strategy有四个值:

①     、AUTO- 可以是identity column类型,或者sequence类型或者table类型,取决于不同的底层数据库.

相当于native

②     、TABLE- 使用表保存id值

③     、IDENTITY- identity column

④     、SEQUENCE- sequence

注意:auto是默认值,也就是说没有后的参数则表示为auto

1、AUTO默认

@Id

    @GeneratedValue

    public int getId() {

        return id;

}

@Id

@GeneratedValue(strategy=GenerationType.AUTO)

public int getId() {

return id;

}

1、  对于mysql,使用auto_increment

2、  对于oracle使用hibernate_sequence(名称固定)

2、IDENTITY

@Id

    @GeneratedValue(strategy=GenerationType.IDENTITY)

    public int getId() {

        return id;

}

对DB2,MySQL, MS SQL Server,Sybase和HypersonicSQL的内置标识字段提供支持。 返回的标识符是long, short 或者int类型的。 (数据库自增)

注意:此生成策略不支持Oracle

3、SEQUENCE

    @Id

    @GeneratedValue(strategy=GenerationType.SEQUENCE)

    public int getId() {

        return id;

    }

在DB2,PostgreSQL,Oracle, SAP DB, McKoi中使用序列(sequence), 而在Interbase中使用生成器(generator)。返回的标识符是long, short或者 int类型的。(数据库自增)

注意:此生成策略不支持MySQL

4、为Oracle指定定义的Sequence

    a)、首先需要在实体类前面申明一个Sequence如下:

        方法:@SequenceGenerator(name="SEQ_Name",sequenceName="SEQ_DB_Name")

            参数注意:SEQ_Name:表示为申明的这个Sequence指定一个名称,以便使用

                    SEQ_DB_Name:表示为数据库中的Sequence指定一个名称。

                    两个参数的名称可以一样。

@Entity

@SequenceGenerator(name="teacherSEQ",sequenceName="teacherSEQ_DB")

public class Teacher {

……

}

b)、然后使用@GeneratedValue注解

    方法:@GeneratedValue(strategy=GenerationType.SEQUENCE,generator="SEQ_Name")

    参数:strategy:固定为GenerationType.SEQUENCE

         Generator:在实体类前面申明的sequnce的名称

@Entity

@SequenceGenerator(name="teacherSEQ",sequenceName="teacherSEQ_DB")

public class Teacher {

    private int id;

   

    @Id

    @GeneratedValue(strategy=GenerationType.SEQUENCE,generator="teacherSEQ")

    public int getId() {

        return id;

}}

5、TABLE - 使用表保存id值

  原理:就是在数据库中建立一个表,这个表包含两个字段,一个字段表示名称,另一个字段表示值。每次在添加数据时,使用第一个字段的名称,来取值作为添加数据的ID,然后再给这个值累加一个值再次存入数据库,以便下次取出使用。

Table主键生成策略的定义:

@javax.persistence.TableGenerator(

        name="Teacher_GEN",              //生成策略的名称

        table="GENERATOR_TABLE",     //在数据库生成表的名称

        pkColumnName = "pk_key",     //表中第一个字段的字段名 类型为varchar,key

        valueColumnName = "pk_value",    //表中第二个字段的字段名 int ,value

        pkColumnValue="teacher",     //这个策略中使用该记录的第一个字段的值(key值)

        initialValue = 1,                //这个策略中使用该记录的第二个字段的值(value值)初始化值

        allocationSize=1                  //每次使用数据后累加的数值

)

这样执行后,会在数据库建立一个表,语句如下:

create tableGENERATOR_TABLE (pk_key varchar(255),pk_value integer )

结构:

Hibernate 学习参考_Hibernate_04

并且表建立好后,就插入了一个记录,如下:

    Hibernate 学习参考_Hibernate_05

注:这条记录的pk_value值为2,是因为刚刚做例程序时,已经插入一条记录了。初始化时为1。

使用TABLE主键生成策略:

@Entity

@javax.persistence.TableGenerator(

        name="Teacher_GEN",              //生成策略的名称

        table="GENERATOR_TABLE",     //在数据库生成表的名称

        pkColumnName = "pk_key",     //表中第一个字段的字段名 类型为varchar,key

        valueColumnName = "pk_value",    //表中第二个字段的字段名 int ,value

        pkColumnValue="teacher",     //这个策略中使用该记录的第一个字段的值(key值)

        initialValue = 1,                //这个策略中使用该记录的第二个字段的值(value值)初始化值

        allocationSize=1                 //每次使用数据后累加的数值

)

public class Teacher {

    private int id;

    @Id

    @GeneratedValue(strategy=GenerationType.TABLE,generator="Teacher_GEN")

    public int getId() {

        return id;}}

注意:这样每次在添加Teacher记录时,都会先到GENERATOR_TABLE表取pk_key=teacher的记录后,使用pk_value值作为记录的主键。然后再给这个pk_value字段累加1,再存入到GENERATOR_TABLE表中,以便下次使用。

    这个表可以给无数的表作为主键表,只是添加一条记录而以(需要保证table、pkColumnName、valueColumnName三个属性值一样就可以了。),这个主键生成策略可以跨数据库平台。

三、 联合主键

     复合主键(联合主键):多个字段构成唯一性。

1、xml方式

a) 实例场景:核算期间

// 核算期间

public class FiscalYearPeriod {   

    private int fiscalYear; //核算年

    private int fiscalPeriod; //核算月  

    private Date beginDate; //开始日期  

    private Date endDate; //结束日期

    private String periodSts; //状态

    public int getFiscalYear() {

        return fiscalYear;

    }

    public void setFiscalYear(int fiscalYear) {

        this.fiscalYear = fiscalYear;

    }

    public int getFiscalPeriod(){  return fiscalPeriod;}

    public void setFiscalPeriod(int fiscalPeriod) {

        this.fiscalPeriod =fiscalPeriod;

    }

    public DategetBeginDate() {return beginDate;}

    public void setBeginDate(DatebeginDate) {  this.beginDate = beginDate; }

    public Date getEndDate(){return endDate;}

    public void setEndDate(DateendDate) {  this.endDate = endDate; }

    public StringgetPeriodSts() {  return periodSts;}

    public voidsetPeriodSts(String periodSts) {this.periodSts = periodSts;}

}

复合主键的映射,一般情况把主键相关的属性抽取出来单独放入一个类中。而这个类是有要求的:必需实现序列化接口(java.io.Serializable)(可以保存到磁盘上),为了确定这个复合主键类所对应对象的唯一性就会产生比较,对象比较就需要复写对象的hashCode()、equals()方法(复写方法如下图片),然后在类中引用这个复合主键类

Hibernate 学习参考_Hibernate_06

Hibernate 学习参考_Hibernate_07

b) 复合主键类:

复合主键必需实现java.io.Serializable接口

public class FiscalYearPeriodPKimplements java.io.Serializable {

  private int fiscalYear;//核算年 

  private int fiscalPeriod;//核算月

  public int getFiscalYear() {

      return fiscalYear;

  }

  public void setFiscalYear(int fiscalYear) {

      this.fiscalYear = fiscalYear;

  }

  public int getFiscalPeriod(){

      return fiscalPeriod;

  }

  public void setFiscalPeriod(int fiscalPeriod) {

      this.fiscalPeriod =fiscalPeriod;

  }  

  @Override

  public int hashCode() {

      final int prime = 31;

      int result = 1;

      result = prime* result + fiscalPeriod;

      result = prime* result + fiscalYear;

      return result;

  }

  @Override

  public boolean equals(Object obj){

      if (this == obj)

          return true;

      if (obj == null)

          return false;

      if (getClass() !=obj.getClass())

          return false;

      FiscalYearPeriodPKother = (FiscalYearPeriodPK) obj;

      if (fiscalPeriod != other.fiscalPeriod)

          return false;

      if (fiscalYear != other.fiscalYear)

          return false;

      return true;

  }}

c) 实体类:(中引用了复合主键类)

public class FiscalYearPeriod{

    private FiscalYearPeriodPK fiscalYearPeriodPK;//引用 复合主键类 

    private Date beginDate;//开始日期   

    private Date endDate;//结束日期 

    private String periodSts;//状态 

    public FiscalYearPeriodPK getFiscalYearPeriodPK() {

        return fiscalYearPeriodPK;

    }

    public void setFiscalYearPeriodPK(FiscalYearPeriodPKfiscalYearPeriodPK) {

        this.fiscalYearPeriodPK = fiscalYearPeriodPK;

    }

………………

d) FiscalYearPeriod.hbm.xml映射文件

<hibernate-mapping>

    <class name="com.bjsxt.hibernate.FiscalYearPeriod"table="t_fiscal_year_period">

        <composite-id name="fiscalYearPeriodPK">

            <key-property name="fiscalYear"/>

            <key-property name="fiscalPeriod"/>

        </composite-id>

        <property name="beginDate"/>

        <property name="endDate"/>

        <property name="periodSts"/>

    </class>

</hibernate-mapping>

e) 导出数据库输出SQL语句:

create table t_fiscalYearPeriod (fiscalYear integer not null, fiscalPeriodinteger not null, beginDate datetime, endDate datetime, periodSts varchar(255),primary key (fiscalYear, fiscalPeriod))//实体映射到数据就是两个字段构成复合主键

f) 数据库表结构:


Hibernate 学习参考_Hibernate_08

g) 复合主键关联映射数据存储:

    session =HibernateUtils.getSession();

    tx =session.beginTransaction();

    FiscalYearPeriod fiscalYearPeriod = new FiscalYearPeriod();

           

    //构造复合主键

    FiscalYearPeriodPK pk = new FiscalYearPeriodPK();

    pk.setFiscalYear(2009);

    pk.setFiscalPeriod(11);

           

    fiscalYearPeriod.setFiscalYearPeriodPK(pk);//为对象设置复合主键

    fiscalYearPeriod.setEndDate(new Date());

    fiscalYearPeriod.setBeginDate(new Date());

    fiscalYearPeriod.setPeriodSts("Y");

       

    session.save(fiscalYearPeriod);        

h) 执行输出SQL语句:

Hibernate: insert into t_fiscalYearPeriod (beginDate, endDate, periodSts,fiscalYear, fiscalPeriod) values (?, ?, ?, ?, ?)

注:如果再存入相同复合主键的记录,就会出错。

i) 数据的加载:

    数据加载非常简单,只是主键是一个对象而以,不是一个普通属性。

2、annotation方式

下面是定义组合主键的几种语法:

将组件类注解为@Embeddable,并将组件的属性注解为@Id

将组件的属性注解为@EmbeddedId

将类注解为@IdClass,并将该实体中所有属于主键的属性都注解为@Id

a) 将组件类注解为@Embeddable,并将组件的属性注解为@Id

组件类:

@Embeddable

public class TeacherPK implementsjava.io.Serializable{

    private int id;

    private String name;

    public int getId() {return id;  }

    public void setId(int id) {this.id = id;}

    public String getName() {   return name;}

    public void setName(Stringname) {  this.name = name;}

    @Override

    public boolean equals(Object o) {       ……}

    @Override

    public int hashCode() { return this.name.hashCode();    }

}

将组件类的属性注解为@Id,实体类中组件的引用

@Entity

public class Teacher {

  private TeacherPK pk;

  private String title;

 

  @Id

  public TeacherPK getPk(){

      return pk;

}}

b)  将组件的属性注解为@EmbeddedId

注意:只需要在实体类中表示复合主键属性前注解为@Entity,表示此主键是一个复合主键

       注意了,复合主键类不需要任何的注意。

@Entity

public class Teacher {

   private TeacherPK pk;

   private String title;  

   @EmbeddedId

   public TeacherPK getPk(){

       return pk;

   }}

c)  类注解为@IdClass,主键的属性都注解为@Id

需要将复合主键类建立好,不需要进行任何注解

在实体类中不需要进行复合主键类的引用

需要在实体类前面注解为@IdClass,并且指定一个value属性,值为复合主键类的class

需要在实体类中进行复合主键成员属性前面注解为@Id

如下:

@Entity

@IdClass(TeacherPK.class)

public class Teacher {

  //private TeacherPK pk;//不再需要

  private int id;

  private String name;   

  @Id

  public int getId() {return id;  }

  public void setId(int id) { this.id = id;   }

 

  @Id

  public String getName() {return name;}

  public void setName(Stringname) {this.name = name;

  }}

第14课 Hibernate核心开发接口(重点)

一、 Configuration(AnnotationConfiguration)

作用:进行配置信息的管理

目标:用来产生SessionFactory

可以在configure方法中指定hibernate配置文件,默认(不指定)时在classpath下加载hibernate.cfg.xml文件

加载默认的hibernate的配置文件

sessionFactory = newAnnotationConfiguration().configure().buildSessionFactory();

加载指定hibernate的配置文件

sessionFactory=newnnotationConfiguration().configure("hibernate.xml").buildSessionFactory();

只需要关注一个方法:buildSessionFactory();

二、 SessionFactory

作用:主要用于产生Session的工厂(数据库连接池)

当它产生一个Session时,会从数据库连接池取出一个连接,交给这个Session

Session session = sessionFactory.getCurrentSession();

         并且可以通过这个Session取出这个连接

         关注两个方法:

         getCurrentSession():表示当前环境没有Session时,则创建一个,否则不用创建

         openSession():     表示创建一个Session(3.0以后不常用),使用后需要关闭这个Session

         两方法的区别:

           ①、openSession永远是每次都打开一个新的Session,而getCurrentSession不是,是从上下文找、只有当前没有Session时,才创建一个新的Session

           ②、OpenSession需要手动close,getCurrentSession不需要手动close,事务提交自动close

           ③、getCurrentSession界定事务边界

上下文:

所指的上下文是指hibernate配置文件(hibernate.cfg.xml)中的“current_session_context_class”所指的值:(可取值:jta|thread|managed|custom.Class)

<property name="current_session_context_class">thread</property>

         常用的是:①、thread:是从上下文找、只有当前没有Session时,才创建一个新的Session,主要从数据界定事务

              ②、jta:主要从分布式界定事务,运行时需要Application Server来支持(Tomcat不支持)

             ③、managed:不常用

④、custom.Class:不常用

           

三、 Session

1、    管理一个数据库的任务单元

2、    save();

session.save(Object)

session的save方法是向数据库中保存一个对象,这个方法产生对象的三种状态

3、    delete() 

session.delete(Object)

   Object对象需要有ID

   对象删除后,对象状态为Transistent状态

4、    load()

格式: Session.load(Class arg0,Serializable arg1) throws HibernateException

    *arg0:需要加载对象的类,例如:User.class

    *arg1:查询条件(实现了序列化接口的对象):例"4028818a245fdd0301245fdd06380001"字符串已经实现了序列化接口。如果是数值类类型,则hibernate会自动使用包装类,例如 1

    * 此方法返回类型为Object,但返回的是代理对象。

    * 执行此方法时不会立即发出查询SQL语句。只有在使用对象时,它才发出查询SQL语句,加载对象。

    * 因为load方法实现了lazy(称为延迟加载、赖加载)

    * 延迟加载:只有真正使用这个对象的时候,才加载(才发出SQL语句)

    *hibernate延迟加载实现原理是代理方式。

    * 采用load()方法加载数据,如果数据库中没有相应的记录,则会抛出异常对象不找到(org.hibernate.ObjectNotFoundException)

    try {

        session =sf.openSession();

        session.beginTransaction();

       

         User user =(User)session.load(User.class,1);

           

        //只有在使用对象时,它才发出查询SQL语句,加载对象。

        System.out.println("user.name=" + user.getName());

           

        //因为此的user为persistent状态,所以数据库进行同步为龙哥。

        user.setName("发哥");

           

        session.getTransaction().commit();

    } catch (HibernateExceptione) {

        e.printStackTrace();

        session.getTransaction().rollback();

    } finally{

            if (session != null){

                if(session.isOpen()){

                    session.close();

                }

            }

    }

5、    Get()     

格式:Session.get(Class arg0,Serializable arg1)方法

* arg0:需要加载对象的类,例如:User.class

    * arg1:查询条件(实现了序列化接口的对象):

例"4028818a245fdd0301245fdd06380001"字符串已经实现了序列化接口。如果是基数类型,则hibernate会自动转换成包装类,如 1

    返回值: 此方法返回类型为Object,也就是对象,然后我们再强行转换为需要加载的对象就可以了。

            如果数据不存在,则返回null;

    注:执行此方法时立即发出查询SQL语句。加载User对象

加载数据库中存在的数据,代码如下:

try {

          session =sf.openSession();

          session.beginTransaction();

                 

           * 此方法返回类型为Object,也就是对象,然后我们再强行转换为需要加载的对象就可以了。

              如果数据不存在,则返回null

           * 执行此方法时立即发出查询SQL语句。加载User对象。

           */

          User user = (User)session.get(User.class, 1);

         

          //数据加载完后的状态为persistent状态。数据将与数据库同步。

          System.out.println("user.name=" + user.getName());

         

          //因为此的user为persistent状态,所以数据库进行同步为龙哥。

          user.setName("龙哥");

         

          session.getTransaction().commit();

      } catch(HibernateException e) {

          e.printStackTrace();

          session.getTransaction().rollback();

      } finally{

          if (session != null){

              if(session.isOpen()){

                  session.close();

              }

          }

6、    load()与get()区别

①、 不存在对应记录时表现不一样;

②、 load返回的是代理对象,等到真正使用对象的内容时才发出sql语句,这样就要求在第一次使用对象时,要求session处于open状态,否则出错

③、 get直接从数据库加载,不会延迟加载

get()和load()只根据主键查询,不能根据其它字段查询,如果想根据非主键查询,可以使用HQL

7、    update()

①        、用来更新detached对象,更新完成后转为为persistent状态(默认更新全部字段)

②        更新transient对象会报错(没有ID)

③        更新自己设定ID的transient对象可以(默认更新全部字段)

④        persistent状态的对象,只要设定字段不同的值,在session提交时,会自动更新(默认更新全部字段)

⑤        更新部分更新的字段(更改了哪个字段就更新哪个字段的内容)

a)       方法1:update/updatable属性

xml:设定<property>标签的update属性,设置在更新时是否参数更新

<property name="name" update="false"/>

注意:update可取值为true(默认):参与更新;false:更新时不参与更新

annotateon:设定@Column的updatable属性值,true参与更新,false:不参与更新

    @Column(updatable=false)

    public String getTitle(){return title;}

注意:此种方法很少用,因为它不灵活

b)       方法二:dynamic-update属性

注意:此方法目前只适合xml方式,JAP1.0annotation没有对应的

在实体类的映射文件中的<class>标签中,使用dynamic-update属性,true:表示修改了哪个字段就更新哪个字段,其它字段不更新,但要求是同一个session(不能跨session),如果跨了session同样会更新所有的字段内容。

<class name="com.bjsxt.Student" dynamic-update="true">

          代码:

@Test

  public void testUpdate5() {    

      Sessionsession = sessionFactory.getCurrentSession();

      session.beginTransaction();

      Student s =(Student)session.get(Student.class, 1);

      s.setName("zhangsan5");

      //提交时,会只更新name字段,因为此时的s为persistent状态

      session.getTransaction().commit();

      s.setName("z4");

      Sessionsession2 = sessionFactory.getCurrentSession();

      session2.beginTransaction();

      //更新时,会更新所有的字段,因为此时的s不是persistent状态

      session2.update(s);

      session2.getTransaction().commit(); }

           如果需要跨session实现更新修改的部分字段,需要使用session.merget()方法,合并字段内容

@Test

  public void testUpdate6() {    

      Sessionsession = sessionFactory.getCurrentSession();

      session.beginTransaction();

      Student s =(Student)session.get(Student.class, 1);

      s.setName("zhangsan6");

      session.getTransaction().commit();

      s.setName("z4");

      Sessionsession2 = sessionFactory.getCurrentSession();

      session2.beginTransaction();

      session2.merge(s);

      session2.getTransaction().commit()}

这样虽然可以实现部分字段更新,但这样会多出一条select语句,因为在字段数据合并时,需要比较字段内容是否已变化,就需要从数据库中取出这条记录进行比较

c)      使用HQL(EJBQL)面向对象的查询语言(建议)

  @Test

  public void testUpdate7() {    

      Sessionsession = sessionFactory.getCurrentSession();

      session.beginTransaction();

      Query q =session.createQuery(

"update Student s sets.name='z5' where s.id = 1");

      q.executeUpdate();

      session.getTransaction().commit();     

  }  

8、    saveOrUpdate()

在执行的时候hibernate会检查,如果对象在数据库中已经有对应的记录(是指主键),则会更新update,否则会添加数据save

9、    clear()

清除session缓存

无论是load还是get,都会首先查找缓存(一级缓存,也叫session级缓存),如果没有,才会去数据库查找,调用clear()方法可以强制清除session缓存

    Session session= sessionFactory.getCurrentSession();

    session.beginTransaction();

    Teacher t =(Teacher)session.load(Teacher.class, 1);

    System.out.println(t.getName());       

    session.clear();       

    Teacher t2 =(Teacher)session.load(Teacher.class, 1);

    System.out.println(t2.getName());

    session.getTransaction().commit();

注意:这样就会发出两条SELECT语句,如果把session.clear()去除,则只会发出一条SELECT语句,因为第二次load时,是使用session缓存中ID为1的对象,而这个对象已经在第一次load到缓存中 了。

10、 flush()

在hibernate中也存在flush这个功能,在默认的情况下session.commit()之前时,其实执行了一个flush命令。

Session.flush功能:

n         清理缓存;

n         执行sql(确定是执行SQL语句(确定生成update、insert、delete语句等),然后执行SQL语句。)

Session在什么情况下执行flush:

①         默认在事务提交时执行;

注意:flush时,可以自己设定,使用session.setFlushMode(FlushMode)来指定。

  session.setFlushMode(FlushMode);

FlushMode的枚举值:

l     FlushMode.ALWAYS:任务一条SQL语句,都会flush一次

l     FlushMode.AUTO  :自动flush(默认)

l     FlushMode.COMMIT: 只有在commit时才flush

l     FlushMode.MANUAL:手动flush。

l    FlushMode.NEVER :永远不flush  此选项在性能优化时可能用,比如session取数据为只读时用,这样就  

不需要与数据库同步了

        注意:设置flush模式时,需要在session开启事务之前设置。

②        可以显示的调用flush;

③        在执行查询前,如:iterate.

注:如果主键生成策略是uuid等不是由数据库生成的,则session.save()时并不会发出SQL语句,只有flush时才会发出SQL语句,但如果主键生成策略是native由数据库生成的,则session.save的同时就发出SQL语句。

11、 evict()

例如:session.evict(user)

作用:从session缓存(EntityEntries属性)中逐出该对象

但是与commit同时使用,会抛出异常

session = HibernateUtils.getSession();

tx = session.beginTransaction();

   

User1 user = new User1();

user.setName("李四");

user.setPassword("123");

user.setCreateTime(new Date());

user.setExpireTime(new Date());

           

//利用Hibernate将实体类对象保存到数据库中,因为user主键生成策略采用的是uuid,所以调用完成save后,只是将user纳入session的管理,不会发出insert语句,但是id已经生成,session中的existsInDatabase状态为false

session.save(user);

               

session.evict(user);//从session缓存(EntityEntries属性)中逐出该对象

//无法成功提交,因为hibernate在清理缓存时,在session的临时集合(insertions)中取出user对象进行insert操作后需要更新entityEntries属性中的existsInDatabase为true,而我们采用evict已经将user从session中逐出了,所以找不到相关数据,无法更新,抛出异常。

           

tx.commit();

解决在逐出session缓存中的对象不抛出异常的方法:

在session.evict()之前进行显示的调用session.flush()方法就可以了。

session.save(user);

               

//flush后hibernate会清理缓存,会将user对象保存到数据库中,将session中的insertions中的user对象清除,并且会设置session中的existsInDatabase状态为false

session.flush();

           

session.evict(user);//从session缓存(EntityEntries属性)中逐出该对象

           

//可以成功提交,因为hibernate在清理缓存时,在Session的insertions中集合中无法找到user对象所以不会发出insert语句,也不会更新session中existsInDatabase的状态。

tx.commit();


第15课 持久化对象的三种状态

一、 瞬时对象(Transient Object):

使用new操作符初始化的对象不是立刻就持久的。它们的状态是瞬时的,也就是说它们没有任何跟数据库表相关联的行为,只要应用不再引用这些对象(不再被任何其它对象所引用),它们的状态将会丢失,并由垃圾回收机制回收

二、 持久化对象(Persistent Object):

持久实例是任何具有数据库标识的实例,它有持久化管理器Session统一管理,持久实例是在事务中进行操作的----它们的状态在事务结束时同数据库进行同步。当事务提交时,通过执行SQL的INSERT、UPDATE和DELETE语句把内存中的状态同步到数据库中。

三、 离线对象(Detached Object):

Session关闭之后,持久化对象就变为离线对象。离线表示这个对象不能再与数据库保持同步,它们不再受hibernate管理。

Hibernate 学习参考_Hibernate_09


四、 三种状态的区分:

1、  有没有ID,(如果没有则是Transient状态)

2、  ID在数据库中有没有

3、  在内存里有没有(session缓存)

五、 总结:

Transient对象:随时可能被垃圾回收器回收(在数据库中没有于之对应的记录,应为是new初始化),而执行save()方法后,就变为Persistent对象(持久性对象),没有纳入session的管理

                            内存中一个对象,没有ID,缓存中也没有

Persistent对象:在数据库有存在的对应的记录,纳入session管理。在清理缓存(脏数据检查)的时候,会和数据库同步。

                            内存中有、缓存中有、数据库有(ID)

Detached对象:也可能被垃圾回收器回收掉(数据库中存在对应的记录,只是没有任何对象引用它是指session引用),注引状态经过Persistent状态,没有纳入session的管理

内存有、缓存没有、数据库有(ID)

第16课 关系映射(重点)

         注意:这里的关系是指:对象之间的关系并不是指数据库的关系,-----红色重要

         存在以下关系:

1、    一对一

u       单向(主键、外键)

u       双向(主键、外键)

2、    一对多

u       单向

u       双向

3、    多对一

u       单向

u       双向

4、    多对多

u       单向

u       双向

5、    集合映射

u       List

u       Set

u       Map

6、    继承关系(不重要)

u       单表

u       多表

u       一张主表、多张子表

7、    组件映射

u       @Embeddable

u       @Embedded

一、 一对一关联映射

  • 收藏
  • 评论
  • 举报