众所周知,抽象出来一个BaseDao的好处是巨大的,可以减少很多后续的DAO的工程量,只需让其他的dao继承这个baseDAO即可,然额问题来了,这个baseDAO,我们要用到HibernateTemplate。
我们把HibernateTemplate注入后,像寻常一样运行,子类像寻常一样调用父类,然后gg,报错空指针,查了半天最终发现,是hibernate没注进去,为啥呢,父类被调用了啊?
这里就要解释一个概念,注入的目标,是实例,意思就是当一个类被实例化出来的时候才会成功完成注入,而父类,始终停留在“类的概念上,它并没有被实例化出来,自然不会完成hibernateTamplate的注入过程,自然会报错。
解决办法是什么呢?
网上说了几个,但鄙人感觉都不能称之为解决办法,效果也与预期不符,试了好久,自己摸索出来一个,新建一个HibernateTemplateUtil类,定义如下
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.orm.hibernate5.HibernateTemplate;
public class HibernateTemplateUtil {
public static HibernateTemplate hibernateTemplate = (HibernateTemplate) new ClassPathXmlApplicationContext("beans.xml").getBean("hibernateTemplate");
}
然后,在baseDAO中,
import com.jeanLeung.shopOnline.service.HibernateTemplateUtil;
import org.hibernate.criterion.DetachedCriteria;
import org.hibernate.criterion.Projections;
import org.springframework.orm.hibernate5.HibernateTemplate;
import java.io.Serializable;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
public class BaseDao<T> {
private HibernateTemplate hibernateTemplate;
private Class clazz;
public BaseDao() {
hibernateTemplate = HibernateTemplateUtil.hibernateTemplate;
//反射:第一步获得class
Class clazz = this.getClass();
// 查看JDK的API
Type type = clazz.getGenericSuperclass(); //参数化类型
//得到这个type就是一个参数化的类型,将type强转成参数化的类型
ParameterizedType pType = (ParameterizedType)type;
//通过参数化类型获得实际类型参数:得到一个实际类型参数的数组
Type[] types = pType.getActualTypeArguments();
//只获得第一个实际类型参数即可
this.clazz = (Class)types[0];
}
}
注意我们的hibernateTemplate是直接从util类中获得的!亲测有效!
那为啥不直接在BaseDao的方法里写
hibernateTemplate = (HibernateTemplate) new ClassPathXmlApplicationContext("beans.xml").getBean("hibernateTemplate"); 呢?
事实是,无论在BaseDao的哪个地方定义new ClassPathXmlApplicationContext("beans.xml"),系统都会报错,说applicationContext已经被初始化过了,不能重复去初始化!
然而,用这个方法却可以解决这个问题,但是这原理。。。原谅我还是个菜逼,暂时还真是没头绪,如果有哪位大佬出来解释下,不胜感激!