Mybatis配置之<properties>屬性配置元素解析

緊接著上篇博客《Mybatis的配置文件入門介紹》,我們開始對mybatis核心配置文件中的各個元素進行詳細的說明,在這篇文章中,我們首先來看下<properties>元素,這個元素從上篇文章中可以看到是最先被解析的,設置的屬性值將會被其他元素所使用。

我們先將之前的配置文件在這裡拷貝一份,以便對比觀察,如下所示:

<!DOCTYPE configuration    
    PUBLIC "-//mybatis.org//DTD Config 3.0//EN"    
    "http://mybatis.org/dtd/mybatis-3-config.dtd">  
<configuration>  
  
    <settings>  
        <setting name="logImpl" value="LOG4J" />  
    </settings>  
  
    <!-- 和spring整合後 environments配置將廢除 -->  
    <environments default="development">  
        <environment id="development">  
            <!-- 使用jdbc事務管理 -->  
            <transactionManager type="JDBC" />  
            <!-- 數據庫連接池 -->  
            <dataSource type="POOLED">  
                <property name="driver" value="com.mysql.jdbc.Driver" />  
                <property name="url"  
                    value="jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8" />  
                <property name="username" value="root" />  
                <property name="password" value="root" />  
            </dataSource>  
        </environment>  
    </environments>  
  
    <mappers>  
        <mapper resource="com\majing\learning\mybatis\dao\UserDaoMapper.xml" />  
    </mappers>    
</configuration>

屬性值有三種方式書寫,接下來我們一個一個的看

(1)通過<properties>元素裡面配置<property>元素

之前的配置文件中<dataSource>元素中設置瞭數據庫的驅動、連接字符串還有賬號密碼等信息,但是我們這裡不想這麼弄,通過設置<property>來進行設置,如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration  
 PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  
 "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration> 
 <properties>
  <property name="username" value="root"/>
  <property name="password" value="root"/>
  <property name="driver" value="com.mysql.jdbc.Driver"/>
  <property name="url" value="jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8"/>
 </properties>
 
 <settings>
  <setting name="logImpl" value="LOG4J" />
 </settings>
 
 <!-- 和spring整合後 environments配置將廢除 -->
 <environments default="development">
  <environment id="development">
   <!-- 使用jdbc事務管理 -->
   <transactionManager type="JDBC" />
   <!-- 數據庫連接池 -->
   <dataSource type="POOLED">
    <property name="driver" value="${driver}" />
    <property name="url" value="${url}" />
    <property name="username" value="${username}" />
    <property name="password" value="${password}" />
   </dataSource>
  </environment>
 </environments>
 
 <mappers>
  <mapper resource="com\majing\learning\mybatis\dao\UserDaoMapper.xml" />
 </mappers>
 
</configuration>

這樣,我們就在需要配置的地方統一到瞭<properties>元素中,便於統一管理。

(2)通過<properties>元素的resource屬性或者url屬性進行配置

這裡我們不用<property>標簽元素進行設置,而是使用屬性配置文件的方式。如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration  
 PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  
 "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
 
 <properties resource="mysql.properties">
 </properties>
 
 <settings>
  <setting name="logImpl" value="LOG4J" />
 </settings>
 
 <!-- 和spring整合後 environments配置將廢除 -->
 <environments default="development">
  <environment id="development">
   <!-- 使用jdbc事務管理 -->
   <transactionManager type="JDBC" />
   <!-- 數據庫連接池 -->
   <dataSource type="POOLED">
    <property name="driver" value="${driver}" />
    <property name="url" value="${url}" />
    <property name="username" value="${username}" />
    <property name="password" value="${password}" />
   </dataSource>
  </environment>
 </environments>
 
 <mappers>
  <mapper resource="com\majing\learning\mybatis\dao\UserDaoMapper.xml" />
 </mappers>
 
</configuration>

而<properties>標簽元素的resource屬性設置的mysql.properties(相對於根目錄的路徑)內容如下所示:

使用配置文件的方式,可以使得一次配置在多個地方重復使用,不需要在不同的項目中CTRL+C和CTRL+V瞭。

(3)通過在初始化的時候,以代碼的方式傳入Properties類實例

具體如下所示:

package com.majing.learning.mybatis; 
import java.io.IOException;
import java.util.Properties; 
import junit.framework.TestCase;
 
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
 
import com.majing.learning.mybatis.dao.UserDao;
import com.majing.learning.mybatis.entity.User; 
public class UserDaoTest1 extends TestCase{
	
	@Test
	public void testFindUserById(){
		SqlSession sqlSession = getSessionFactory().openSession(true);  
		UserDao userMapper = sqlSession.getMapper(UserDao.class);  
 
		User user = userMapper.findUserById(10);  
		System.out.println("記錄為:"+user);
	}
 
	// Mybatis 通過SqlSessionFactory獲取SqlSession, 然後才能通過SqlSession與數據庫進行交互
	private static SqlSessionFactory getSessionFactory() {
		SqlSessionFactory sessionFactory = null;
		String resource = "configuration.xml";
		try {
			sessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsReader(resource), buildInitProperties());
		} catch (IOException e) {
			e.printStackTrace();
		}
		return sessionFactory;
	}
	
	private static Properties buildInitProperties(){
		Properties properties = new Properties();
		properties.put("driver", "com.mysql.jdbc.Driver");
		properties.put("url", "jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8");
		properties.put("username", "root");
		properties.put("password", "root");
		return properties;
	}
}

從上可以看出,在創建SqlSessionFactory的時候,人為寫代碼傳入瞭一套屬性配置。

上面三種方式都可以實現相同的功能,那就是給mybatis初始化的時候設置一系列的屬性值以供使用。

但是這三者又有什麼區別呢?

通過查看源碼,一個直觀的感覺就是這三種配置是有優先級關系的且不同方式配置的配置項是可以並存的,優先級次序如下:第三種方式>第二種方式>第一種方式。

即如果三種方式都配置瞭同一個配置項,那麼優先級高的配置方式的配置值生效。

這主要還是因為mybats的源碼解析過程導致的。

下面我們看下具體的解析邏輯:

 private void propertiesElement(XNode context) throws Exception {
    if (context != null) {
      Properties defaults = context.getChildrenAsProperties();
      String resource = context.getStringAttribute("resource");
      String url = context.getStringAttribute("url");
      if (resource != null && url != null) {
        throw new BuilderException("The properties element cannot specify both a URL and a resource based property file reference.  Please specify one or the other.");
      }
      if (resource != null) {
        defaults.putAll(Resources.getResourceAsProperties(resource));
      } else if (url != null) {
        defaults.putAll(Resources.getUrlAsProperties(url));
      }
      Properties vars = configuration.getVariables();
      if (vars != null) {
        defaults.putAll(vars);
      }
      parser.setVariables(defaults);
      configuration.setVariables(defaults);
    }
  }

從代碼看,<properties>元素的resource屬性和url屬性是不能同時設置的,否則會報異常。

同時,解析的時候是先解析的<property>標簽元素,而後從resource或者url指定的配置文件開始讀取配置,如果之前有瞭相同的配置項則進行覆蓋,如果沒有則進行添加。

在這之後,開始判斷是否有第三種方式的屬性配置,如果有,則將相關配置添加到之前的屬性集合中,如果存在同名的配置也進行覆蓋。這樣的邏輯也是導致為什麼會有優先級的直接原因。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: