Hibernate源码分析_hibernate 源码类关系-程序员宅基地

技术标签: Java  Hibernate  null  session  callback  insert  hibernate  sql  

这段时间本人利用空闲时间解读了一下Hibernate3的源码,饶有收获,愿与大家共享。

 

废话不多说,首先我们先对Hibernate有一个大致的印象

l  设计模式Hibernate=监听器,实际上是回调

l  Hibernate3支持拦截器

 

Hibernate配置方面的大原则:

l  bhn.xml文件所有配置都是描述本实体,除了cascade描述级联,即如何将本实体的操作(增删查改)传递给关联方。

l  inverse属性表示本实体是否拥有主动权,在一条cascade链路传递过程中,当出现inverse=false表示不再返回原cascade链路,而是从此处重新开始链路。inverse只有在非many方才有,也就是many-to-many或者one-to-many的set,List等。
下面是注明inverse=true与inverse=false的cascade链路的区别:

 

说明:若关联属性inverse=true,操作的结果将是校对A的属性所生成的sql;若关联属性inverse=false,结果将是丢弃先前A的操作,而转向对B的属性的校验所生成的sql;如果B中的属性也关联着inverse=false,则仍丢弃B继续新开启链路,直至没有关联方为inervse=false。不必担心,关联着的双方只有一方拥有inverse属性,所以不会一直传递下去。还有,丢弃了先前的操作不等于之前的对象操作无效,其效果相当于,原先的session.save(A),变成了session(A.B)而在B校对属性时总会找回A对象的。

 

测试用例:(暂不考虑inverse=false)

测试1:save()一个实体对象操作,预计insert发生在拥有外键方的表,拥有外键方的表是一对多中的多方。

结论1:如果实体对象的外键属性为null,表示不会产生关联,可直接生成sql;如果外键属性不为空,根据配置中的cascade去做关联。如果cascade=all则生成此表的insert和关联表update的sql,也就是说此时要求关联属性的主键id不能为null;如果cascade=save-update则生成此表的insert和关联表的insert/update的sql(关键属性的主键为null为insert,否则为update)。

见下图:

 

 

 

举例说明:

---------PO类:A中有B类型的关联属性

class A{

    private int id;

    private B b;

}

class B{

   private int id;

   private String str;

}

---------调用处关键代码:

   A a=new A();//待操作的实体对象

   B b=new B();//关联属性

   a.setB(b);//设置关联属性

   //----

   //a.setId(1);//save()操作不允许预定一个id,hibernate的Id必须使用配置中的方式生成

   //a.setB(null);//关联属性为null

   //----

   b.setId(1);/*关联属性的主键有值。只有cascade链在B对象校验为update操作才有效,也就是说A.bhn.xml中B

                    *的级联设为cascade=save-update。*/

   b.setId(null);/*关联属性的主键为null。支持cascade=all与cascade=save-update的操作,最终在B生成的

                       * 是insert操作*/

   //--------

   b.setStr("当前用于测试");//------数据项

   //-------操作

   session.save(a);

 

源码解读:(粗略)

Hibernate主要是事件监听模式(回调的一种实现),其核心类为Session类,Session类承载了CRUS操作和Commit操作。

补充知识点:回调的好处在于事件源对象eventSource和数据对象Object,被集中在监听器Listener里完成业务,集中的好处在于新写Listener就可以达到功能的扩展。Listener的处理方法的参数为,事件对象eventObject,事件对象包含事件源eventSource和数据对象,相当于Listener传的是两个参数,也就是说Listener得到了此数据模型中的所有数据,自然可以完成任何功能,其余部分在模型中可理解为仅是为了给Listener传参做准备。通常的运行流程是事件源先被调用方法,所以事件源的方法里完成了业务功能,所谓回调就是形式上还是调用了事件源的方法,但是业务功能的代码却在第三方的Listener类中完成,而事件源的方法里只是为了实现如何传参给Listener。这样就像是与传统编程相反由Listener去调用事件源。

 

 

案例1:Session的S查询操作

过程略:

小结:Hibernate的Select操作直接生成sql,当然通过了内存缓存才生成的sql。

 

案例2:Session调用CRU操作(非S操作,增删改)。

以Save()操作为例

步骤1. SessionImpl.save(obj); SessionImpl.save(null,obj);--从save(Object,Object)统一调用

步骤2. new SaveOrUpdateEvent(entityName, object, this)—创建并组装事件对象(用于Listener的参数)

