我们在获取SessionFactory的时候,第一个语句就是:
如果用户想用XML配置hibernate,就需要编写如下代码:
Configuration configuration = new Configuration();
查看源码可知,Configuration类的公共构造方法只有一个,并且是无参数的:
public Configuration() {
this( new SettingsFactory() );
}
这个构造方法调用了一个受保护的构造方法:
protected Configuration(SettingsFactory settingsFactory) {
this.settingsFactory = settingsFactory;
reset();
}
受保护的构造方法里调用了私有方法reset,reset方法里有一句这样的代码:
properties = Environment.getProperties();
这句代码会从Hibernate环境里去获取hibernate配置,查看Environment类的方法getProperties:
public static Properties getProperties() {
Properties copy = new Properties();
copy.putAll(GLOBAL_PROPERTIES);
return copy;
}
这个方法得到的是全局配置属性的副本。再看看全局属性GLOBAL_PROPERTIES是如何初始化的。Environment的静态代码块里有如下代码:
InputStream stream = ConfigHelper.getResourceAsStream("/hibernate.properties");
try {
GLOBAL_PROPERTIES.load(stream);
log.info( "loaded properties from resource hibernate.properties: "
+ PropertiesHelper.maskOut(GLOBAL_PROPERTIES, PASS) );
}
代码跟踪到这里,我们可以知道hibernate首先一定会加载属性配置文件hibernate.properties,而且此文件的路径是写死的。
如果用户想用XML配置hibernate,就需要编写如下代码:
configuration.configure();
configure有五个重载方法:
看一下缺省的重载方法:
public Configuration configure() throws HibernateException {
configure( "/hibernate.cfg.xml" );
return this;
}
它调用的是另一个重载方法,加载的配置文件是写死的。这五个方法最后调用同一个方法doConfigure(Document doc):
protected Configuration doConfigure(Document doc) throws HibernateException {
Element sfNode = doc.getRootElement().element( "session-factory" );
String name = sfNode.attributeValue( "name" );
if ( name != null ) {
properties.setProperty( Environment.SESSION_FACTORY_NAME, name );
}
addProperties( sfNode );
parseSessionFactory( sfNode, name );
Element secNode = doc.getRootElement().element( "security" );
if ( secNode != null ) {
parseSecurity( secNode );
}
log.info( "Configured SessionFactory: " + name );
log.debug( "properties: " + properties );
return this;
}
看一下方法addProperties:
private void addProperties(Element parent) {
Iterator itr = parent.elementIterator( "property" );
while ( itr.hasNext() ) {
Element node = (Element) itr.next();
String name = node.attributeValue( "name" );
String value = node.getText().trim();
log.debug( name + "=" + value );
properties.setProperty( name, value );
if ( !name.startsWith( "hibernate" ) ) {
properties.setProperty( "hibernate." + name, value );
}
}
Environment.verifyProperties( properties );
}
可以看到,如果配置属性名不是以“hibernate”开头会自动加上“hibernate.”,这就是为什么,我们在用XML配置hibernate的时候,属性名“hibernate.”可以省去,
但是在使用属性文件或者编程方式配置时,“hibernate.”是不能省掉的。
hibernate加载配置的顺序是:properties配置——》XML配置或者编程方式配置。至于是先加载XML配置还是编程方式的配置,就要看用户的语句顺序了,但是有一点是确定的:后加载的配置会覆盖先加载的配置。
最后我们也可以看到,hibernate将基本的配置信息(不包括实体映射信息)保存到了configuration的properties成员属性中。
获取sessionFactory:
SessionFactory sessionFactory = configuration.buildSessionFactory();