解释 NH get与load的区别 http://pan.baidu.com/s/1mhaJEti

 

执行时机:

Get()执行的时候,就立即到数据库读取了。而Load()是延迟加载,到要用的时候,采取数据库读取,不用不读。

返回结果:

get()执行方法如果没取取得数据则显示对象为NULL

load()执行方法如没有取得数据会在对象内部报 DeptName = “load.DeptName”引发了“NHibernate.ObjectNotFoundException”类型的异常

检索执行机制上对比:

get方法和find方法都是直接从数据库中检索
而load方法的执行则比较复杂
1,首先查找session的persistent Context中是否有缓存,如果有则直接返回


2,如果判断是lazy=true/false分二种情况(是否是延迟加载)

2.1当lazy=true 延迟加载:则执行:

则需要建立代理对象,对象的initialized属性为false,target属性为null;

 在访问获得的代理对象的属性时,检索数据库,如果找到记录则把该记录的对象复制到代理对象的target
上,并将initialized=true,如果找不到就抛出异常 。

2.2 lazy=false不是延迟加载:直接访问数据库检索,查到记录返回,查不到抛出异常

 

static void Main(string[] args)
        {
           
            //创建工厂第一种方式
            ISessionFactory sessionFactory = CreateISessionFactory2();
            ISession session = sessionFactory.OpenSession();
            Console.WriteLine("Get之前");
            EmployeeInfo get = session.Get<EmployeeInfo>(6);  //马上执行查询
            Console.WriteLine("Get之后");
            Console.WriteLine("读取名字" + get.EmpName);   
            Console.WriteLine("=====================================");

            Console.WriteLine("Load之前");
            EmployeeInfo pLoad = session.Load<EmployeeInfo>(1); //汴意:没有查询
            Console.WriteLine("Load之后");
            Console.WriteLine("读取名字" + pLoad.EmpName); //汴意:使用时才查询

              
        }

 

 

 

 

可参考博客:

Hibernate中No row with the given identifier exists问题的原因及解决

            产生此问题的原因:

             有两张表,table1table2.产生此问题的原因就是table1里做了关联<one-to-one>或者<many-to-one unique="true">(特殊的多对一映射,实际就是一对一)来关联table2.当hibernate查找的时候,table2里的数据没有与table1相匹配的,这样就会报No row with the given identifier exists这个错.(一句话,就是数据的问题!)

           假如说,table1里有自身的主键id1,还有table2的主键id2,这两个字段.

          如果hibenrate设置的单项关联,即使table1中的id2为null值,table2中id2中有值,查询都不会出错.但是如果table1中的id2字段有值,但是这个值在table2中主键值里并没有,就会报上面的错!

         如果hibernate是双向关联,那么table1中的id2为null值,但是table2中如果有值,就会报这个错.这种情况目前的解决办法就是改成单项关联,或者把不对应的数据改对!

           这就是报这个错的原因了,知道原因了就相应的改就行了.或许还有些人迷惑hibernate关联都配好了,怎么会出现这样的错?其实这是编程的时候出现的问题,假如说我在添加信息的时候,页面传过来的struts的formbean到dao方法中需要封装成hibernate的po(就是hibenrate的bean),要是一个个po.get(form.set())实在太麻烦了,这样一般都会写个专门的方法来封装,遇到po.get(form.set())这种情况直接把struts的formbean对象传到此方法中封装就行了,假如我有个字段是创建人id,那么这个字段是永远不会改的,我在添加的时候还调用这个方法,这个专门封装的方法是有一些判断的,假如说我判断一下,如果遇到创建人id传过来为空值,我判断如果是空值,我把创建人id设为0,但是用户表中userid是主键从1开始自增的,那么这样数据就对应不上了,一查就会出这个错了.这个错在开发刚开始的时候经常发生,因为每个人的模块都是由相应的人独立开发完成以后再整合在一起的,每个人写单独那一块的时候往往会忽略这些,所以整合的时候这些问题往往就都一下子全冒出来了....整合很辛苦,tnnd!

 

 

hibernate的查询的比较
hibernate的查询有很多,Query,find,Criteria,get,load

query使用hsql语句,可以设置参数是常用的一种方式

criteria的方式,尽量避免了写hql语句,看起来更面向对象了。

find方式,这种方式已经被新的hibernate丢弃

get和load方式是根据id取得一个记录
下边详细说一下get和load的不同,因为有些时候为了对比也会把find加进来。

1,从返回结果上对比:
load方式检索不到的话会抛出org.hibernate.ObjectNotFoundException异常
get方法检索不到的话会返回null

2,从检索执行机制上对比:
get方法和find方法都是直接从数据库中检索
而load方法的执行则比较复杂
1,首先查找session的persistent Context中是否有缓存,如果有则直接返回
2,如果没有则判断是否是lazy,如果不是直接访问数据库检索,查到记录返回,查不到抛出异常
3,如果是lazy则需要建立代理对象,对象的initialized属性为false,target属性为null
4, 在访问获得的代理对象的属性时,检索数据库,如果找到记录则把该记录的对象复制到代理对象的target
上,并将initialized=true,如果找不到就抛出异常 。

posted on 2016-01-09 14:15  高达  阅读(546)  评论(0)    收藏  举报

导航