步骤3. SessionImpl.fireSave(SaveOrUpdateEvent);--触发事件,即调用Listener的处理方法,目的在于传参步骤2中new出的SaveOrUpdateEvent事件对象。

代码如下:

private Serializable fireSave(SaveOrUpdateEvent event) {

     errorIfClosed();

     checkTransactionSynchStatus();

     SaveOrUpdateEventListener[] saveEventListener =listeners.getSaveEventListeners();

     for ( int i = 0; i < saveEventListener.length; i++ ) {

         saveEventListener[i].onSaveOrUpdate(event);

     }

     return event.getResultId();

  }
红色为关键代码,其中listenersSessionEventListeners属性。

EventListeners包含有一系列的监听器,而各种监听器以数组的形式允许有多个并且按顺序调用。本例中调用的监听器种类为saveOrUpdateEventListeners,处理方法为onSaveOrUpdate()方法。实现onSaveOrUpdate(event)的类是DefaultSaveOrUpdateEventListener。所以业务实现代码应该在DefaultSaveOrUpdateEventListener.onSaveOrUpdate()中

EventListeners类中的一系列Listener []属性:

    private LoadEventListener[] loadEventListeners = { new DefaultLoadEventListener() };

    private SaveOrUpdateEventListener[] saveOrUpdateEventListeners = { newDefaultSaveOrUpdateEventListener() };

    private MergeEventListener[] mergeEventListeners = { newDefaultMergeEventListener() };

    private PersistEventListener[] persistEventListeners = { newDefaultPersistEventListener() };

    private PersistEventListener[] persistOnFlushEventListeners = { newDefaultPersistOnFlushEventListener() };

    private ReplicateEventListener[] replicateEventListeners = { newDefaultReplicateEventListener() };

    private DeleteEventListener[] deleteEventListeners = { newDefaultDeleteEventListener() };

    private AutoFlushEventListener[] autoFlushEventListeners = { newDefaultAutoFlushEventListener() };

    private DirtyCheckEventListener[] dirtyCheckEventListeners = { newDefaultDirtyCheckEventListener() };

    private FlushEventListener[] flushEventListeners = { newDefaultFlushEventListener() };

    private EvictEventListener[] evictEventListeners = { newDefaultEvictEventListener() };

    private LockEventListener[] lockEventListeners = { new DefaultLockEventListener() };

    private RefreshEventListener[] refreshEventListeners = { newDefaultRefreshEventListener() };

    private FlushEntityEventListener[] flushEntityEventListeners = { newDefaultFlushEntityEventListener() };

    private InitializeCollectionEventListener[] initializeCollectionEventListeners =

           new DefaultInitializeCollectionEventListener() };

 

步骤4. DefaultSaveOrUpdateEventListener.onSaveOrUpdate()----业务功能代码。由于方法的实现涉及内容比较多,此处暂不作详细介绍。大致功能有为了补充齐全SaveOrUpdateEvent事件对象的其他属性,可见事件对象是记录Hibernate操作过程的容器。

 

步骤5.根据我们提交给Hibernate的Session的CRU指令操作,重复步骤1到步骤4多次直到最后到tran.commit()操作,tran是Session开启的Transction对象,默认JDBCTransction实现,根据hibernate.cfg.xml中配置确定了在new Configuration().configure()创立的,假定JDBCTransction.commit()的实现,具体代码如下:

public void commit() throws HibernateException {

       if (!begun) {

           throw new TransactionException("Transaction not successfully started");

       }

 

       log.debug("commit");

 

       if ( !transactionContext.isFlushModeNever() && callback ) {

           transactionContext.managedFlush(); //if an exception occurs during flush, user must call rollback()

       }

 

       notifyLocalSynchsBeforeTransactionCompletion();

       if ( callback ) {

           jdbcContext.beforeTransactionCompletion( this );

       }

 

       try {

           commitAndResetAutoCommit();

           log.debug("committed JDBC Connection");

           committed = true;

           if ( callback ) {

              jdbcContext.afterTransactionCompletion( truethis );

           }

           notifyLocalSynchsAfterTransactionCompletion( Status.STATUS_COMMITTED );

       }

       catch (SQLException e) {

           log.error("JDBC commit failed", e);

           commitFailed = true;

           if ( callback ) {

              jdbcContext.afterTransactionCompletion( falsethis );

           }

           notifyLocalSynchsAfterTransactionCompletion( Status.STATUS_UNKNOWN );

           throw new TransactionException("JDBC commit failed", e);

       }

       finally {

           closeIfRequired();

       }

    }

