相信大家在使用hibernate(我这里使用的是3.6,因为我偏爱spring,但是hibernate4.x系以后和spring很多重复的地方,而且,spring也认为,hibernate4.x是一个很完善的框架了,spring3.x以后废弃掉了很多hibernate的支持类,所以我就不升级hibernate了)的时候都有这样的困惑,就是在配置hibernate映射文件的时候,如果有一对多(或者多对多)属性的时候,hinernate提供了一个优化机制,lazy属性,但是问题在于,配置文件是一次性配置,假如我项目中有的地方需要延迟加载,有的地方不需要延迟加载,这样的话就必须强制lazy=false,这样的话 这个优化机制就失去了他的意义.
经过一段时间的经验积累,终于相处了一套可行性比较强的方案:
项目中所有的lazy属性默认为true,不再设置为false.在项目中添加了一个commonService,提供了两个方法,一个是查询单个对象,一个是查询列表:
@Override public Object findById(Class<?> Objclass, Class<?> idClass,Object id,String[] fields) { Object o = dao.findById(Objclass, (Serializable) idClass.cast(id)); if(fields!=null){ for(String field:fields){ String targetMethod = "get"+upperFirstWord(field); Method[] methods = Objclass.getDeclaredMethods(); for (Method m : methods) { if(m.getName().equals(targetMethod)){ try { Hibernate.initialize(m.invoke(o)); break; } catch (Exception e) { e.printStackTrace(); } } } } } return o; }
Objclass 需要查询的对象的classidClass id字段的类型id id的值fields 需要初始化的字段
public List<?> find(Class<?> entityClazz, QueryParameterSetter paraSetter, Sorter sorter, Page page,String[] fields) { DetachedCriteria dc = DetachedCriteria.forClass(entityClazz); if (paraSetter != null) paraSetter.setParameters(dc); if (sorter != null && sorter.getOrder() != null) dc.addOrder(sorter.getOrder()); List<?> l = findPageListByCriteria(dc, page.getPageSize(), page.getRecordStartIndex()); lazyInitialize(entityClazz,l,fields); return l; } private void lazyInitialize(Class<?> entityClazz,List<?> l ,String[] fields){ if(fields!=null){ for(String field:fields){ String targetMethod = "get"+upperFirstWord(field); Method[] methods = entityClazz.getDeclaredMethods(); for (Method m : methods) { if(m.getName().equals(targetMethod)){ try { for(Object o:l){ Hibernate.initialize(m.invoke(o)); } } catch (Exception e) { e.printStackTrace(); } } } } } }
entityClazz 实体类型paraSetter 参数设置接口sorter 排序规则 page 分页对象
fields需要初始化的字段
这样的话在项目中,那些需要加载关联对象数据的时候只需要使用这两个方法就可以了,而且是可以指定属性的,不用全部加载,在性能上有了很大的提高.
这里只是一个代码片段,相信大部分程序员都能看懂,至于看不懂的伸手党程序员就自己琢磨吧