• 四、XML映射配置文件


    MyBatis的XML配置文件包含了影响MyBatis行为甚深的设置和属性信息。XML文档的高层级结构如下:
    ----configuration配置
    --------properties属性
    --------settings设置
    --------typeAliases 类型命名
    --------typeHandlers 类型处理器
    --------objectFactory对象工厂
    --------plugins插件
    --------environments环境
    ------------environment环境变量
    ----------------transactionManager 事务管理器
    ----------------dataSource数据源
    --------映射器

    properties

    这些是外部化的,可替代的属性,这些属性也可以配置在典型的Java属性配置文件中,或者通过properties元素的子元素来传递。例如:

    <properties resource="org/mybatis/example/config.properties">
      <property name="username" value="dev_user"/>
      <property name="password" value="F2Fa3!33TYyg"/>
    </properties>

    其中的属性就可以在整个配置文件中使用,使用可替换的属性来实现动态配置。比如:

    <dataSource type="POOLED">
      <property name="driver" value="${driver}"/>
      <property name="url" value="${url}"/>
      <property name="username" value="${username}"/>
      <property name="password" value="${password}"/>
    </dataSource>

    如果在这些地方,属性多于一个的话,MyBatis按照如下的顺序加载它们:

    • 在properties元素体内指定的属性首先被读取。
    • 从类路径下资源或properties元素的url属性中加载的属性第二被读取,它会覆盖已经存在的完全一样的属性。
    • 作为方法参数传递的属性最后被读取,它也会覆盖任一已经存在的完全一样的属性,这些属性可能是从properties元素体内和资源/url 属性中加载的。

     因此,最高优先级的属性是那些作为方法参数的,然后是资源/url 属性,最后是properties元素中指定的属性

    Settings

    这些是极其重要的调整,它们会修改MyBatis在运行时的行为方式。下面这个表格描述了设置信息,它们的含义和默认值。

    一个设置信息元素的示例,完全的配置如下所示:

    <settings>
    <setting name="cacheEnabled" value="true"/>
      <setting name="lazyLoadingEnabled" value="true"/>
      <setting name="multipleResultSetsEnabled" value="true"/>
      <setting name="useColumnLabel" value="true"/>
      <setting name="useGeneratedKeys" value="false"/>
      <setting name="enhancementEnabled" value="false"/>
      <setting name="defaultExecutorType" value="SIMPLE"/>
      <setting name="defaultStatementTimeout" value="25000"/>
    </settings>

    typeAliases

    类型别名是为Java类型命名一个短的名字。它只和XML配置有关,只用来减少类完全限定名的多余部分。例如:

    <typeAliases>
      <typeAlias alias="Author" type="domain.blog.Author"/>
      <typeAlias alias="Blog" type="domain.blog.Blog"/>
      <typeAlias alias="Comment" type="domain.blog.Comment"/>
      <typeAlias alias="Post" type="domain.blog.Post"/>
      <typeAlias alias="Section" type="domain.blog.Section"/>
      <typeAlias alias="Tag" type="domain.blog.Tag"/>
    </typeAliases>

    使用这个配置,“Blog”可以任意用来替代“domain.blog. Blog”所使用的地方。 对于普通的Java 类型,有许多内建的类型别名。它们都是大小写不敏感的,由于重载的名字,要注意原生类型的特殊处理。

    typeHandlers

    无论是MyBatis在预处理语句中设置一个参数,还是从结果集中取出一个值时,类型处理器被用来将获取的值以合适的方式转换成Java类型。下面这个表格描述了默认的类型处理器。

    你可以重写类型处理器或创建你自己的类型处理器来处理不支持的或非标准的类型。要这样做的话,简单实现TypeHandler接口(org.mybatis.type),然后映射新的类型处理器类到Java类型,还有可选的一个JDBC类型。例如:

     1 // Example Type Handler.java 
     2 public class ExampleTypeHandler implements TypeHandler { 
     3 public void setParameter(PreparedStatement ps, int i, Object 
     4 parameter,JdbcType jdbcType) throws SQLException { 
     5 ps.setString(i, (String) parame ter); 
     6 } 
     7 public Object getResult(ResultSet rs, String columnName) 
     8 throws SQLException { 
     9 return rs.getString(col umnName ); 
    10 } 
    11 public Object getResult(CallableStatement cs, int columnIndex) 
    12 throws SQLException { 
    13 return cs.getString(col umnInde x); 
    14 } 
    15 } 
    16   // MapperConfig.xml 
    17 <typeHandlers> 
    18 <typeHandler javaType="String" jdbcType="VARCHAR" 
    19 handler="org.mybatis.example.ExampleTypeHandler"/> 
    20 </typeHandlers> 

    使用这样的类型处理器将会覆盖已经存在的处理Java的String类型属性和VARCHAR参数及结果的类型处理器。要注意MyBatis不会审视数据库元信息来决定使用哪种类型,所以你必须在参数和结果映射中指定那是VARCHAR类型的字段,来绑定到正确的类型处理器上。这是因为MyBatis 直到语句被执行都不知道数据类型的这个现实导致的。

    objectFactory 

    MyBatis每次创建结果对象新的实例时,它使用一个ObjectFactory实例来完成。如果参数映射存在,默认的ObjectFactory不比使用默认构造方法或带参数的构造方法实例化目标类做的工作多。如果你想重写默认的ObjectFactory,你可以创建你自己的。比如:

     1 public class ExampleObjectFactory extends DefaultObjectFactory {
     2     public Object create(Class type) {
     3         return super.create(type);
     4     }
     5 
     6     public Object create(Class type, List<Class> constructorArgTypes,
     7             List<Object> constructorArgs) {
     8         return super.create(type, constructorArgTypes, constructorArgs);
     9     }
    10 
    11     public void setProperties(Properties properties) {
    12         super.setProperties(properties);
    13     }
    14 }
    15 // MapperConfig.xml 
    16 <objectFactory type="org.mybatis.example.ExampleObjectFactory"> 
    17     <property name="someProperty" value="100"/> 
    18 </objectFactory> 

    ObjectFactory接口很简单。它包含两个创建用的方法,一个是处理默认构造方法的,另外一个是处理带参数构造方法的。最终,setProperties 方法可以被用来配置ObjectFactory。在初始化你的ObjectFactory实例后,objectFactory 元素体中定义的属性会被传递给setProperties方法。

    plugins

    MyBatis允许你在某一点拦截已映射语句执行的调用。默认情况下,MyBatis允许使用插件来拦截方法调用:

    • Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)
    • ParameterHandler (getParameterObject, setParameters)
    • ResultSetHandler (handleResultSets, handleOutputParameters)
    • StatementHandler (prepare, parameterize, batch, update, query)

    这些类中方法的详情可以通过查看每个方法的签名来发现,而且它们的源代码在MyBatis的发行包中有。你应该理解你覆盖方法的行为,假设你所做的要比监视调用要多。如果你尝试修改或覆盖一个给定的方法,你可能会打破MyBatis的核心。这是低层次的类和方法,要谨慎使用插件。

    使用插件是它们提供的非常简单的力量。简单实现拦截器接口,要确定你想拦截的指定签名。

     1 @Intercepts({ @Signature(type = Executor.class, method = "update", args = {
     2         MappedStatement.class, Object.class }) })
     3 public class ExamplePlugin implements Interceptor {
     4     public Object intercept(Invocation invocation) throws Throwable {
     5         return invocation.proceed();
     6     }
     7 
     8     public Object plugin(Object target) {
     9         return Plugin.wrap(target, this);
    10     }
    11 
    12     public void setProperties(Properties properties) {
    13     }
    14 }

    // MapperConfig.xml
    <plugins>
    <plugin interceptor="org.mybatis.example.Example Plugin">
    <property name="someProperty" value="100"/>
    </plugin>
    </plugins>

    上面的插件将会拦截在Executor实例中所有的“update”方法调用,它也是负责低层次映射语句执行的内部对象。

    environments 

    MyBatis可以配置多种环境。这会帮助你将SQL映射应用于多种数据库之中。例如,你也许为开发要设置不同的配置,测试和生产环境。或者你可能有多种生产级数据库却共享相同的模式,所以你会想对不同数据库使用相同的SQL映射。这种用例是很多的。

    一个很重要的问题要记得:你可以配置多种环境,但你只能为每个SqlSessionFactory实例选择一个。

    所以,如果你想连接两个数据库,你需要创建两个SqlSessionFactory实例,每个数据库对应一个。而如果是三个数据库,你就需要三个实例,以此类推。记忆起来很简单:

    • 每个数据库对应一个SqlSessionFactor

    为了明确创建哪种环境,你可以将它作为可选的参数传递给SqlSessionFactoryBuilder。可以接受环境配置的两个方法签名是:

    SqlSessionFactory factory = sqlSessionFactoryBuilder.build(reader, environment);
    SqlSessionFactory factory = sqlSessionFactoryBuilder.build(reader, environment,properties);

    如果环境被忽略,那么默认环境将会被加载,如下进行: 

    SqlSessionFactory factory = sqlSessionFactoryBuilder.build(reader);
    SqlSessionFactory factory = sqlSessionFactoryBuilder.build(reader,properties);

    环境元素定义了如何配置环境。

    <environments default="development">
      <environment id="development">
        <transactionManager type="JDBC">
        <property name="..." value="..."/>
         </transactionManager>
        <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>

    注意这里的键:

    • 默认的环境ID(比如:default=”development”)。
    • 每个environment元素定义的环境ID(比如:id=”development”)。
    • 事务管理器的配置(比如:type=”JDBC”)。
    • 数据源的配置(比如:type=”POOLED”)。

    默认的环境和环境ID是自我解释的。你可以使用你喜欢的名称来命名,只要确定默认的要匹配其中之一。

    transactionManager

    在MyBatis中有两种事务管理器类型(也就是type=”[JDBC|MANAGED]”):

    • DBC – 这个配置直接简单使用了JDBC的提交和回滚设置。它依赖于从数据源得到的连接来管理事务范围。
    • MANAGED – 这个配置几乎没做什么。它从来不提交或回滚一个连接。而它会让容器来管理事务的整个生命周期(比如Spring或JEE应用服务器的上下文)。默认情况下它会关闭连接。然而一些容器并不希望这样,因此如果你需要从连接中停止它,将closeConnection属性设置为false。例如:

    <transactionManager type="MANAGED">
      <property name="closeConnection" value="false"/>
    </transactionManager>

    这两种事务管理器都不需要任何属性。然而它们都是类型别名,要替换使用它们,你需要放置将你自己的类的完全限定名或类型别名,它们引用了你对TransacFactory接口的实现类。

    public interface TransactionFactory {
      void setProperties(Properties props);
      Transaction newTransaction(Connection conn, boolean autoCommit);
    }

    任何在XML中配置的属性在实例化之后将会被传递给setProperties()方法。你的实现类需要创建一个事务接口的实现,这个接口也很简单:

    public interface Transaction {
      Connection getConnection();
      void commit() throws SQLException;
      void rollback() throws SQLException;
      void close() throws SQLException;
    }
    使用这两个接口,你可以完全自定义MyBatis对事务的处理。

    dataSsource

    dataSource元素使用基本的JDBC数据源接口来配置JDBC连接对象的资源。 

      许多MyBatis的应用程序将会按示例中的例子来配置数据源。然而它并不是必须的。要知道为了方便使用延迟加载,数据源才是必须的。

    有三种内建的数据源类型(也就是type=”???”):

    • UNPOOLED--这个数据源的实现是每次被请求时简单打开和关闭连接。它有一点慢,这是对简单应用程序的一个很好的选择,因为它不需要及时的可用连接。不同的数据库对这个的表现也是不一样的,所以对某些数据库来说配置数据源并不重要,这个配置也是闲置的。

      UNPOOLED类型的数据源仅仅用来配置以下4种属性:

    • driver – 这是JDBC驱动的Java类的完全限定名(如果你的驱动包含,它也不是数据源类)。
    • url – 这是数据库的JDBC URL地址。
    • username – 登录数据库的用户名。
    • password – 登录数据库的密码。
    • defaultTransactionIsolationLevel – 默认的连接事务隔离级别/.

    作为可选项,你可以传递数据库驱动的属性。要这样做,属性的前缀是以“driver.”开头的,例如:

    • driver.encoding=UTF8

    这样就会传递以值“UTF8”来传递属性“encoding ”,它是通过DriverManager.getConnection(url,driverProperties)方法传递给数据库驱动。

    • POOLED – 这是JDBC连接对象的数据源连接池的实现,用来避免创建新的连接实例时必要的初始连接和认证时间。这是一种当前Web应用程序用来快速响应请求很流行的方法。

    除了上述(UNPOOLED)的属性之外,还有很多属性可以用来配置POOLED数据源:

    • poolMaximumActiveConnections – 在任意时间存在的活动(也就是正在使用)连接的数量。默认值:10
    • poolMaximumIdleConnections – 任意时间存在的空闲连接数。
    • poolMaximumCheck outTime – 在被强制返回之前,池中连接被检查的时间。默认值:20000毫秒(也就是20秒)
    • poolTimeToWait – 这是给连接池一个打印日志状态机会的低层次设置,还有重新尝试获得连接,这些情况下往往需要很长时间(为了避免连接池没有配置时静默失败)。默认值:20000毫秒(也就是20秒) 
    • poolPingQuery – 发送到数据的侦测查询,用来验证连接是否正常工作,并且准备接受请求。默认是“NO PING QUERY SET”,这会引起许多数据库驱动连接由一个错误信息而导致失败。
    • poolPingEnabled – 这是开启或禁用侦测查询。如果开启,你必须用一个合法的SQL语句(最好是很快速的)设置poolPingQuery属性。默认值:false。
    • poolPingConnectionsNotUsedFor – 这是用来配置poolPingQuery多次时间被用一次。这可以被设置匹配标准的数据库连接超时时间,来避免不必要的侦测。默认值:0(也就是所有连接每一时刻都被侦测-但仅仅当poolPingEnabled为true 时适用)。

    JNDI – 这个数据源的实现是为了使用如Spring或应用服务器这类的容器,容器可以集中或在外部配置数据源,然后放置一个JNDI上下文的引用。这个数据源配置只需要两个属性:

    • initial_context – 这个属性 用来从 初始 上下 文中寻 找环 境( 也就是initialContext.lookup(initial——context))。这是个可选属性,如果被忽略,那么data_source属性将会直接以initialContext 为背景再次寻找。
    • data_source – 这是引用数据源实例位置的上下文的路径。它会以由initial_context查询返回的环境为背景来查找,如果initial_context 没有返回结果时,直接以初始上下文为环境来查找。

    和其他数据源配置相似,它也可以通过名为“env.”的前缀直接向初始上下文发送属性。比如:

    •  env.encoding=UTF8

    在初始化之后,这就会以值“UTF8”向初始上下文的构造方法传递名为“encoding”的属性。

     mappers

    既然MyBatis的行为已经由上述元素配置完了,我们现在就要定义SQL映射语句了。但是,首先我们需要告诉MyBatis到哪里去找到这些语句。Java在这方面没有提供一个很好的方法,所以最佳的方式是告诉MyBatis到哪里去找映射文件。你可以使用相对于类路径的资源引用,或者字符表示,或url引用的完全限定名(包括file:///URLs)。例如:

     1 // Using classpath relative resources
     2 <mappers>
     3     <mapper resource="org/mybatis/builder/AuthorMapper.xml" />
     4     <mapper resource="org/mybatis/builder/BlogMapper.xml" />
     5     <mapper resource="org/mybatis/builder/PostMapper.xml" />
     6 </mappers>
     7 // Using url fully qualified paths
     8 <mappers>
     9     <mapper url="file:///var/sqlmaps/AuthorMapper.xml" />
    10     <mapper url="file:///var/sqlmaps/BlogMapper.xml" />
    11     <mapper url="file:///var/sqlmaps/PostMapper.xml" />
    12 </mappers> 

     这些语句简单告诉了MyBatis去哪里找映射文件。其余的细节就是在每个SQL映射文件中了,下面的部分我们来讨论SQL映射文件。

  • 相关阅读:
    【LCA倍增】POJ1330-Nearest Common Ancestors
    【AC自动机/fail树】BZOJ3172- [Tjoi2013]单词
    【费用流】BZOJ1221-[HNOI2001] 软件开发
    【KM】POJ2195/HDU1533-Going home
    【KM算法】HDU2255-奔小康赚大钱
    【匈牙利算法】BZOJ1059-[ZJOI2007]矩阵游戏
    【KMP】BZOJ3670-[Noi2014]动物园
    【Treap】BZOJ1588-[HNOI2002]营业额统计
    【Treap模板详细注释】BZOJ3224-普通平衡树
    【tarjan求割顶】BZOJ2730-[HNOI2012]矿场搭建
  • 原文地址:https://www.cnblogs.com/xunol/p/3240780.html
Copyright © 2020-2023  润新知