其中红色部分是生成sql的方法,这里暂不展开说明,生成sql依据的是上面步骤1-4所补充完整的事件对象。

绿色部分是Hibernate3对拦截器的支持,我们都知道Hibernate3比较之前的版本的一个重要的新特性就是支持拦截器,而这一特性就体现在此处。

 

 

 

小结:Session的非查询操作,只有到tran.commit()才生成sql,期间所有的CRU操作的结果都存放到对应的EventObject对象,对于保存C操作和更改U操作都存放在SaveOrUpdateEvent,删除操作R存放在DeleteEvent,而后commit()完成所有EventObject生成sql的规则。

各种操作与相应的流程如下面:

操作/流程

入口

创建事件对象

触发事件

事件处理方法

save

SessionImpl.save()

new SaveOrUpdateEvent()

fireSave()

DefaultSaveOrUpdateEventListener

update

SessionImpl.update()

new SaveOrUpdateEvent()

fireUpdate()

DefaultSaveOrUpdateEventListener

Delete

SessionImpl.delete()

new DeleteEvent()

fireDelete()

DefaultDeleteEventListener

补充:所有业务处理监听器都在org.hibernate.event.def包下

 

总结:

1.save操作:commit()时,数据库执行,并增加缓存中的对象。

2.delete操作:要求含有主键,commit()时,数据库执行,并删除缓存中的对象。

       如果删除执行记录数无影响,即没有找到要删除的记录,报错。

3.Select操作:直接查询数据库,更新缓存中的对象。

4.update操作:要求含有主键,commit()时,数据库执行,并更新缓存中的对象。

       如果更新执行记录数无影响,即没有找到要修改的记录,报错。

 

使用Hibernate时需要确定表的结构,只有确定了表的结构才能确定表的执行顺序,虽然Hibernate的目地是让我们编程只关心要操作对象,但是我们要明白维护(cascade链路方向)的方向是单向的,即使我们说Hibernate支持双向关联。

  • 维护方向是cascade和inverse配置出来的,Hibernate会遵循配置,去生成sql;
  • 另外所谓的双向关联只不过是维护方向单向查询出所关联的对象而后在内存中进行回填。如,User:Address=1:1,双向关联的目标是查询User对象,可得到User=User.getAddress().getUser()的结果,Hibernate的实现是单向查询得到User即关联属性Address对象,而后User.getAddress().setUser(User)进行回填。

 

SessionImpl==EventSource 事件源

SaveOrUpdateEvent==Event事件对象

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/wankaiming/article/details/7655048

智能推荐

稀疏编码的数学基础与理论分析-程序员宅基地

文章浏览阅读290次,点赞8次,收藏10次。1.背景介绍稀疏编码是一种用于处理稀疏数据的编码技术,其主要应用于信息传输、存储和处理等领域。稀疏数据是指数据中大部分元素为零或近似于零的数据,例如文本、图像、音频、视频等。稀疏编码的核心思想是将稀疏数据表示为非零元素和它们对应的位置信息,从而减少存储空间和计算复杂度。稀疏编码的研究起源于1990年代,随着大数据时代的到来,稀疏编码技术的应用范围和影响力不断扩大。目前,稀疏编码已经成为计算...

EasyGBS国标流媒体服务器GB28181国标方案安装使用文档-程序员宅基地

文章浏览阅读217次。EasyGBS - GB28181 国标方案安装使用文档下载安装包下载,正式使用需商业授权, 功能一致在线演示在线API架构图EasySIPCMSSIP 中心信令服务, 单节点, 自带一个 Redis Server, 随 EasySIPCMS 自启动, 不需要手动运行EasySIPSMSSIP 流媒体服务, 根..._easygbs-windows-2.6.0-23042316使用文档

【Web】记录巅峰极客2023 BabyURL题目复现——Jackson原生链_原生jackson 反序列化链子-程序员宅基地

文章浏览阅读1.2k次,点赞27次,收藏7次。2023巅峰极客 BabyURL之前AliyunCTF Bypassit I这题考查了这样一条链子:其实就是Jackson的原生反序列化利用今天复现的这题也是大同小异,一起来整一下。_原生jackson 反序列化链子

一文搞懂SpringCloud,详解干货,做好笔记_spring cloud-程序员宅基地

