先看下面一个用eclipse自动生成的HibernateSessionFactory代码:
public class HibernateSessionFactory {
private static String CONFIG_FILE_LOCATION = "/hibernate.cfg.xml";//表示配置文件位置
private static final ThreadLocal threadLocal = new ThreadLocal();
private static Configuration configuration = new Configuration();
private static org.hibernate.SessionFactory sessionFactory;
private static String configFile = CONFIG_FILE_LOCATION;
static {
try {
configuration.configure(configFile);//读取配置文件
sessionFactory = configuration.buildSessionFactory();//创建sessionFactory
} catch (Exception e) {
System.err
.println("%%%% Error Creating SessionFactory %%%%");
e.printStackTrace();
}
}
private HibernateSessionFactory() {
}
public static Session getSession() throws HibernateException {
Session session = (Session) threadLocal.get();
if (session == null || !session.isOpen()) {
if (sessionFactory == null) {
rebuildSessionFactory();
}
session = (sessionFactory != null) ? sessionFactory.openSession()//创建新的session
: null;
threadLocal.set(session);
}
return session;
}
public static void rebuildSessionFactory() {//重新建立一个sessionFactory
try {
configuration.configure(configFile);
sessionFactory = configuration.buildSessionFactory();
} catch (Exception e) {
System.err
.println("%%%% Error Creating SessionFactory %%%%");
e.printStackTrace();
}
}
public static void closeSession() throws HibernateException {//关闭session
Session session = (Session) threadLocal.get();
threadLocal.set(null);
if (session != null) {
session.close();
}
}
public static org.hibernate.SessionFactory getSessionFactory() {
return sessionFactory;//返回sessionFactory
}
public static void setConfigFile(String configFile) {
HibernateSessionFactory.configFile = configFile;//设置配置文件的值
sessionFactory = null;
}
public static Configuration getConfiguration() {
return configuration;//返回配置文件的值
}
}
首先,很明显,sessionFactory肯定是创建session用的(要区别这个session和httpSession是不一样的),创建session的过程不外乎就是读取配置文件,根据配置文件创建一个sessionFactory,再用这个sessionFactory去创建一个session。下面对这代码做详细个人分析。。。
private static String CONFIG_FILE_LOCATION = "/hibernate.cfg.xml";
private static String configFile = CONFIG_FILE_LOCATION;
个人觉得private static String configFile = CONFIG_FILE_LOCATION;这句有多余之嫌,去掉之后进行测试,连接数据库进行增删改查完全没有问题,个人认为应该是为了后面修改配置文件属性或者值而创建configFile 变量,防止直接改CONFIG_FILE_LOCATION造成后面调用的不便。
private static final ThreadLocal threadLocal = new ThreadLocal();
这一句是很重要的,就是为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副本,而不会和其它线程的副本冲突。从线程的角度看,就好像每一个线程都完全拥有该变量。
注意,threadLocal 只是一个线程的局部变量,并不是一个线程。下面是度娘的一段资料:
ThreadLocal类接口很简单,只有4个方法,我们先来了解一下:
- void set(Object value)
设置当前线程的线程局部变量的值。
- public Object get()
该方法返回当前线程所对应的线程局部变量。
- public void remove()
将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。
- protected Object initialValue()
返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。
值得一提的是,在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。API方法也相应进行了调整,新版本的API方法分别是void set(T value)、T get()以及T initialValue()。
ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中有一个Map,用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。
public static Session getSession()主要用于判断线程副本里面是否有session存在或者session是否open(就是打开可以使用),否则就创建一个session。创建之前判断下sessionFactory是否存在,否则还要先创建一个sessionFactory。
个人认为后面closeSession()(这个只是关闭了线程变量里面的session),getSessionFactory() ,setConfigFile(String configFile),getConfiguration()暂时还是用不上的,大家可以根据需要进行一个选择。