- 1.首先get不支持lazy懒加载,load是支持的。
- 2.
Student student=session.load(Student.class,1);//不发送SQL,1是主键
student.getName();//此时发送SQL,而Student student=(Student)session.get(Student.class,1);//会直接发送sql
- 3.session.load()返回的是实体的代理类,利用反射机制,session.get()方法永远直接返回的是实力类。
- 4.如果通过主键查询该条记录不存在,get()方法返回的是null,load()方法返回的是ObjectNotFoundException异常。
- 5.load方法会利用Hibernate中的内部缓存和二级缓存,与Mybatis中的一级缓存和二级缓存类似,而get方法仅会在内部缓存查询数据,如果没有数据,则直接越过二级缓存执行SQL进行数据库查询。
- 拓展Hibernate的缓存机制
- 首先,Hibernate中维持了两级缓存。第一级缓存由Session实例维护,其中保持了Session当前所有关联实体的数据,也称为内部缓存。而第二级缓存则存在于SessionFactory层次,由当前所有由本SessionFactory构造的Session实例共享。出于性能考虑,避免无谓的数据库访问,Session在调用数据库查询功能之前,会先在缓存中进行查询。首先在第一级缓存中,通过实体类型和id进行查找,如果第一级缓存查找命中,且数据状态合法,则直接返回。
之后,Session会在当前“NonExists”记录中进行查找,如果“NonExists”记录中存在同样的查询条件,则返回null。“NonExists”记录了当前Session实例在之前所有查询操作中,未能查询到有效数据的查询条件(相当于一个查询黑名单列表)。如此一来,如果Session中一个无效的查询条件重复出现,即可迅速作出判断,从而获得最佳的性能表现。
对于load方法而言,如果内部缓存中未发现有效数据,则查询第二级缓存,如果第二级缓存命中,则返回。
如在缓存中未发现有效数据,则发起数据库查询操作(Select SQL),如经过查询未发现对应记录,则将此次查询的信息在“NonExists”中加以记录,并返回null。
根据映射配置和Select SQL得到的ResultSet,创建对应的数据对象。
将其数据对象纳入当前Session实体管理容器(一级缓存)。
执行Interceptor.onLoad方法(如果有对应的Interceptor)。
将数据对象纳入二级缓存。
如果数据对象实现了LifeCycle接口,则调用数据对象的onLoad方法。
返回数据对象。 - Mybtis中的一级缓存是针对的SqlSession,突出的Statement ID一致,且SQL查询条件一致,这些一致的条件是key,根据key直接将缓存中的数据返回。二级缓存是针对同一命名空间namespace中的MAPPER,应用业务场景是那些复杂的综合计算SQL,对那些实时性较高的sql是不适应的。