文章浏览阅读734次,点赞9次,收藏7次。微服务架构简单的说就是将单体应用进一步拆分,拆分成更小的服务,每个服务都是一个可以独立运行的项目。这么多小服务,如何管理他们?(服务治理 注册中心[服务注册 发现 剔除])这么多小服务,他们之间如何通讯?这么多小服务,客户端怎么访问他们?(网关)这么多小服务,一旦出现问题了,应该如何自处理?(容错)这么多小服务,一旦出现问题了,应该如何排错?(链路追踪)对于上面的问题,是任何一个微服务设计者都不能绕过去的,因此大部分的微服务产品都针对每一个问题提供了相应的组件来解决它们。_spring cloud

Js实现图片点击切换与轮播-程序员宅基地

文章浏览阅读5.9k次,点赞6次,收藏20次。Js实现图片点击切换与轮播图片点击切换<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title></title> <script type="text/ja..._点击图片进行轮播图切换

tensorflow-gpu版本安装教程(过程详细)_tensorflow gpu版本安装-程序员宅基地

文章浏览阅读10w+次,点赞245次,收藏1.5k次。在开始安装前,如果你的电脑装过tensorflow,请先把他们卸载干净,包括依赖的包(tensorflow-estimator、tensorboard、tensorflow、keras-applications、keras-preprocessing),不然后续安装了tensorflow-gpu可能会出现找不到cuda的问题。cuda、cudnn。..._tensorflow gpu版本安装

随便推点

物联网时代 权限滥用漏洞的攻击及防御-程序员宅基地

文章浏览阅读243次。0x00 简介权限滥用漏洞一般归类于逻辑问题,是指服务端功能开放过多或权限限制不严格,导致攻击者可以通过直接或间接调用的方式达到攻击效果。随着物联网时代的到来,这种漏洞已经屡见不鲜,各种漏洞组合利用也是千奇百怪、五花八门,这里总结漏洞是为了更好地应对和预防,如有不妥之处还请业内人士多多指教。0x01 背景2014年4月,在比特币飞涨的时代某网站曾经..._使用物联网漏洞的使用者

Visual Odometry and Depth Calculation--Epipolar Geometry--Direct Method--PnP_normalized plane coordinates-程序员宅基地

文章浏览阅读786次。A. Epipolar geometry and triangulationThe epipolar geometry mainly adopts the feature point method, such as SIFT, SURF and ORB, etc. to obtain the feature points corresponding to two frames of images. As shown in Figure 1, let the first image be ​ and th_normalized plane coordinates

开放信息抽取(OIE)系统(三)-- 第二代开放信息抽取系统(人工规则, rule-based, 先抽取关系)_语义角色增强的关系抽取-程序员宅基地

文章浏览阅读708次,点赞2次,收藏3次。开放信息抽取(OIE)系统(三)-- 第二代开放信息抽取系统(人工规则, rule-based, 先关系再实体)一.第二代开放信息抽取系统背景​ 第一代开放信息抽取系统(Open Information Extraction, OIE, learning-based, 自学习, 先抽取实体)通常抽取大量冗余信息,为了消除这些冗余信息,诞生了第二代开放信息抽取系统。二.第二代开放信息抽取系统历史第二代开放信息抽取系统着眼于解决第一代系统的三大问题: 大量非信息性提取(即省略关键信息的提取)、_语义角色增强的关系抽取

10个顶尖响应式HTML5网页_html欢迎页面-程序员宅基地

文章浏览阅读1.1w次,点赞6次,收藏51次。快速完成网页设计,10个顶尖响应式HTML5网页模板助你一臂之力为了寻找一个优质的网页模板,网页设计师和开发者往往可能会花上大半天的时间。不过幸运的是,现在的网页设计师和开发人员已经开始共享HTML5,Bootstrap和CSS3中的免费网页模板资源。鉴于网站模板的灵活性和强大的功能,现在广大设计师和开发者对html5网站的实际需求日益增长。为了造福大众,Mockplus的小伙伴整理了2018年最..._html欢迎页面

计算机二级 考试科目,2018全国计算机等级考试调整,一、二级都增加了考试科目...-程序员宅基地

文章浏览阅读282次。原标题:2018全国计算机等级考试调整,一、二级都增加了考试科目全国计算机等级考试将于9月15-17日举行。在备考的最后冲刺阶段,小编为大家整理了今年新公布的全国计算机等级考试调整方案,希望对备考的小伙伴有所帮助,快随小编往下看吧!从2018年3月开始,全国计算机等级考试实施2018版考试大纲,并按新体系开考各个考试级别。具体调整内容如下:一、考试级别及科目1.一级新增“网络安全素质教育”科目(代..._计算机二级增报科目什么意思

conan简单使用_apt install conan-程序员宅基地

文章浏览阅读240次。conan简单使用。_apt install conan