国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

深入淺出Mybatis系列(三)---配置詳解之properties與environments(mybatis源碼篇)

2019-11-11 07:00:26
字體:
來源:轉載
供稿:網友
深入淺出Mybatis系列(三)---配置詳解之PRoperties與environments(mybatis源碼篇)

  上篇文章《深入淺出Mybatis系列(二)---配置簡介(mybatis源碼篇)》我們通過對mybatis源碼的簡單分析,可看出,在mybatis配置文件中,在configuration根節點下面,可配置properties、typeAliases、plugins、objectFactory、objectWrapperFactory、settings、environments、databaseIdProvider、typeHandlers、mappers這些節點。那么本次,就會先介紹properties節點和environments節點。

  為了讓大家能夠更好地閱讀mybatis源碼,我先簡單的給大家示例一下properties的使用方法。

<configuration><!-- 方法一: 從外部指定properties配置文件, 除了使用resource屬性指定外,還可通過url屬性指定url    <properties resource="dbConfig.properties"></properties>   -->  <!-- 方法二: 直接配置為xml -->  <properties>      <property name="driver" value="com.MySQL.jdbc.Driver"/>      <property name="url" value="jdbc:mysql://localhost:3306/test1"/>      <property name="username" value="root"/>      <property name="passWord" value="root"/>  </properties>

那么,我要是 兩種方法都同時用了,那么哪種方法優先?

  當以上兩種方法都xml配置優先, 外部指定properties配置其次。至于為什么,接下來的源碼分析會提到,請留意一下。

  再看一下envirements元素節點的使用方法吧:

<environments default="development">    <environment id="development">      <transactionManager type="JDBC"/>      <dataSource type="POOLED">          <!--          如果上面沒有指定數據庫配置的properties文件,那么此處可以這樣直接配置         <property name="driver" value="com.mysql.jdbc.Driver"/>        <property name="url" value="jdbc:mysql://localhost:3306/test1"/>        <property name="username" value="root"/>        <property name="password" value="root"/>         -->                  <!-- 上面指定了數據庫配置文件, 配置文件里面也是對應的這四個屬性 -->         <property name="driver" value="${driver}"/>         <property name="url" value="${url}"/>         <property name="username" value="${username}"/>         <property name="password" value="${password}"/>               </dataSource>    </environment>        <!-- 我再指定一個environment -->    <environment id="test">      <transactionManager type="JDBC"/>      <dataSource type="POOLED">        <property name="driver" value="com.mysql.jdbc.Driver"/>        <!-- 與上面的url不一樣 -->        <property name="url" value="jdbc:mysql://localhost:3306/demo"/>        <property name="username" value="root"/>        <property name="password" value="root"/>      </dataSource>    </environment>      </environments>

environments元素節點可以配置多個environment子節點, 怎么理解呢? 

  假如我們系統的開發環境和正式環境所用的數據庫不一樣(這是肯定的), 那么可以設置兩個environment, 兩個id分別對應開發環境(dev)和正式環境(final),那么通過配置environments的default屬性就能選擇對應的environment了, 例如,我將environments的deault屬性的值配置為dev, 那么就會選擇dev的environment。 至于這個是怎么實現的, 下面源碼就會講。

  好啦,上面簡單給大家介紹了一下properties 和 environments 的配置, 接下來就正式開始看源碼了:

  上次我們說過mybatis 是通過XMLConfigBuilder這個類在解析mybatis配置文件的,那么本次就接著看看XMLConfigBuilder對于properties和environments的解析:

XMLConfigBuilder:

public class XMLConfigBuilder extends BaseBuilder {    private boolean parsed;    //xml解析器    private XPathParser parser;    private String environment;      //上次說到這個方法是在解析mybatis配置文件中能配置的元素節點    //今天首先要看的就是properties節點和environments節點    private void parseConfiguration(XNode root) {        try {          //解析properties元素          propertiesElement(root.evalNode("properties")); //issue #117 read properties first          typeAliasesElement(root.evalNode("typeAliases"));          pluginElement(root.evalNode("plugins"));          objectFactoryElement(root.evalNode("objectFactory"));          objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));          settingsElement(root.evalNode("settings"));          //解析environments元素          environmentsElement(root.evalNode("environments")); // read it after objectFactory and objectWrapperFactory issue #631          databaseIdProviderElement(root.evalNode("databaseIdProvider"));          typeHandlerElement(root.evalNode("typeHandlers"));          mapperElement(root.evalNode("mappers"));        } catch (Exception e) {          throw new BuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);        }    }          //下面就看看解析properties的具體方法    private void propertiesElement(XNode context) throws Exception {        if (context != null) {          //將子節點的 name 以及value屬性set進properties對象          //這兒可以注意一下順序,xml配置優先, 外部指定properties配置其次          Properties defaults = context.getChildrenasproperties();          //獲取properties節點上 resource屬性的值          String resource = context.getStringAttribute("resource");          //獲取properties節點上 url屬性的值, resource和url不能同時配置          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.");          }          //把解析出的properties文件set進Properties對象          if (resource != null) {            defaults.putAll(Resources.getResourceAsProperties(resource));          } else if (url != null) {            defaults.putAll(Resources.getUrlAsProperties(url));          }          //將configuration對象中已配置的Properties屬性與剛剛解析的融合          //configuration這個對象會裝載所解析mybatis配置文件的所有節點元素,以后也會頻頻提到這個對象          //既然configuration對象用有一系列的get/set方法, 那是否就標志著我們可以使用java代碼直接配置?           //答案是肯定的, 不過使用配置文件進行配置,優勢不言而喻          Properties vars = configuration.getVariables();          if (vars != null) {            defaults.putAll(vars);          }          //把裝有解析配置propertis對象set進解析器, 因為后面可能會用到          parser.setVariables(defaults);          //set進configuration對象          configuration.setVariables(defaults);        }    }        //下面再看看解析enviroments元素節點的方法    private void environmentsElement(XNode context) throws Exception {        if (context != null) {            if (environment == null) {                //解析environments節點的default屬性的值                //例如: <environments default="development">                environment = context.getStringAttribute("default");            }            //遞歸解析environments子節點            for (XNode child : context.getChildren()) {                //<environment id="development">, 只有enviroment節點有id屬性,那么這個屬性有何作用?                //environments 節點下可以擁有多個 environment子節點                //類似于這樣: <environments default="development"><environment id="development">...</environment><environment id="test">...</environments>                //意思就是我們可以對應多個環境,比如開發環境,測試環境等, 由environments的default屬性去選擇對應的enviroment                String id = child.getStringAttribute("id");                //isSpecial就是根據由environments的default屬性去選擇對應的enviroment                if (isSpecifiedEnvironment(id)) {                    //事務, mybatis有兩種:JDBC 和 MANAGED, 配置為JDBC則直接使用JDBC的事務,配置為MANAGED則是將事務托管給容器,                     TransactionFactory txFactory = transactionManagerElement(child.evalNode("transactionManager"));                    //enviroment節點下面就是dataSource節點了,解析dataSource節點(下面會貼出解析dataSource的具體方法)                    DataSourceFactory dsFactory = dataSourceElement(child.evalNode("dataSource"));                    DataSource dataSource = dsFactory.getDataSource();                    Environment.Builder environmentBuilder = new Environment.Builder(id)                          .transactionFactory(txFactory)                          .dataSource(dataSource);                    //老規矩,會將dataSource設置進configuration對象                    configuration.setEnvironment(environmentBuilder.build());                }            }        }    }        //下面看看dataSource的解析方法    private DataSourceFactory dataSourceElement(XNode context) throws Exception {        if (context != null) {            //dataSource的連接池            String type = context.getStringAttribute("type");            //子節點 name, value屬性set進一個properties對象            Properties props = context.getChildrenAsProperties();            //創建dataSourceFactory            DataSourceFactory factory = (DataSourceFactory) resolveClass(type).newInstance();            factory.setProperties(props);            return factory;        }        throw new BuilderException("Environment declaration requires a DataSourceFactory.");    } }

通過以上對mybatis源碼的解讀,相信大家對mybatis的配置又有了一個深入的認識。

  還有一個問題, 上面我們看到,在配置dataSource的時候使用了 ${driver} 這種表達式, 這種形式是怎么解析的?其實,是通過PropertyParser這個類解析:

PropertyParser:

/** * 這個類解析${}這種形式的表達式 */public class PropertyParser {  public static String parse(String string, Properties variables) {    VariableTokenHandler handler = new VariableTokenHandler(variables);    GenericTokenParser parser = new GenericTokenParser("${", "}", handler);    return parser.parse(string);  }  private static class VariableTokenHandler implements TokenHandler {    private Properties variables;    public VariableTokenHandler(Properties variables) {      this.variables = variables;    }    public String handleToken(String content) {      if (variables != null && variables.containsKey(content)) {        return variables.getProperty(content);      }      return "${" + content + "}";    }  }}好啦,以上就是對于properties 和 environments元素節點的分析,比較重要的都在對于源碼的注釋中標出。本次文章到此結束,接下來的文章會繼續分析其他節點的配置。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 三门县| 新巴尔虎右旗| 奈曼旗| 锡林郭勒盟| 顺平县| 乐业县| 东乌珠穆沁旗| 巧家县| 磐安县| 孝义市| 长丰县| 温州市| 黄山市| 河北省| 廊坊市| 通渭县| 连平县| 武强县| 始兴县| 蒙自县| 西华县| 延庆县| 柳州市| 保德县| 湘西| 苗栗市| 铁力市| 海林市| 阿鲁科尔沁旗| 吴川市| 高密市| 蓬莱市| 绥化市| 体育| 公安县| 合水县| 庐江县| 仁怀市| 西乡县| 岗巴县| 耿马|