Java架构师之旅(二十九 附录《MyBatis3 用户指南》中文版)

夜光序言:

 

岁月波光粼粼,赋予爱与生命,唯有生活不能被他人代替,只会有寂寞相随~~

 

Java架构师之旅(二十九 附录《MyBatis3 用户指南》中文版)_第1张图片

 

 

 

 

正文:

MyBatis 3

2010.08.01  翻译的一个版本,虽难比较老了,但是有一些基础还是值得学习,毕竟是中文版

 

Contents

MyBatis是什么? 6

准备开始 6

从XML中创建 SqlSessionFactory 实例 6

如何不使用XML来创建SqlSessionFactory 7

从SqlSessionFactory 获取SqlSession 7

探索映射SQL语句 8

关于命名空间 9

作用域和生命周期 10

Mapper XML配置 11

properties元素 12

Settings元素 13

typeAliases元素 14

typeHandlers元素 15

objectFactory元素 16

Plugins元素 17

Environments元素 18

事务管理器 20

dataSource元素 21

Mappers元素 23

SQL 映射 XML 文件 23

Select元素 24

Insert、 update、 delete元素 25

Sql元素 28

参数(Parameters) 28

resultMap元素 30

高级结果映射 32

id, result元素 34

支持的JDBC类型 35

Constructor元素 35

Association元素 36

Collection元素 40

Discriminator元素 42

Cache元素 43

cache-ref元素 46

动态SQL(Dynamic SQL) 46

if元素 46

choose, when, otherwise元素 47

trim, where, set元素 48

Foreach元素 50

Java API 52

目录结构 52

SqlSessions 53

SqlSessionFactoryBuilder 53

SqlSessionFactory 55

SqlSession 57

SelectBuilder 64

SqlBuilder 67

结束语 69

附录1 对象模型 70

附录2创建数据库 73

附录3 MyBatis实例 77

 简单select 77

 update,delete,insert 84

 自动生成主键 85

 处理NULL值 87

 使用接口映射类 88

 使用Constructor元素 90

 使用Association元素 92

 使用Collection元素 100

附录4 XML中的特殊字符 104

 

 

 

MyBatis是什么?

MyBatis是一款一流的支持自定义SQL、存储过程和高级映射的持久化框架。MyBatis几乎消除了所有的JDBC代码,也基本不需要手工去设置参数和获取检索结果。MyBatis能够使用简单的 XML格式或者注解进行来配置,能够映射基本数据元素、Map接口和POJOs(普通java对象)到数据库中的记录。

准备开始

所有的MyBatis应用都以SqlSessionFactory实例为中心。SqlSessionFactory实例通过 SqlSessionFactoryBuilder来获得,SqlSessionFactoryBuilder能够从XML配置文件或者通过自定义编写的配置类(Configuration class),来创建一个SqlSessionFactory实例。

从XML中创建 SqlSessionFactory 实例

从XML中创建SqlSessionFactory实例非常简单。建议您使用类资源路径(classpath

resource)来加载配置文件,但是您也能够使用任何方式,包括文本文件路径或者以file:// 开头URL的方式。MyBatis包括一个叫做Resources的工具类(utility class),其中包含了一系列方法,使之能简单地从classpath或其它地方加载配置文件。

String resource = "org/mybatis/example/Configuration.xml"; Reader reader = Resources.getResourceAsReader(resource); sqlMapper = new SqlSessionFactoryBuilder().build(reader);

XML配置文件包含MyBatis框架的核心设置,包括获取数据库连接的DataSource实例,和包括决定事务作用域范围和控制的事务管理等。您将能够在后面的章节中找到详细的XML配置,在这里我们先展示一个简单的例子:

PUBLIC "-//mybatis.org//DTD Config 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-config.dtd">

虽然XML配置文件中还有很多其它的配置细节,但是,上面的示例显示了最重要的部分。注意XML配置文件的头部,会使用DTD验证文档来验证该XML配置文件。body部分的environment 元素,包含了事务管理和连接池配置。Mappers元素指定了映射配置文件--包含SQL语句和映射定义的XML文件。

如何不使用XML来创建SqlSessionFactory

如果您喜欢直接通过java代码而不是通过XML创建配置选项,或者想创建您自己的配置生成器。MyBatis提供了一个完整的配置类(Configuration class),它提供了与XML文件相同的配置选项。

TransactionFactory transactionFactory = new JdbcTransactionFactory();

Environment environment = new Environment("development", transactionFactory, dataSource); Configuration configuration = new Configuration(environment);

configuration.addMapper(BlogMapper.class); SqlSessionFactory sqlSessionFactory =

new SqlSessionFactoryBuilder().build(configuration);

请注意,这种方式下的配置添加一个映射类(mapper class)。映射类是包含SQL映射注解

的Java类,从而避免了使用XML。但是,由于注解的一些局限性以及MyBatis映射的复杂性, XML仍然是一些高级的映射功能(如嵌套连接映射,Nested Join Mapping)所必须的方式。基于这个原因,如果存在XML文件,MyBatis自动寻找并加载这个XML文件。在这种情况下, BlogMapper.xml将会被类路径下名称为BlogMapper.class的类加载。详述请见后面章节。

从SqlSessionFactory 获取SqlSession

现在您已经创建了一个SqlSessionFactory(指上面的sqlMapper),正如它名字暗示那样,

您可以通过它来创建一个SqlSession 实例。SqlSession包含了所有执行数据库SQL语句的方法。您能够直接地通过SqlSession实例执行映射SQL语句。例如:

SqlSession session = sqlMapper.openSession(); try {

Blog blog = (Blog) session.selectOne(

"org.mybatis.example.BlogMapper.selectBlog", 101);

} finally {

session.close(); }

虽然这种方法很有效,MyBatis以前版本的用户对此也可能很熟悉,但现在有一个更简便的

方式,那就是对给定的映射语句,使用一个正确描述参数与返回值的接口(如

BlogMapper.class),您就能更清晰地执行类型安全的代码,从而避免错误和异常。 如:

SqlSession session = sqlSessionFactory.openSession(); try {

BlogMapper mapper = session.getMapper(BlogMapper.class);

Blog blog = mapper.selectBlog(101);

} finally {

session.close(); }

现在,让我们一起探索它们究竟是如何执行的。

探索映射SQL语句

此时,您可能想知道SqlSession 或者映射器类(Mapper class)是怎样执行的。映射SQL

语句是一个很大的主题,该主题将可能占据本文档的大部分内容。但是,为了让您看到它是怎样运行的,这里举两个例子。

在上面的例子中,映射语句已经在XML配置文件或注解中定义。让我们首先来看看XML配置文件,所有MyBatis提供的功能特性都可以通过基于XML映射配置文件配置来实现。如果您以前使用过MyBatis,对此就会很熟悉。但许多改进使XML映射文件变得更简洁清晰。下面是一个基于XML映射配置的例子,满足上述SqlSession的调用。

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

这个简单的例子看起来有不少的开销,但它确实非常地轻巧,只要您喜欢,您可以在一个映射XML文件中定义许多映射语句。虽然会有一些XML头部和DOCTYPE声明,但该文件余下的部分是自解(self explanatory,可理解为不加解释就能明白)的。它定义了映射语句的名称

“selectBlog”,在命名空间“org.mybatis.example.BlogMapper”,允许您通过指定完整类名

“org.mybatis.example.BlogMapper”来访问上面的例子:

Blog blog = (Blog) session.selectOne(

"org.mybatis.example.BlogMapper.selectBlog", 101);

这非常类似java中通过完整类名来调用方法,而这样做是有原因的。这个名称可以直接映射

到一个具在相同命名空间的映射类,这个映射类有一个方法的名称、参数及返回类型都与select 映射语句相匹配。正如您前面看到的,这使您很简单地调用映射类里的方法,下面是另外的一个例子:

BlogMapper mapper = session.getMapper(BlogMapper.class);

Blog blog = mapper.selectBlog(101);

第二种方法有很多好处。第一,它不依赖于字符串,所以更安全。第二,如果您的IDE有自动完成功能,您可以利用这功能很快导航到您的映射SQL语句。第三,您不需要关注返回类型,不需要进行强制转换,因为使用BlogMapper 接口已经限定了返回类型,它会安全地返回。

关于命名空间

  • 命名空间:在以前的版本中是可选的、复杂的且没多大用处。但在这个版本中,命名空间是必须的,并且不仅仅是简单地使用完整类名来隔离区分语句。

正如您看到的那样,命名空间能够进行接口绑定,即使您认为现在不会使用到它,但您应该按照这些准则来做。一旦使用了命名空间并且放入适当的java包命名空间中将会使您的代码清晰,并提高MyBatis的长期可用性。

  • 名称解析: 为了减少大量地输入, MyBatis对所有的配置元素,包括statements、result maps、 caches等使用下面的名称解析规则:
    • 完整类名 :(例如: “com.mypackage.MyMapper.selectAllThings”)可用来直接查找并使用。
    • 短名称: (例如: “selectAllThings”)可用来引用明确的实体对象。但是,如果出现有两个或更多(例如“com.foo.selectAllThings和com.bar.selectAllThings”)实体对象,您将收到一个错误报告(短名称含糊不清),因此,这时就必须使用完整类名。

对映射类还有一个更好的方法,就像前面的BlogMapper。它们的映射语句不需要完全在XML 中配置。相反,它们可以使用Java注解。例如上面的XML配置可以替换为:

package org.mybatis.example;

public interface BlogMapper {

@Select("SELECT * FROM blog WHERE id = #{id}")

Blog selectBlog(int id);

}

对简单的映射语句,使用注解可以显得非常地清晰。但是java注解本身的局限难于应付更复杂的语句。如果您准备要做某些复杂的事情,最好使用XML文件来配置映射语句。这将由您和您的项目小组来确定哪种方式是适合的,以一致的方式来定义映射语句是非常重要的。也就是说,您不会被限于仅用一种方式。您能够非常容易地从基于注解的映射语句移植到

XML配置文件中,反之亦然。

作用域和生命周期

理解到目前为止所讨论的类的作用域和生命周期是非常重要的。如果使用不当可导致严重的并发性问题。

SqlSessionFactoryBuilder

这个类可以在任何时候被实例化、使用和销毁。一旦您创造了SqlSessionFactory就不需要再保留它了。所以SqlSessionFactoryBuilder实例的最好的作用域是方法体内(即一个本地方法变量)。您能重用SqlSessionFactoryBuilder创建多个SqlSessionFactory实例,但最好不要把时间、资源放在解析XML文件上,而是要从中解放出来做最重要事情。

SqlSessionFactory

一旦创建,SqlSessionFactory将会存在于您的应用程序整个运行生命周期中。很少或根本没有理由去销毁它或重新创建它。最佳实践是不要在一个应用中多次创建SqlSessionFactory。这样做会被视为“没品味”。所是SqlSessionFactory最好的作用域范围是一个应用的生命周期范围。这可以由多种方式来实现,最简单的方式是使用Singleton模式或静态Singleton模式。但这不是被广泛接受的最佳做法,相反,您可能更愿意使用像Google Guice 或Spring的依赖注入方式。这些框架允许您创造一个管理器,用于管理SqlSessionFactory的生命周期。

SqlSession

每个线程都有一个SqlSession实例,SqlSession实例是不被共享的,并且不是线程安全的。因此最好的作用域是request或者method。决不要用一个静态字段或者一个类的实例字段来保存SqlSession实例引用。也不要用任何一个管理作用域,如Servlet框架中的HttpSession,来保存SqlSession的引用。如果您正在用一个WEB框架,可以把SqlSession的作用域看作类似于HTTP的请求范围。也就是说,在收到一个HTTP请求,您可以打开一个SqlSession,当您把 response返回时,就可以把SqlSession关闭。关闭会话是非常重要的,您应该要确保会话在一个finally块中被关闭。

SqlSession session = sqlSessionFactory.openSession(); try {

// do work

} finally {

session.close();

}

在您的代码里都使用这一模式将保证所有的数据库资源被正确地关闭(假如您没有把您自己

的数据库连接传递给MyBatis管理,这就对MyBatis表明您希望自己管理连接)。

Mapper实例

Mappers是创建来绑定映射语句的接口,该Mapper实例是从SqlSession得到的。因此,所有mapper实例的作用域跟创建它的SqlSession一样。但是,mapper实例最好的作用域是 method,也就是它们应该在方法内被调用,使用完即被销毁。并且mapper实例不用显式地被关闭。虽然把mapper实例保持在一个request范围(与SqlSession相似)不会产生太大的问题,但是您可能会发现,在这个层次上管理太多资源可能会失控。保持简单,就是让Mappers保持在一个方法内。下面的例子演示了这种做法。

SqlSession session = sqlSessionFactory.openSession(); try {

BlogMapper mapper = session.getMapper(BlogMapper.class);

// do work

} finally {

session.close();

}

Mapper XML配置

MyBatis的XML配置文件包含了设置和影响MyBatis行为的属性。XML配置文件的层次结构如下:

  • configuration o properties o settings o typeAliases o typeHandlers o objectFactory o plugins o environments

environment

  • transactionManager
  • dataSource

o mappers

properties元素

它们都是外部化,可替代的属性。可以配置在一个典型的Java属性文件中,或者通过

properties元素的子元素进行配置。例如:

在整个配置文件中,这些属性能够被可动态替换(即使用占位符)的属性值引用,例如:

示例中的username和password将会被替换为配置在properties元素中的相应值。driver

和 url属性则会被config.properties文件中的相应值替换。这里提供了大量的配置选项。这些属性也可以传递给sqlSessionFactoryBuilder.build()方法。例如:

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

// ... or ...

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

如果一个属性存在于多个地方,MyBatis将使用下面的顺序加载:

  • 首先读入properties元素主体中指定的属性。
  • 然后会加载类路径或者properties元素中指定的url的资源文件属性。它会覆盖前面已经读入的重复属性。
  • 通过方法参数来传递的属性将最后读取(即通过sqlSessionFactoryBuilder.build),同样也会覆盖从properties元素指定的和resource/url指定的重复属性。

因此最优先的属性是通过方法参数来传递的属性,然后是通过resource/url配置的属性,最后是在MyBatis的Mapper配置文件中,properties元素主体中指定的属性。

Settings元素

Setting元素下是些非常重要的设置选项,用于设置和改变MyBatis运行中的行为。下面的表格列出了Setting元素支持的属性、默认值及其功能。

设置选项

描述

可用值

默认值

cacheEnabled

全局性地启用或禁用所有在mapper 配置文件中配置的缓存。

true | false

true

lazyLoadingEnabled

全局性地启用或禁用延迟加载。当禁用时,所有关联的配置都会立即加载。

true | false

true

aggressiveLazyLoading

当启用后,一个有延迟加载属性的对象的任何一个延迟属性被加载时,该对象的所有的属性都会被加载。否则,所有属性都是按需加载。

true | false

true

multipleResultSetsEnabled

允许或禁止从单一的语句返回多个结果集(需要驱动程序兼容)。

true | false

true

useColumnLabel

使用列的标签而不是列的名称。在这方面,不同的驱动程序可能有不同的实现。参考驱动程序的文档或者进行测试来确定您所使用的驱动程的行为

true | false

true

useGeneratedKeys

允许JDBC 自动生成主键。需要驱动程序兼容。如果设置为true则会强行自动生成主键,然而有些则不会自动生成主键(驱动程序不兼容),但依旧会工作(如Derby)。

true | false

False

autoMappingBehavior

指定MyBatis是否以及如何自动将列映射到字段/属性。

PARTIAL: 只是自动映射简单、非嵌套的结果集。

FULL: 将会自动映射任何复杂的(嵌套或非嵌套)的结果集。

NONE,

PARTIAL,

FULL

PARTIAL

defaultExecutorType

配置默认的执行器(executor)。

SIMPLE :简单的执行器。

REUSE :重用prepared statements的执行器。

BATCH:重用statements并且进行批量更新的执行器。

SIMPLE

REUSE

BATCH

SIMPLE

defaultStatementTimeout

设置查询数据库超时时间。

任何正整数

Not Set (null)

一个Settings元素完整的配置例子如下:

typeAliases元素

别名是一个较短的Java类型的名称。这只是与XML配置文件相关联,减少输入多余的完整类

名。例如:

在这个配置中,您就可以在想要使用"domain.blog.Blog"的地方使用别名“Blog”了。

对常用的java类型,已经内置了一些别名支持。这些别名都是不区分大小写的。注意java 的基本数据类型,它们进行了特别处理,加了“_”前缀。

别名

对应的java类型

_byte

byte

_long

long

_short

short

_int

int

_integer

int

_double

double

_float

float

_boolean

boolean

string

String

Byte

Byte

Long

Long

short

Short

Int

Integer

integer

Integer

double

Double

Float

Float

boolean

Boolean

Date

Date

decimal

BigDecimal

bigdecimal

BigDecimal

object

Object

Map

Map

hashmap

HashMap

List

List

arraylist

ArrayList

collection

Collection

iterator

Iterator

typeHandlers元素

每当MyBatis设置参数到PreparedStatement或者从ResultSet结果集中取得值时,就会使用TypeHandler来处理数据库类型与java类型之间转换。下表描述了默认的TypeHandlers。

Type Handler

Java Types

JDBC Types

 

BooleanTypeHandler

Boolean, boolean

任何兼容的 BOOLEAN

 

ByteTypeHandler

Byte, byte

任何兼容的NUMERIC 或

BYTE

ShortTypeHandler

Short, short

任何兼容的NUMERIC或

SHORT INTEGER

IntegerTypeHandler

Integer, int

任何兼容的NUMERIC或

INTEGER

LongTypeHandler

Long, long

任何兼容的NUMERIC

LONG INTEGER

FloatTypeHandler

Float, float

任何兼容的NUMERIC

FLOAT

DoubleTypeHandler

Double, double

任何兼容的NUMERIC

DOUBLE

BigDecimalTypeHandler

BigDecimal

任何兼容的NUMERIC

DECIMAL

StringTypeHandler

String

CHAR, VARCHAR

 

 

ClobTypeHandler

String

CLOB, LONGVARCHAR

 

 

NStringTypeHandler

String

NVARCHAR, NCHAR

 

 

NClobTypeHandler

String

NCLOB

 

 

ByteArrayTypeHandler

byte[]

任何兼容的byte stream type

BlobTypeHandler

byte[]

BLOB, LONGVARBINARY

DateTypeHandler

Date (java.util)

TIMESTAMP

DateOnlyTypeHandler

Date (java.util)

DATE

TimeOnlyTypeHandler

Date (java.util)

TIME

SqlTimestampTypeHandler

Timestamp

(java.sql)

TIMESTAMP

SqlDateTypeHadler

Date (java.sql)

DATE

SqlTimeTypeHandler

Time (java.sql)

TIME

ObjectTypeHandler

Any

OTHER, or unspecified

EnumTypeHandler

Enumeration Type

VARCHAR –任何与string兼容的类型,储存的是编码而不是索引。

您能够重写类型处理器(type handlers),或者创建您自己的类型处理器去处理没有被支持

的或非标准的类型。要做到这一点,只要实现TypeHandler接口(org.mybatis.type),并且将您的TypeHandler类映射到java类型和可选的JDBC类型即可。例如:

// ExampleTypeHandler.java

public class ExampleTypeHandler implements TypeHandler {

public void setParameter(

PreparedStatement ps, int i, Object parameter, JdbcType jdbcType) throws SQLException { ps.setString(i, (String) parameter);

}

public Object getResult(

ResultSet rs, String columnName) throws SQLException {

return rs.getString(columnName);

}

public Object getResult(

CallableStatement cs, int columnIndex) throws SQLException {

return cs.getString(columnIndex);

}

}

// MapperConfig.xml

使用上面的TypeHandler将会重写已经存在的用来处理java的String属性、VARCHAR参数

和结果集的类型处理器。注意,MyBatis并不会通过数据库的元数据来确认类型,所以您必须指定它的一个类型处理器,用于将VARCHAR字段的参数和结果映射到正确的类型上。这是因为 MyBatis在语句的执行之前都不知道它要处理的数据类型是什么。

objectFactory元素

MyBatis每次创建一个结果对象实例都会使用ObjectFactory实例。使用默认的 ObjectFactory与使用默认的构造函数(或含参数的构造函数)来实例化目标类没什么差别。如果您想重写ObjectFactory来改变其默认行为,那您能通过创造您自己的ObjectFactory来做到。看下面的例子:

// ExampleObjectFactory.java

public class ExampleObjectFactory extends DefaultObjectFactory {

public Object create(Class type) {

return super.create(type);

}

public Object create(

Class type,

List constructorArgTypes,

List constructorArgs) {

return super.create(type, constructorArgTypes, constructorArgs);

}

public void setProperties(Properties properties) {

super.setProperties(properties);

}

}

// MapperConfig.xml

ObjectFactory接口非常简单,它包含两个create的方法,一个是默认构造器,还有一个是含参数的构造器。最后的setProperties方法用来配置ObjectFactory。在初始化您自己的 ObjectFactory实例之后,定义在objectFactory元素主体中的属性会以参数的形式传递给 setProperties方法。

Plugins元素

MyBatis允许您在映射语句执行的某些点拦截方法调用。默认情况下,MyBatis允许插件

(plug-ins)拦截下面的方法:

  • Executor

(update, query, flushStatements, commit, rollback, getTransaction, close, isClosed) ParameterHandler

(getParameterObject, setParameters)

  • ResultSetHandler

(handleResultSets, handleOutputParameters) StatementHandler

(prepare, parameterize, batch, update, query)

通过寻找完整的方法特征能够发现这些类方法的细节,以及在每个MyBatis发布版本中的源代码中找到。假如您要做的不仅仅是监视方法的调用情况,您就应该清楚您将重写的方法的行为动作。如果您试图修改或者重写给定的方法,您很可能会改变MyBatis的核心行为。这些都是比较底层的类和方法,所以要小心使用插件。

// ExamplePlugin.java

@Intercepts({@Signature( type= Executor.class, method = "update",

args = {MappedStatement.class,Object.class})})

public class ExamplePlugin implements Interceptor {

public Object intercept(Invocation invocation) throws Throwable {

return invocation.proceed();

}

public Object plugin(Object target) {

return Plugin.wrap(target, this);

}

public void setProperties(Properties properties) { }

}

// MapperConfig.xml

上面定义的插件将会拦截所有对Executor实例update方法的调用。Executor实例是一个负

责底层映射语句执行的内部对象。

重写配置类(Configuration Class)

除了能用插件方式修改MyBatis的核心行为,您也可以完全重写配置类(Configuration Class)。简单地扩展它和重写内部的任何方法,然后作为参数传递给

sqlSessionFactoryBuilder.build(myConfig)方法。再次提醒,这可能对MyBatis行为产生严重影响,因此要小心。

Environments元素

MyBatis能够配置多套运行环境,这有助于将您的SQL映射到多个数据库上。例如,在您的开发、测试、生产环境中,您可能有不同的配置。或者您可能有多个共享同一schema的生产用数据库,或者您想将相同的SQL映射应用到两个数据库等等许多用例。

但是请记住:虽然您可以配置多个运行环境,但是每个SqlSessionFactory实例只能选择一

个运行环境。

因此,如果您想连接两个数据库,就需要创建两个SqlSessionFactory实例,一个数据库对应一个SqlSessionFactory实例。如果是三个数据库,那就创建三个实例,如此类推。这真的非常容易记住:

每个数据库对应一个SqlSessionFactory实例。

要指定哪个运行环境被创建,只需要简单地将运行环境作为可选参数传递给

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 元素定义了运行环境是怎么配置的:

注意这里关键的部分:

  • 默认的运行环境ID,引用一个已经定义好的运行环境ID(例如: default=“development”)
  • 每个定义的运行环境ID(例如:id=“development”)
  • 事务管理器配置(例如: type=“JDBC”) 数据源配置(例如:type=“POOLED”)

默认的环境和环境ID是自解(self explanatory)的,只要您喜欢,就可以随意取一个名字,只要确保默认的运行环境引用一个已定义的运行环境就可以了。

译者注:

……

……

……

上面例子中,配置表明,目前使用的是test的运行环境。当然,您也可以修改为使用production的运行环境:

事务管理器

MyBatis有两种事务管理类型(即type=”[JDBC|MANAGED]”):

  • JDBC – 这个配置直接使用JDBC的提交和回滚功能。它依赖于从数据源获得连接来管理事务的生命周期。
  • MANAGED – 这个配置基本上什么都不做。它从不提交或者回滚一个连接的事务。而是让容器(例如:Spring或者J2EE应用服务器)来管理事务的生命周期 。默认情况下,它会关闭连接,但是一些容器并不会如此,因此,如果您需要通过关闭连接来停止事务,将属性closeConnection设置为false。例如:

这两个事务管理类型都不需要任何属性。然而它们都是类型别名,换句话说,您可以设置成指向己实现了TransactionFactory接口的完整类名或者别名。

public interface TransactionFactory {

void setProperties(Properties props);

Transaction newTransaction(Connection conn, boolean autoCommit); }

实例化后,任何在XML文件配置的属性都将传递给setProperties()方法。在您的实现中还

需要创建一个非常简单的Transaction接口的实现:

public interface Transaction { Connection getConnection(); void commit() throws SQLException; void rollback() throws SQLException; void close() throws SQLException; }

通过这两个接口,您能够完全自定义MyBatis如何来处理事务。

dataSource元素

dataSource元素使用标准的JDBC数据源接口来配置JDBC连接对象源。

大部分MyBatis应用都像上面例子那样配置一个数据源,但这不是必须的。需要认清的是,只有使用了延迟加载才需要数据源。

MyBatis内置了三种数据源类型 (如: type=”????”):

UNPOOLED – 这个类型的数据源实现只是在每次需要的时候简单地打开和关闭连接。虽然有点慢,但是对于不需要立即响应的简单的应用来说,不失为一种好的选择。不同的数据库在性能方

面也会有所不同,因此对于一些数据库,不使用连接池时,这个配置就是比较理想的。

UNPOOLED数据源有四个配置属性:

  • driver – 指定JDBC驱动器。
  • url – 连接数据库实例的JDBC URL。
  • username –登陆数据库的用户名。
  • password - 登陆数据库的密码。
  • defaultTransactionIsolationLevel – 指定连接的默认事务隔离级别。

另外,您也可以通过在属性前加前缀“driver”的方式,把属性传递给数据库驱动器,例如:

  • driver.encoding=UTF8

这将会通过DriverManager.getConnection(url, driverProperties) 方法,将值是

“UTF8”的属性“encoding”传递给数据库驱动器。

POOLED – 这个数据源的实现缓存了JDBC连接对象,用于避免每次创建新的数据库连接时都初始化和进行认证,加快程序响应。并发WEB应用通常通过这种做法来获得快速响应。

另外,除了上面(UNPOOLED)的属性外,对POOLED数据源,还有很多属性可以设置。

  • poolMaximumActiveConnections – 在任何特定的时间内激活(能够被使用)的连接数量,默认是10。
  • poolMaximumIdleConnections –在任何特定的时间内空闲的连接数。
  • poolMaximumCheckoutTime – 在连接池被强行返回前,一个连接能够“检出”的总时间。默认是20000ms(20秒)。
  • poolTimeToWait – 这是一上比较底层的设置,如果连接占用了很长时间,能够给连接池一个机会去打印日志,并重新尝试连接。默认是20000ms(20秒)。
  • poolPingQuery –Ping Query 是发送给数据库的Ping 信息,测试数据库连接是否良好和是否准备好了接受请求。默认值是“NO PING QUERY SET”,让大部分数据库都不使用 Ping,返回一个友好的错误信息(译者注:MyBatis通过向数据执行SQL语句来确定与数据库连接状况)。
  • poolPingEnabled – 这是允许或者禁ping query 的开关。如果允许,您同时也要用一条可用的(并且应该是最高效的)SQL语句来设置poolPingQuery 属性的值。默认是: false(即禁止)。
  • poolPingConnectionsNotUsedFor – 这个属性配置执行poolPingQuery 的间隔时间。通常设置为与数据库连接的超时时间,来避免不必要的pings 。默认是:0(允许所有连接随时进行ping测试,当然只有poolPingEnabled设置为true才会生效)。

JNDI – 这个数据源的配置是为了准备与像Spring或应用服务器能够在外部或者内部配置数据源的容器一起使用,然后在JNDI上下文中引用它。这个数据源只需配置两个属性:

  • initial_context – 这个属性被用来从InitialContext 中查找一个上下文。如:

initialContext.lookup(initial_context)

这个属性是可选的,如果忽略,那么数据源就会直接从InitialContext中查找。

  • data_source – 这个属性是引用一个能够被找到的数据源实例的上下文路径。它会查找根据initial_context 从initialContext中搜寻返回的上下文。或者在 initial_context没有提供的情况下直接在InitialContext 中进行查找。

译者注:

Context context = (Context) initialContext.lookup(initial_context);//返回

一个上下文

//Context context = (Context)initContext.lookup("java:/comp/env");

DataSource ds = (DataSource) context.lookup(data_source); //返回一个数据源

Connection conn = ds.getConnection();

//DataSource ds = (DataSource)context.lookup("jdbc/myoracle"); 如果initial_context没有配置,那么数据源就会直接从InitialContext进行查

找,如:

DataSource ds = (DataSource) initialContext.lookup(data_source);

跟数据源的其它属性配置一样,可以通过在属性名前加“env.”的方式直接传递给

InitialContext。例如:

env.encoding=UTF8

这将会把属性“encoding”及它的值“UTF8”在InitialContext实例化的时候传递给它的构造器。

Mappers元素

现在,MyBatis的行为属性都已经在上面的配置元素中配置好了,接下来开始定义映射SQL

语句。但首先,我们需要告诉MyBatis在哪里能够找到我们定义的映射SQL语句。在这方面, JAVA自动发现没有提供好的方法,因此最好的方法是告诉MyBatis在哪里能够找到这些映射文件。您可以使用类资源路径或者URL(包括file:/// URLs),例如:

// Using classpath relative resources

// Using url fully qualified paths

这些配置告诉MyBatis在哪里找到SQL映射文件。而其它的更详细的信息配置在每一个SQL

映射文件里。这些将在下一章节里讨论。

SQL 映射 XML 文件

MyBatis真正强大之处就在这些映射语句,也就是它的魔力所在。对于它的强大功能,SQL映射文件的配置却非常简单。如果您比较SQL映射文件配置与JDBC代码,您很快可以发现,使用 SQL映射文件配置可以节省95%的代码量。MyBatis被创建来专注于SQL,但又给您自己的实现极大的空间。

SQL映射XML文件只有一些基本的元素需要配置,并且要按照下面的顺序来定义(in the order that they should be defined):

  • cache –在特定的命名空间配置缓存。
  • cache-ref – 引用另外一个命名空间配置的缓存.
  • resultMap – 最复杂也是最强大的元素,用来描述如何从数据库结果集里加载对象。
  • parameterMap – 不推荐使用! 在旧的版本里使用的映射配置,这个元素在将来可能会被删除,因此不再进行描述。
  • sql – 能够被其它语句重用的SQL块。
  • insert –INSERT映射语句
  • update –UPDATE映射语句
  • delete –DELEETE映射语句
  • select –SELECT 映射语句

接下来的章节将会对每一个元素进行描述。

Select元素

Select是MyBatis中最常用的元素之一。除非您把数据取回来,否则把数据放在数据库是没什么意义的,因此大部分的应用查询数据远多于修改数据。对每一次插入、修改、删除数据都可能伴有大量的查询。这是MyBatis基本设计原则之一,也就是为什么把这么多的焦点和努力放在查询和结果集映射上。对简单的查询,select元素的配置是相当简单的,如:

这条语句叫做selectPerson,以int型(或者Integer型)作为参数,并返回一个以数据库

列名作为键值的HashMap。

注意这个参数的表示方法:#{id}

它告诉MyBatis生成PreparedStatement参数。对于JDBC,像这个参数会被标识为“?”,

然后传递给PreparedStatement,像这样:

// Similar JDBC code, NOT MyBatis…

String selectPerson = “SELECT * FROM PERSON WHERE ID=?”; PreparedStatement ps = conn.prepareStatement(selectPerson); ps.setInt(1,id);

当然,如果单独使用JDBC 去提取这个结果集并把结果集映射到对象上的话,则需要更多的代码,而这些,MyBatis 都已经为您做到了。关于参数和结果集映射,还有许多要学的,以后有专门的章节来讨论。

select 语句有很多的属性允许您详细配置每一条语句。

select from some_table

where id = #{id}

参数(Parameters)

在前面的语句中,我们可以看到一些例子中简单的参数(用于代入SQL语句中的可替换变

量,如#{id})。参数是MyBatis中非常强大的配置属性,基本上,90%的情况都会用到,如:

上面的例子演示了一个非常简单的命名参数映射,parameterType 被设置为“int”。参数可以设置为任何类型。像基本数据类型或者像Integer和String这样的简单的数据对象,(因为没有相关属性)将使用全部参数值。而如果传递的是复杂对象(一般是指JavaBean),那情况就有所不同。例如::

insert into users (id, username, password)

values (#{id}, #{username}, #{password})

如果参数对象User被传递给SQL语句,那么将会搜寻PreparedStatement里的id,username

和password属性,并被User对象里相应的属性值替换。这种传递参数到语句的方式非常优雅和简单。但参数映射还在很多特性。

首先,像MyBatis 其它部分一样,参数可以指定许多的数据类型。

#{property,javaType=int,jdbcType=NUMERIC}

像MyBatis 的其它部分一样,这个javaType 是由参数对象决定,除了HashMap 以外。然后

这个javaType应该确保指定正确的TypeHandler被使用。

Note:如果传递了一个空值,那这个JDBC Type 对于所有JDBC 允许为空的列来说是必须的。您可以研读一下关于PreparedStatement.setNull()的JavaDocs 文档。

对于需要自定义类型处理,您也可以指定一个特殊的TypeHandler 类或者别名,如:

#{age,javaType=int,jdbcType=NUMERIC,typeHandler=MyTypeHandler}

当然,这看起来更加复杂了,不过,这种情况比较少见。

对于数据类型,可以使用numericScale 来指定小数位的长度。

#{height,javaType=double,jdbcType=NUMERIC,numericScale=2}

最后,mode属性允许您指定IN、 OUT 或 INOUT 参数。如果参数是OUT 或 INOUT,参数对象

属的实际值将会改变,正如您希望调用一个输出参数。如果mode=OUT (或者 INOUT),并且

jdbcType=CURSOR (如Oracle 的REFCURSOR),您必须指定一个resultMap映射结果集给这个参数类型。注意这里的javaType 类型是可选的,如果为空值而jdbcType =CURSOR 的话,则会自动地设给ResultSet。

#{department, mode=OUT, jdbcType=CURSOR, javaType=ResultSet, resultMap=departmentResultMap}

MyBatis也支持像structs这样高级的数据类型,但当您把mode 设置为out的时候,您必须把类型名告诉执行语句。例如:

#{middleInitial, mode=OUT, jdbcType=STRUCT, jdbcTypeName=MY_TYPE, resultMap=departmentResultMap}

尽管有这些强大的选项,但是大多数情况下您只需指定属性名,MyBatis将会识别其它部分

的设置。最多就是给可以为null值的列指定jdbcType:

#{firstName}

#{middleInitial,jdbcType=VARCHAR} #{lastName}

字符串替换

默认的情况下,使用#{}语法会促使MyBatis 生成PreparedStatement并且安全地设置 PreparedStatement 参数(=?)值。尽量这是安全、快捷并且是经常使用的,但有时候您可能想直接未更改的字符串代入到SQL 语句中。比如说,对于ORDER BY,您可以这样使用

ORDER BY ${columnName}

这样MyBatis就不会修改这个字符串了。

警告: 这种不加修改地接收用户输入并应用到语句的方式,是非常不安全的。这使用户能够进行SQL注入,破坏代码。所以,要么这些字段不允许用户输入,要么用户每次输入后都进行检测和规避。

resultMap元素

resultMap元素是MyBatis中最重要最强大的元素。与使用JDBC从结果集获取数据相比,它可

以省掉90%的代码,也可以允许您做一些JDBC不支持的事。事实上,要写一个类似于连结映射

(join mapping)这样复杂的交互代码,可能需要上千行的代码。设计ResultMaps 的目的,就是只使用简单的配置语句而不需要详细地处理结果集映射,对更复杂的语句除了使用一些必须的语句描述以外,就不需要其它的处理了。您可能已经看到过这样简单映射语句,它并没有使用resultMap,例如:

select id, username, hashedPassword from some_table

where id = #{id}

别忘了别名是您的朋友,使用别名您不用输入那长长的类路径。例如:

select

 

user_id

as “id”,

user_name

as “userName”,

hashed_password from some_table where id = #{id}

as “hashedPassword”

ResultMaps的知识您可能已经学到了许多,但还有一个您从没见到过。为了举例,让我们看看最后一个例子,作为另一种解决列名不匹配的方法。

这个语句将会被resultMap 属性引用(注意,我们没有使用resultType)。如:

select

B.id as blog_id,

B.title as blog_title,

B.author_id as blog_author_id,

A.id as author_id,

A.username as author_username,

A.password as author_password,

A.email as author_email,

A.bio as author_bio,

A.favourite_section as author_favourite_section,

P.id as post_id,

P.blog_id as post_blog_id,

P.author_id as post_author_id,

P.created_on as post_created_on,

P.section as post_section,

P.subject as post_subject,

P.draft as draft,

P.body as post_body,

C.id as comment_id,

C.post_id as comment_post_id,

C.name as comment_name,

C.comment as comment_text,

T.id as tag_id,

T.name as tag_name from Blog B left outer join Author A on B.author_id = A.id left outer join Post P on B.id = P.blog_id left outer join Comment C on P.id = C.post_id left outer join Post_Tag PT on PT.post_id = P.id left outer join Tag T on PT.tag_id = T.id

where B.id = #{id}

您可能想要把它映射到一个智能的对象模型,包括由一个作者写的一个博客,有许多文章

Post,帖子),每个文章由0个或者多个评论和标签。下面是一个复杂ResultMap 的完整例子(假定作者、博客、文章、评论和标签都是别名)。仔细看看这个例子,但是不用太担心,我们会一步步地来分析,一眼看上去可能让人沮丧,但是实际上非常简单的

这个resultMap 的元素的子元素比较多,讨论起来比较宽泛。下面我们从概念上概览一下这个resultMap的元素。

resultMap

  • constructor – 实例化的时候通过构造器将结果集注入到类中 o idArg – ID 参数; 将结果集标记为ID,以方便全局调用 o arg – 注入构造器的结果集
  • id –结果集ID,将结果集标记为ID,以方便全局调用
  • result – 注入一个字段或者javabean属性的结果
  • association – 复杂类型联合; 许多查询结果合成这个类型
    • 嵌套结果映射– associations能引用自身, 或者从其它地方引用
  • collection – 复杂类型集合
    • 嵌套结果映射– collections能引用自身, 或者从其它地方引用
  • discriminator –使用一个结果值以决定使用哪个resultMap o case – 基于不同值的结果映射
    • 嵌套结果映射–case也能引用它自身, 所以也能包含这些同样的元素。它也可以从外部引用resultMap
    • 最佳实践: 逐步地生成resultMap,单元测试对此非常有帮助。如果您尝试一下子就

生成像上面这样巨大的resultMap,可能会出错,并且工作起来非常吃力。从简单地开始,再一步步地扩展,并且进行单元测试。使用框架开发有一个缺点,它们有时像是一个黑盒子。为了确保达到您所预想的行为,最好的方式就是进行单元测试。这对提交bugs 也非常有用。

下一节,我们一步步地查看这些细节。

id, result元素

这是最基本的结果集映射。idresult将列映射到属性或简单的数据类型字段 (String, int, double, Date等)。

这两者唯一不同的是,在比较对象实例时id作为结果集的标识属性。这有助于提高总体性能,特别是应用缓存和嵌套结果映射的时候。

id、result属性如下:

Attribute

Description

property

映射数据库列的字段或属性。如果JavaBean 的属性与给定的名称匹配,就会使用匹配的名字。否则,MyBatis 将搜索给定名称的字段。两种情况下您都可以使用逗点的属性形式。比如,您可以映射到“username”,也可以映射到

“address.street.number”。

column

数据库的列名或者列标签别名。与传递给resultSet.getString(columnName)的参数名称相同。

javaType

完整java类名或别名 (参考上面的内置别名列表)。如果映射到一个JavaBean,那MyBatis 通常会自行检测到。然而,如果映射到一个HashMap,那您应该明确

 

指定javaType 来确保所需行为。

jdbcType

这张表下面支持的JDBC类型列表列出的JDBC类型。这个属性只在insert,

update 或delete 的时候针对允许空的列有用。JDBC 需要这项,但MyBatis 不需要。如果您直接编写JDBC代码,在允许为空值的情况下需要指定这个类型。

typeHandler

我们已经在文档中讨论过默认类型处理器。使用这个属性可以重写默认类型处理器。它的值可以是一个TypeHandler实现的完整类名,也可以是一个类型别名。

支持的JDBC类型

MyBatis支持如下的JDBC类型:

BIT

FLOAT

CHAR

TIMESTAMP

OTHER

UNDEFINED

TINYINT

REAL

VARCHAR

BINARY

BLOB

NVARCHAR

SMALLINT

DOUBLE

LONGVARCHAR

VARBINARY

CLOB

NCHAR

INTEGER

NUMERIC

DATE

LONGVARBINARY

BOOLEAN

NCLOB

BIGINT

DECIMAL

TIME

NULL

CURSOR

 

Constructor元素

当属性与DTO,或者与您自己的域模型一起工作的时候,许多场合要用到不变类。通常,包含

引用,或者查找的数据很少或者数据不会改变的的表,适合映射到不变类中。构造器注入允许您在类实例化后给类设值,这不需要通过public方法。MyBatis 同样也支持private 属性和 JavaBeans 的私有属性达到这一点,但是一些用户可能更喜欢使用构造器注入。构造器元素可以做到这点。

考虑下面的构造器:

public class User { //…

public User(int id, String username) {

//…

}

//…

}

为了将结果注入构造器,MyBatis需要使用它的参数类型来标记构造器。Java没有办法通过

参数名称来反射获得。因此当创建constructor 元素,确保参数是按顺序的并且指定了正确的类型。

其它的属性与规则与id、result元素的一样。

Attribute

Description

column

数据库的列名或者列标签别名。与传递给resultSet.getString(columnName)的参数名称相同。

javaType

完整java类名或别名 (参考上面的内置别名列表)。如果映射到一个JavaBean,那MyBatis 通常会自行检测到。然而,如果映射到一个HashMap,那您应该明确指定javaType 来确保所需行为。

jdbcType

支持的JDBC类型列表中列出的JDBC类型。这个属性只在insert,update 或

delete 的时候针对允许空的列有用。JDBC 需要这项,但MyBatis 不需要。如果您直接编写JDBC代码,在允许为空值的情况下需要指定这个类型。

typeHandler

我们已经在文档中讨论过默认类型处理器。使用这个属性可以重写默认类型处理器。它的值可以是一个TypeHandler实现的完整类名,也可以是一个类型别名。

Association元素

Association元素处理“has-one”(一对一)这种类型关系。比如在我们的例子中,一个 Blog有一个Author。联合映射与其它的结果集映射工作方式差不多,指定property、column、 javaType(通常MyBatis会自动识别)、jdbcType(如果需要)、typeHandler。

不同的地方是您需要告诉MyBatis 如何加载一个联合查询。MyBatis使用两种方式来加载:

  • Nested Select: 通过执行另一个返回预期复杂类型的映射SQL语句(即引用外部定义好的SQL语句块)。
  • Nested Results: 通过嵌套结果映射(nested result mappings)来处理联接结果集(joined results)的重复子集。

首先,让我们检查一下元素属性。正如您看到的,它不同于普通只有select和resultMap属性的结果映射。

Attribute

Description

property

映射数据库列的字段或属性。如果JavaBean 的属性与给定的名称匹配,就会使用匹配的名字。否则,MyBatis 将搜索给定名称的字段。两种情况下您都可以使用逗点的属性形式。比如,您可以映射到”username”,也可以映射到更复杂点的”address.street.number”。

column

数据库的列名或者列标签别名。与传递给resultSet.getString(columnName)的参数名称相同。

注意: 在处理组合键时, 您可以使用column= “{prop1=col1,prop2=col2}” 这样的语法,设置多个列名传入到嵌套查询语句。这就会把prop1和prop2设置

 

到目标嵌套选择语句的参数对象中。

javaType

完整java类名或别名 (参考上面的内置别名列表)。如果映射到一个JavaBean,那MyBatis 通常会自行检测到。然而,如果映射到一个HashMap,那您应该明确指定javaType 来确保所需行为。

jdbcType

支持的JDBC类型列表中列出的JDBC类型。这个属性只在insert,update 或

delete 的时候针对允许空的列有用。JDBC 需要这项,但MyBatis 不需要。如果您直接编写JDBC代码,在允许为空值的情况下需要指定这个类型。

typeHandler

我们已经在文档中讨论过默认类型处理器。使用这个属性可以重写默认类型处理器。它的值可以是一个TypeHandler实现的完整类名,也可以是一个类型别名。

联合嵌套选择(Nested Select for Association)

select

通过这个属性,通过ID引用另一个加载复杂类型的映射语句。从指定列属性中返

回的值,将作为参数设置给目标select 语句。表格下方将有一个例子。注意:在处理组合键时,您可以使用column=”{prop1=col1,prop2=col2}”这样的语

法,设置多个列名传入到嵌套语句。这就会把prop1和prop2设置到目标嵌套语句的参数对象中。

例如:

我们使用两个select语句:一个用来加载Blog,另一个用来加载Author。Blog的

resultMap 描述了使用“selectAuthor”语句来加载author的属性。如果列名和属性名称相匹配的话,所有匹配的属性都会自动加载。

译者注:

上面的例子,首先执行查出的数据都会自动赋值给”blogResult”的与列名匹配的属性,这时blog_id,title等就被赋值了。同时“blogResult”还有一个关联属性"Author",执行嵌套查询select=”selectAuthor”后,Author对象的属性id,username,password,email,bio也被赋于与数据库匹配的值。

Blog

{

blog_id; title;

Author author

{ id; username; password; email; bio;

}

}

建议不要使用Batatis的自动赋值,这样不能够清晰地知道要映射哪些属性,并且有时候还不能保证正确地映射数据库检索结果。

虽然这个方法简单,但是对于大数据集或列表查询,就不尽如人意了。这个问题被称为“N+1 选择问题”(N+1 Selects Problem)。概括地说,N+1选择问题是这样产生的:

  • 您执行单条SQL语句去获取一个列表的记录 ( “+1”)。
  • 对列表中的每一条记录,再执行一个联合select 语句来加载每条记录更加详细的信息

(“N”)。

译者注:

如:执行一条SQL语句获得了10条记录,这10条记录的每一条再执行一条SQL语句去加载更详细的信息,这就执行了10+1次查询。

这个问题会导致成千上万的SQL语句的执行,因此并非总是可取的。

上面的例子,MyBatis可以使用延迟加载这些查询,因此这些查询立马可节省开销。然而,如果您加载一个列表后立即迭代访问嵌套的数据,这将会调用所有的延迟加载,因此性能会变得非常糟糕。

鉴于此,这有另外一种方式。

联合嵌套结果集(Nested Results for Association)

resultMap

一个可以映射联合嵌套结果集到一个适合的对象视图上的ResultMap 。这是一个替代的方式去调用另一个select 语句。它允许您去联合多个表到一个结果集里。这样的结果集可能包括冗余的、重复的需要分解和正确映射到一个嵌套对象视图的数据组。简言之,MyBatis 让您把结果映射‘链接’到一起,用来处理嵌套结果。举个例子会更好理解,例子在表格下方。

您已经在上面看到了一个非常复杂的嵌套联合的例子,接下的演示的例子会更简单一些。我们把Blog和Author表联接起来查询,而不是执行分开的查询语句:

注意到这个连接(join),要确保所有的别名都是唯一且无歧义的。这使映射容易多了,现

在我们来映射结果集:

在上面的例子中,您会看到Blog的作者(“author”)联合一个“authorResult”结果映射

来加载Author实例。

重点提示: id元素在嵌套结果映射中扮演了非常重要的角色,您应该总是指定一个或多个属性来唯一标识这个结果集。事实上,如果您没有那样做,MyBatis也会工作,但是会导致严重性能开销。选择尽量少的属性来唯一标识结果,而使用主键是最明显的选择(即使是复合主键)。

上面的例子使用一个扩展的resultMap 元素来联合映射。这可使Author结果映射可重复使

用。然后,如果您不需要重用它,您可以直接嵌套这个联合结果映射。下面例子就是使用这样的方式:

在上面的例子中您已经看到如果处理“一对一”(“ has one”)类型的联合查询。但是对

于“一对多”(“has many”)的情况如果处理呢?这个问题在下一节讨论。

Collection元素

collection元素的作用差不多和association元素的作用一样。事实上,它们非常相似,以

至于再对相似点进行描述会显得冗余,因此我们只关注它们的不同点。

继续我们上面的例子,一个Blog只有一个Author。但一个Blog有许多帖子(文章)。在 Blog类中,会像下面这样定义相应属性:

private List posts;

映射一个嵌套结果集到一个列表,我们使用collection元素。就像association元素那

样,我们使用嵌套查询,或者从连接中嵌套结果集。

集合嵌套选择(Nested Select for Collection)

首先我们使用嵌套选择来加载Blog的文章。

一看上去这有许多东西需要注意,但大部分看起与我们在association元素中学过的相似。首

先,您会注意到我们使用了collection元素,然后会注意到一个新的属性“ofType”。这个元素是用来区别JavaBean属性(或者字段)类型和集合所包括的类型。因此您会读到下面这段代码。

ofType="Post" select=”selectPostsForBlog”/>

理解为: “一个名为posts,类型为Post的ArrayList集合( A collection of posts in an ArrayList of type Post)” 。

javaType 属性不是必须的,通常MyBatis 会自动识别,所以您通常可以简略地写成:

集合的嵌套结果集(Nested Results for Collection)

这时候,您可能已经猜出嵌套结果集是怎样工作的了,因为它与association非常相似,只

不过多了一个属性“ofType”。让我们看下这个SQL:

同样,我们把Blog和Post两张表连接在一起,并且也保证列标签名在映射的时候是唯一且

无歧义的。现在将Blog和Post的集合映射在一起是多么简单:

再次强调一下,id元素是非常重要的。如果您忘了或者不知道id元素的作用,请先读一下上面association一节。

如果希望结果映射有更好的可重用性,您可以使用下面的方式:

Note: 在您的映射中没有深度、宽度、联合和集合数目的限制。但应该谨记,在进行映射的时候也要考虑性能的因素。应用程序的单元测试和性能测试帮助您发现最好的方式可能要花很长时间。但幸运的是,MyBatis允许您以后可以修改您的想法,这时只需要修改少量代码就行了。

关于高级联合和集合映射是一个比较深入的课题,文档只能帮您了解到这里,多做一些实

践,一切将很快变得容易理解。

Discriminator元素

有时候一条数据库查询可能会返回包括各种不同的数据类型的结果集。Discriminator(识别

器)元素被设计来处理这种情况,以及其它像类继承层次情况。识别器非常好理解,它就像java 里的switch语句。

Discriminator定义要指定columnjavaType属性。列是MyBatis 将要取出进行比较的值, javaType 用来确定适当的测试是否正确运行(即使是String 在大部分情况下也可以工作),例:

在这个例子中,MyBatis将会从结果集中取出每条记录,然后比较它的vehicle type的值。如果匹配任何discriminator中的case,它将使用由case指定的resultMap。这是排它性的,换句话说,其它的case的resultMap将会被忽略(除非使用我们下面说到的extended)。如果没有匹配到任何case,MyBatis只是简单的使用定义在discriminator块外面的resultMap。所以,如果carResult像下面这样定义:

那么,只有doorCount属性会被加载。这样做是为了与识别器cases群组完全独立开来,哪怕

它与上一层的resultMap 一点关系都没有。在刚才的例子里我们当然知道cars和vehicles的关

系,a Car is-a Vehicle。因此,我们也要把其它属性加载进来。我们要稍稍改动一下

resultMap:

现在,vehicleResult 和carResult 的所有属性都会被加载。

可能有人会认为这样扩展映射定义有一点单调了,所以还有一种可选的更加简单明了的映射

风格语法。例如:

记住:对于这么多的结果映射,如果您不指定任何的结果集,那么MyBatis 会自动地将列名与属性相匹配。所以上面所举的例子比实际中需要的要详细。尽管如此,大部分数据库有点复杂,并且它并不是所有情况都是完全可以适用的。

Cache元素

MyBatis 包含一个强大的、可配置、可定制的查询缓存机制。MyBatis 3 的缓存实现有了许多改进,使它更强大更容易配置。默认的情况,缓存是没有开启的,除了会话缓存以外,会话缓存可以提高性能,且能解决循环依赖。开启二级缓存,您只需要在SQL映射文件中加入简单的一行:

这句简单的语句作用如下:

  • 所有映射文件里的select语句的结果都会被缓存。
  • 所有映射文件里的 insert、 update 和 delete 语句执行都会清空缓存 。
  • 缓存使用最近最少使用算法(LRU)来回收。
  • 缓存不会被设定的时间所清空。
  • 每个缓存可以存储1024 个列表或对象的引用(不管查询方法返回的是什么)。
  • 缓存将作为“读/写”缓存,意味着检索的对象不是共享的且可以被调用者安全地修改,

而不会被其它调用者或者线程干扰。

所有这些特性都可以通过cache元素进行修改。例如:

readOnly="true"/>

这种高级的配置创建一个每60秒刷新一次的FIFO 缓存,存储512个结果对象或列表的引用,并且返回的对象是只读的。因此在不用的线程里的调用者修改它们可能会引用冲突。

可用的回收算法如下:

  • LRU – 最近最少使用:移出最近最长时间内都没有被使用的对象。
  • FIFO –先进先出:移除最先进入缓存的对象。
  • SOFT – 软引用: 基于垃圾回收机制和软引用规则来移除对象(空间内存不足时才进行回收)。
  • WEAK – 弱引用: 基于垃圾回收机制和弱引用规则(垃圾回收器扫描到时即进行回收)。

默认使用LRU。

flushInterval :设置任何正整数,代表一个以毫秒为单位的合理时间。默认是没有设置,因此没有刷新间隔时间被使用,在语句每次调用时才进行刷新。

Size:属性可以设置为一个正整数,您需要留意您要缓存对象的大小和环境中可用的内存空间。默认是1024。

readOnly:属性可以被设置为true 或false。只读缓存将对所有调用者返回同一个实例。因此这些对象都不能被修改,这可以极大的提高性能。可写的缓存将通过序列化来返回一个缓存对象的拷贝。这会比较慢,但是比较安全。所以默认值是false。

使用自定义缓存

除了上面已经定义好的缓存方式,您能够通过您自己的缓存实现来完全重写缓存行为,或者

通过创建第三方缓存解决方案的适配器。

这个例子演示了如果自定义缓存实现。由type指定的类必须实现org.mybatis.cache.Cache

接口。这个接口是MyBatis框架比较复杂的接口之一,先给个示例:

public interface Cache { String getId(); int getSize();

void putObject(Object key, Object value); Object getObject(Object key); boolean hasKey(Object key); Object removeObject(Object key); void clear();

ReadWriteLock getReadWriteLock(); }

要配置您的缓存,简单地添加一个公共的JavaBeans 属性到您的缓存实现中,然后通过cache 元素设置属性进行传递,下面示例,将在您的缓存实现上调用一个setCacheFile(String file)方法。

您可以使用所有简单的JavaBeans属性,MyBatis会自动进行转换。需要牢记的是一个缓存配置和缓存实例都绑定到一个SQL Map 文件命名空间。因此,所有的这个相同命名空间的语句也都和这个缓存绑定。语句可以修改如何与这个缓存相匹配,或者使用两个简单的属性来完全排除它们自己。默认情况下,语句像下面这样来配置:

parameterType=”Blog” resultType=”Blog”>

SELECT * FROM BLOG

WHERE state = ‘ACTIVE’

AND title like #{title}

这条语句提供一个带功能性的可选的文字。如果您没有传入标题,那么将返回所有激活的 Blog。如果您传入了一个标题,那它就会查找与这个标题匹配的Blog(在这种情况下,您的参数值可能需要包括任何masking或者通配符)。如果我们想要可选地根据标题或者作者查询怎么办?首先,我把语句的名称稍稍改一下,使

得看起来更直观。然后简单地加上另外一个条件。

choose, when, otherwise元素

有时候我们不想应用所有的条件,而是想从多个选项中选择一个。与java中的switch语句

相似,MyBatis提供了一个choose元素。

让我们继续使用上面的例子,但这次我们只搜索有提供查询标题的,或者只搜索有提供查询

作者的数据。如果两者都没有提供,那只返回加精的Blog (可能是管理员有选择性的查询,而不是返回大量无意义的随机的Blog)。

trim, where, set元素

考虑一下我们上面提到的‘if ’的例子中,如果现在我们把‘ACTIVE=1’也做为条件,会发

生什么情况。

如果我们一个条件都不设置,会发生什么呢?语句最终可能会变成这个样子:

SELECT * FROM BLOG WHERE

这将会执行失败。如果只有第二个条件满足呢?语句最终会变成这样:

SELECT * FROM BLOG

WHERE

AND title like ‘someTitle’

这同样会执行失败。这个问题仅用条件很难简单地解决,如果您已经这么写了,那您可能以

后永远都不想犯这样的错了。

MyBatis有个简单的方案能解决这里面90%的问题。如果where没有出现的时候,您可以自定一个。修改一下,就能完全解决:

where元素知道插入“where”如果它包含的标签中有内容返回的话。此外,如果返回的内容以“AND” 或者 “OR”开头,它会把“AND” 或者 “OR”去掉。

如果 where元素的行为并没有完全按您想象的那样,您还可以使用trim元素来自定义。例如,下面的trim与where元素实现相同功能:

overrides属性使用了管道分隔的文本列表来覆写,而且它的空白也不能忽略的。这样的结果是移出了指定在overrides 属性里字符,而在开头插入prefix属性中指定的字符。

译者注,下面的两种配置方法效果是一样的:

trim >

下面的使用SET元素也类似。

在动态update语句里相似的解决方式叫做set,这个set元素能够动态地更新列。例如:

update Author

username=#{username},

password=#{password},

email=#{email},

bio=#{bio}

where id=#{id}

set元素将动态的配置SET 关键字,也用来剔除追加到条件末尾的任何不相关的逗号。

您想知道等同的trim元素该怎么写吧,它就像这样 :

注意这种情况,我们剔除了一个后缀, 同时追加了一个前缀。

Foreach元素

另一个动态SQL经常使用到的功能是集合迭代,通常用在IN条件句。例如:

译者注:

对上面这个映射SQL语句的java调用代码示例如下:

List authorIdList = new ArrayList(); postList.add(2); postList.add(3); postList.add(4);

List postList = (List)session.selectList(

"selectPostIn", postList);

//将会查询出ID是2、3、4的文章。

foreach 元素非常强大,允许您指定一个集合,申明能够用在元素体内的项和索引变量。也允许您指定开始和结束的字符,也可以加入一个分隔符到迭代器之间。这个元素的聪明之处在于它不会意外地追加额外的分隔符。

注意: 您可以把一个List 实例或者一个数组作为一个参数对象传递给MyBatis。如果您这么做,MyBatis会自动将它包装成一个Map,并以名字作为key。List实例会以

“list”作为key,array实例会以“ array” 作为key。

关于XML配置文件及XML映射文件就讨论到这了,下一章节我们将详细讨论Java API。

 

Java API

现在您知道如何配置MyBatis和生成映射,您已经收获良多。MyBatis的Java API让您的努力获得回报。正如您将看到的,相比JDBC,MyBatis极大地简化了您的代码,并使您的代码保持清晰、容易理解和维护。MyBatis3推出了一系列重大的改进来使SQL映射更好地工作。

目录结构

在我们深入Java API 之前,理解目录结构的最佳实践是非常重要的。MyBatis 非常灵活,您可以对您的文件做任何事,但是做为一个框架,总有一个首选的方式。

让我们来看一下典型的应用目录结构:

/my_application

/bin

/devlib

/lib

/src

/org/myapp/

/action /data

/SqlMapConfig.xml

/BlogMapper.java

/BlogMapper.xml

/model

/service

/view

/properties

/test

/org/myapp/

/action

/data

/model

/service

  • MyBatis *.jar 文件存放在这里。
  • MyBatis 物件放在这里。如: 映射器类(Mapper Classes), XML 配置文件, XML映射文件。
  • Properties 存放您自己的属性配置文件

记住,这只是推荐, 并不是必须的, 但使用这样通用的目录结构,其它开

/view

/properties 发人员将会感激您。

/web

/WEB-INF

/web.xml

这章节的例子都是假定您按照上面的目录结构来存放文件。

SqlSessions

SqlSession是与MyBatis一起工作的基本java接口。通过这个接口,您可以执行命令、获得映射和管理事务。我们很快就会讨论SqlSession,但首先我们要了解如果获得一个SqlSession 实例。SqlSessions是由SqlSessionFactory实例创建的。SqlSessionFactory包含从不同的方式创建SqlSessions实例的方法。而SqlSessionFactory又是SqlSessionFactoryBuilder从XML文件,注解或者手动编写java配置代码中创建的。

SqlSessionFactoryBuilder

SqlSessionFactoryBuilder 有五个 build() 方法, 每个方法允许您从不同来源中创建

SqlSession

SqlSessionFactory build(Reader reader)

SqlSessionFactory build(Reader reader, String environment)

SqlSessionFactory build(Reader reader, Properties properties)

SqlSessionFactory build(Reader reader, String env, Properties props) SqlSessionFactory build(Configuration config)

前四个方法较为常用,它们使用一个引用XML文件的Reader实例,或者更具体地说是上面讨

论的SqlMapConfig.xml文件。可选参数是environmentproperties。Environment决定加载的环境(包括数据源和事务管理)。例如:

如果您调用一个传递environment参数的build方法,MyBatis将使用所传递的环境的配

置。当然,如果您指定一个不可用的环境,您将收到一个错误。如果您调用的build方法没有传

environment参数,将使用默认的环境(像上面的例子中由default=“development”指定)。

如果您调用一个传递properties实例的方法,MyBatis将会加载传递进来的属性,并使这些

属性在配置文件中生效。这些属性能够应用于配置文件中使用${propName}语法的地方。

回想一下,属性可以从SqlMapConfig.xml文件中被引用,也可以直接指定它。因此很重要的

是要理解它们的优先顺序。在文档前面已经提到过,这里再次引用以供参考:如果属性存在于多个地方,MyBatis按照下面的顺序来加载:

  • 首先读入properties元素主体中指定的属性。
  • 然后会加载类路径或者properties元素中指定的url的资源文件属性。它会覆盖前面已经读入的重复属性。
  • 通过方法参数来传递的属性将最后读取(即通过sqlSessionFactoryBuilder.build),同样也会覆盖从properties元素指定的和resource/url指定的重复属性。

因此优先级最高的属性是通过方法参数来传递的属性,然后是通过resource/url配置的属

性,最后是在MyBatis的映射配置文件中,properties元素主体中指定的属性。

总得来说,前面四个方法大致相同,通过重载的方式允许您可选地指定environment和/或者 properties。这里是一个从SqlMapConfig.xml文件中创建SqlSessionFactory的例子:

String resource = "org/mybatis/builder/MapperConfig.xml";

Reader reader = Resources.getResourceAsReader(resource);

SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory factory = builder.build(reader);

注意,我们使用了Resources工具类,Resources工具类放在org.mybatis.io包中。Resources 类,正如它的名字暗示,帮助我们从类路径、文件系统或者WEB URL加载资源。浏览一下这个类的源代码,或者使用您的IDE 查看,就会显露一系列有用的方法,简单列表如下:

URL getResourceURL(String resource)

URL getResourceURL(ClassLoader loader, String resource)

InputStream getResourceAsStream(String resource)

InputStream getResourceAsStream(ClassLoader loader, String resource)

Properties getResourceAsProperties(String resource)

Properties getResourceAsProperties(ClassLoader loader, String resource)

Reader getResourceAsReader(String resource)

Reader getResourceAsReader(ClassLoader loader, String resource)

File getResourceAsFile(String resource)

File getResourceAsFile(ClassLoader loader, String resource)

InputStream getUrlAsStream(String urlString)

Reader getUrlAsReader(String urlString)

Properties getUrlAsProperties(String urlString) Class classForName(String className)

最后一个build方法传递一个Configuration的实例。Configuration类包含您需要了解的关于SqlSessionFactory实例的所有事情。Configuration类对于内省的配置很有用,包括发现和操作SQL映射。Configuration类有您已经学过的所有配置开关,像java API那样提供方法暴露出来。下面是一个例子,演示如何操作Configuration实例,如何把这个实例传递给build() 方法来创建SqlSessionFactory。

DataSource dataSource = BaseDataTest.createBlogDataSource();

TransactionFactory transactionFactory = new JdbcTransactionFactory();

Environment environment = new Environment("development", transactionFactory, dataSource);

Configuration configuration = new Configuration(environment);

configuration.setLazyLoadingEnabled(true); configuration.setEnhancementEnabled(true); configuration.getTypeAliasRegistry().registerAlias(Blog.class); configuration.getTypeAliasRegistry().registerAlias(Post.class); configuration.getTypeAliasRegistry().registerAlias(Author.class); configuration.addMapper(BoundBlogMapper.class); configuration.addMapper(BoundAuthorMapper.class);

SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory factory = builder.build(configuration);

现在您拥有了一个SqlSessionFactory,您就能用它来创建SqlSession实例了。

SqlSessionFactory

SqlSessionFactory 有六个方法用来创建SqlSession实例。在一般情况下,选择其中一个方法要考虑下面几个方面:

  • 事务(Transaction): 您是否想为会话使用事务作用域,或者自动提交 (通常是指数据库或者JDBC驱动没有事务的情况下)
  • 连接(Connection): 您想从配置数据源获得一个连接,还是想自己提供一个?
  • 执行(Execution): 您想让MyBatis重复使用用PreparedStatements 还是希望批量更新(包括插入和删除)?

重载的openSession()方法集,允许您选择任何一种有意义的组合。

SqlSession openSession()

SqlSession openSession(boolean autoCommit)

SqlSession openSession(Connection connection)

SqlSession openSession(TransactionIsolationLevel level)

SqlSession openSession(ExecutorType execType,TransactionIsolationLevel level)

SqlSession openSession(ExecutorType execType)

SqlSession openSession(ExecutorType execType, boolean autoCommit)

SqlSession openSession(ExecutorType execType, Connection connection)

Configuration getConfiguration();

默认的openSession()方法不需要参数,创建的SqlSession有以下特征:

  • 将会启用一个事务作用域 (即不会自动提交)
  • 一个连接对象将从正在生效的运行环境所配置的数据源实例中获得
  • 事务隔离级别是由驱动或数据源使用的默认级别
  • PreparedStatements不会被重用,也不会进行批量更新

大部分方法自身都是很容易理解的。要启用自动提交,传入“true”值给可选参数

autoCommit。要使用自己的连接,可以传入一个连接实例给参数connection。注意,没有提供重载的方法同时传入Connection和autoCommit参数,因为MyBatis将会使用所提供的连接对象正在使用的设置。MyBatis使用java枚举包装器作为事务隔离级别TransactionIsolationLevel,并有5个JDBC支持的级别:NONE、READ_UNCOMMITTED、READ_COMMITTED、REPEATABLE_READ、 SERIALIZABLE,如果不设置,它们按默认方式工作。

还有一个前面没有遇到过的新参数ExecutorType,这个枚举定义了三个值:

ExecutorType.SIMPLE

这个类型不做特殊的事情,它只为每个执行语句创建一个PreparedStatement。

ExecutorType.REUSE

这个类型的执行器(Executor)重用PreparedStatements。

ExecutorType.BATCH

这个执行器(Executor)将会批量执行更新语句,如果SELECT在更新语句中被执行,要根据需要进行区分,确保动作易于理解。

译者注:

1. 使用ExecutorType.SIMPLE session = factory.openSession(ExecutorType.SIMPLE);

对于XML映射配置文件中定义的所有SQL语句在执行时都由Connection创建一个

PreparedStatement对象来执行。即使在执行同一条SQL语句,也要创建一个新的

PreparedStatement对象。如:

执行N次session.selectOne("selectBlog", #{id}),就要创建N个PreparedStatement 对象,了解一下数据库引擎工作原理,就知道这样会带来不小的开销。

  1. 使用ExecutorType.REUSE

session = factory.openSession(ExecutorType.REUSE);

对于XML映射配置文件中定义的所有SQL语句在执行时都由Connection创建一个

PreparedStatement对象来执行。但MyBatis会把每条SQL语句的PreparedStatement对象缓存起来,等到下次再执行相同的SQL语句,则使用缓存的PreparedStatement对象。如,执行N次 session.selectOne("selectBlog", #{id}),只是创建一个PreparedStatement对象。

  1. 使用ExecutorType.BATCH

执行批量更新,把要进行批量更新的SQL语句作为整体进行打包、编译、优化等,可减少网络流量等。

  • 注意:SqlSessionFactory 中还有一个方法我们没有提到,那就是getConfiguration()。这个方法返回一个能够在运行期通过反射的方式得到的Configuration 实例。
  • 注意: 如果您已经使用过以前的MyBatis 版本,您可能会回想起会话、事务和批量处理都是分开的。现在不再是这个样子,这三者都完美地包括在一个sesson 作用域里。您不需要分开处理事务和批量处理就能获得它们的全部好处。

SqlSession

正如前面提到的,SqlSession实例是MyBatis里最强大的类。SqlSession实例里您会找到所有的执行语句、提交或者回滚事务、获得mapper实例的方法。

在SqlSession类里有超过20个方法,现在让我们把它们拆分成容易理解的分组。

语句执行方法组(Statement Execution Methods)

这些方法用来执行定义在SQL映射XML文件中的select , insert,update 和delete语句。它们都很好理解,执行时使用语句的ID和并传入参数对象(基本类型,javaBean,POJO或者 Map)。

Object selectOne(String statement, Object parameter) List selectList(String statement, Object parameter) int insert(String statement, Object parameter) int update(String statement, Object parameter) int delete(String statement, Object parameter)

selectOne方法和selectList方法的不同之处是selectOne必须返回一个对象,如果返回一个以上的对象或者返回0个对象,将会抛出异常。如果不能预知会返回多少对象,最好使用

selectList方法。如果您只是检测一下是否存在一个对象,可以使用SQL语法中的COUNT返回一个计数(0或者1)。因为不是所有语句都需要传入参数的,因此这些方法可以重载为不需要参数的版本。

Object selectOne(String statement) List selectList(String statement) int insert(String statement) int update(String statement) int delete(String statement)

最后,它们还有三个select方法的高级版本,允许您限定返回行记录的范围或者提供自定义

的结果处理逻辑。这三个方法一般用来处理大量的数据集

List selectList

(String statement, Object parameter, RowBounds rowBounds) void select

(String statement, Object parameter, ResultHandler handler) void select

(String statement, Object parameter, RowBounds rowBounds, ResultHandler handler)

RowBounds参数促使MyBatis跳过指定数量的记录,或者返回指定数量的结果集。RowBounds 类有一个构造函数,设定偏移量或者指定条数,如果没进行设置,那就没有限制了。

int offset = 100; int limit = 25;

RowBounds rowBounds = new RowBounds(offset, limit);

不同的驱动程序在这方面可以达到的效率是不同的。为了得到最好性能,使用

SCROLL_SENSITIVE 或者 SCROLL_INSENSITIVE参数设置来返回结果集(换名话说,不要使用

FORWARD_ONLY)。

(译者注)默认情况下获得的结果集是不能更新的,且只有一个向前移动的光标。下面

是一个可更新可滚动的结果集:

Statement stmt = con.createStatement(

ResultSet.TYPE_SCROLL_INSENSITIVE,

ResultSet.CONCUR_UPDATABLE); ResultSet rs = stmt.executeQuery("SELECT a, b FROM TABLE2");

ResultHandler参数允许您按照自己喜欢的方式对每条记录进行操作。您能把每条记录加入一个list、Map或者Set,或者只是进行数量统计。您能让ResultHandler做许多事情,它同样被MyBatis用来生成结果集列表(result set lists)。

这个接口非常简单:

package org.mybatis.executor.result; public interface ResultHandler {

void handleResult(ResultContext context);

}

ResultContext参数允许您访问结果对象本身,创建一定数量的结果对象,或者使用返回值是Boolean 的stop()方法来停止加载更多的结果集。

事务控制方法组(Transaction Control Methods)

有四个控制事务作用域的方法,当然,如果您使用了自动提交或者正在使用的是外部事务管理器,那这四个方法就没什么作用。然而,如果您使用由Connection 实例管理的JDBC的事务管理器,那这四个方法就非常管用:

void commit() void commit(boolean force) void rollback()

void rollback(boolean force)

默认地MyBatis不会自动提交,除非它检测到数据库被insert,update或者delete改变。

如果在哪里作了修改而没有使用这些方法,那么,您需要传入true到commit和rollback方法,以保证它会提交。注意:您依然不能把一个会话或者一个使用外部事务的管理器强制设成自动提交模式。大部分时间,您不需要调用rollback()方法,因为如果您不调用commit 方法的话,

MyBatis会为您进行回滚。然而,如果您需要在一个有多个提交或者多个回滚可能的会话中获得更好(更细粒度)的控制,那么可以使用rollback选项来实现

清除会话层缓存(Clearing the Session Level Cache)

void clearCache()

SqlSession实例有一个本地缓存,这个缓存在每次提交,回滚和关闭时进行清除。如果不想在每次提交或者回滚时都清空缓存,可以明确地调用clearCache()方法来关闭。

确保SqlSession 已经关闭(Ensuring that SqlSession is Closed)

void close()

最重要的事情是要确保关闭已经打开的会话,而最好的方式来保证是用下面的模式:

SqlSession session = sqlSessionFactory.openSession(); try {

// following 3 lines pseudocod for “doing some work” session.insert(…); session.update(…); session.delete(…);

session.commit();

} finally {

session.close(); }

注意: 就像SqlSessionFactory,您能够使用SqlSession的getConfiguration()方法来获得Configuration 的实例。

Configuration getConfiguration()

使用 Mappers

T getMapper(Class type)

虽然前面提到的insert, update, delete 和 select方法很强大,但同时,它们也有点冗

长,且不是类型安全的,IDE或者单元测试也帮助发现不了其中的错误。我们在文章最开始就看到过使用Mappers的例子,我们可以回顾一下。

因此,一个最常用的方式是使用Mapper接口来执行映射语句。一个Mapper接口定义的方法

要与SqlSession执行的方法相匹配,即Mapper接口方法名与映射SQL文件中的映射语句ID相同。下面例子演示了这种用法。

public interface AuthorMapper {

// (Author) selectOne(“selectAuthor”,5);

Author selectAuthor(int id);

// (List) selectList(“selectAuthors”)

List selectAuthors();

// insert(“insertAuthor”, author) void insertAuthor(Author author);

// updateAuthor(“updateAuhor”, author) void updateAuthor(Author author);

// delete(“deleteAuthor”,5) void deleteAuthor(int id);

}

总体来说,每个Mapper接口方法签名,都要与一个SqlSession 的方法相对应,方法名也要与映射语句的ID相对应。

另外,返回的类型要与期望的结果类型一致。支持返回所有类型,包括:基本数据类型,

Maps,POJOs和JavaBeans。

  • Mapper 接口不需要实现任何接口或者继承任何类,只要方法签名能够唯一地与映射语句相对应就可以了。
  • Mapper 接口可以继承其它接口。当您使用XML绑定Mapper接口时,要确保在同一命名空间里有您使用的语句。唯一的限制是不能同时有相同的方法签名在两个不同层次的接口。

您能够传递多个参数给mapper方法,如果这么做,要它们默认的参数列表位置来命名,例如这样:#{1}, #{2}等,如果您想改变参数的名字(多参数情况下),那您可以使用

@Param(“paramName”) 注解的方式来传入参数。

您同样也可以传递一个RowBounds实例到方法,用来限定查询结果范围。

Mapper 注解

从很早的时候,MyBatis就使用XML驱动的框架,基于XML配置,映射语句都定义在

XML中。在MyBatis3中,还有新的可选方案。MyBatis3建立于广泛且强大的java配置API之

上。java配置API是基于XML的MyBatis配置的基础,同时也是基于注解的配置基础。注解提供了一个简单的方式来执行简单映射语句而不引入大量的开销。

  • 注意: 很不幸,java注解在表现力与灵活性上是有限的。尽管花了很多时间来研究,设计与试验,但是强大的MyBatis映射不能够建立在注解之上。C#属性则不会有这种限制。虽然如此,基于注解的配置并非没有好处的。

注解配置属性如下表:

注解

使用范围

等同XML

描述

@CacheNamespace

Class

在指定的命令空间里配置缓存,如类里。

属性有: implementation, eviction,

 

 

 

 

flushInterval, size 和 readWrite.

@CacheNamespaceRef

Class

引用另一个命名空间的缓存。

属性:value:命名空间名 (也就是完全类路径名).

@ConstructorArgs

Method

收集一组结果集传入到一个结果对象的构造器。

属性:value:参数对象的数组。

@Arg

Method

单个构造器参数,是ConstructorArgs 集合的一部分。

属性: Id, column, javaType, jdbcType,typeHandler。

这个id 属性值是一个字符串类型的值,

用于标识这个属性,并被用来进行比

较。类似于这个XML 元素(原文:The id attribute is a boolean value that identifies the property to be used for comparisons, similar to the XML element.说是 boolean值应该有误)。

@TypeDiscriminator

Method

一组cases 值,用来决定哪一个结果会被映射。

属性: column, javaType, jdbcType, typeHandler, cases。 cases 属性是一个Cases数组 。

@Case

Method

单个值的case ,与映射对应。属性:value, type,results。

Results属性是一个结果集数组,因此 Case注解与ResultMap很相似,由下面的Results注解指定。

@Results

Method

结果映射集。其中包含如何将结果列映射到属性或者字段的详细信息。

属性:

Value:一个结果集注解的数组。

@Result

Method

一个列和属性或字段之间的单个结果映射。

属性:id, column, property,

javaType, jdbcType,typeHandler, one, many。

这个id 属性值是一个字符串类型的值,

用于标识这个属性,并被用来进行比

较。类似于这个XML 元素。one 属性是单个association,与

元素相似。它们的命名要避免与类名相冲突。

 

@One

Method

映射到单个复杂类型属性。

属性:select:映射语句全称(也就是映射方法),能够加载一个相应类型的实例。

注意:您会发现连接映射没有被注解API 支持。这就是注解的局限性,它不允许循环引用。

@Many

Method

映射到一个复杂类型的集合属性。

属性:select:映射语句全称(也就是映射方法),能够加载一个相应集合的实例。

注意:您会发现连接映射没有被注解API 支持。这就是注解的局限性,它不允许循环引用。

@Options

Method

映射语句属性

这个注解提供访问各种开关和配置选项。Options注解提供一致的和清晰的方式来访问这些开关与配置。

属性:useCache=true, flushCache=false, resultSetType=FORWARD_ONLY, statementType=PREPARED, fetchSize=-1, timeout=-1, useGeneratedKeys=false, keyProperty=“id”。

理解java注解是很重要的,它没有指定“null”作为一个值的方式。因此,一旦使用Options注解,您的语句将使用所有默认选项值。注意默认的值以避免出现不是您所期待的行为结果。

@Insert

@Update

@Delete

@Select

Method

其中的每一个注解都代表着要被执行的实际SQL。它们都可以传入一个字符串数组(或者单个字符串)。如果传递了一个字符串数组,它们就会使用一个空格将这些字符串分开,然后连接在一起。

这样帮助在java代码中生成SQL语句时避免空格丢失问题。当然如果您喜欢,传入一条字符串也可以。

属性:value:传入的字符串数组。这些数组将形成单一SQL语句。

@InsertProvider

@UpdateProvider

@DeleteProvider

@SelectProvider

Method

这些可选的SQL 注解允许您在运行的过程中,执行指定类和方法返回的SQL 。当执行映射语句时,MyBatis 将会实例

化provider指定的类,执行其中的方法。这方法可选地接受参数对象作为它唯一的参数,但必须只指定一个参数或

 

 

 

者不指定任何参数。

属性:type,method。

type 属性指定类的全名,method属性指定这个类中的方法名称。注意:下一章中,我们会讨论SelectBuilder 类,它可以用一种更清晰、更易读的方式生成动态SQL。

@Param

Parameter

N/A

如果您的映射方法有多个参数,那这个注解能为方法的每个参数设定一个参数名称。否则,多个参数会使用它们的顺序号命名(不包括任何RowBounds参

数),例如:#{1},#{2}等等,这是默

认的方式。如果使用@Param(“person

“ ) , 则这个参数就被叫做

#{person}。

SelectBuilder

Java开发人员最讨厌的事情就是不得不在java代码中嵌入SQL语句。通常这样做的原因是 SQL必须动态生成,要不然,您可以把SQL定义在外部文件或者存储过程中。正如您所看到的, MyBatis在它的XML映射功能中对动态SQL的生成有一个很强大的解决方案。然而,有时候还是不得不在java代码中生成SQL语句字符串,这种情况下,MyBatis还有一个功能可以帮助到您,用于减少加号、引号、换行、格式问题和在嵌套条件中处理额外的逗号与AND连词……事实上,在java中动态生成SQL代码可以说是一场真正的噩梦。

MyBatis3引入一个稍微有点不同的概念来处理这个问题。我们可以生成一个类的实例,然后调用这个实例的方法来一步生成一条SQL语句,最终的SQL看起来更像是java代码而不像SQL代码。我们正在尝试不同的东西,最终结果接近一种领域特定语言(DSL),这也是java目前形态要达到的目标。

SelectBuilder 的秘密

SelectBuilder类并没有什么神奇的地方,如果您不了解它如何工作,它对我们也没任何好处。SelectBuilder使用一组静态导入方法和一个ThreadLocal变量来启用一个能够很容易地组合条件并会注意所有SQL格式的语法。例如:

public String selectBlogsSql() {

BEGIN(); // Clears ThreadLocal variable

SELECT("*");

FROM("BLOG"); return SQL();

}

这只是一个您可能只选择静态生成SQL的简单例子,下面是一个更复杂的例子:

private String selectPersonSql() {

BEGIN(); // Clears ThreadLocal variable

SELECT("P.ID, P.USERNAME, P.PASSWORD, P.FULL_NAME");

SELECT("P.LAST_NAME, P.CREATED_ON, P.UPDATED_ON");

FROM("PERSON P");

FROM("ACCOUNT A");

INNER_JOIN("DEPARTMENT D on D.ID = P.DEPARTMENT_ID");

INNER_JOIN("COMPANY C on D.COMPANY_ID = C.ID");

WHERE("P.ID = A.ID");

WHERE("P.FIRST_NAME like ?");

OR();

WHERE("P.LAST_NAME like ?");

GROUP_BY("P.ID");

HAVING("P.LAST_NAME like ?");

OR();

HAVING("P.FIRST_NAME like ?");

ORDER_BY("P.ID"); ORDER_BY("P.FULL_NAME");

return SQL(); }

如果使用字符串连接,上面的SQL语句就会像下面这样来生成:

"SELECT P.ID, P.USERNAME, P.PASSWORD, P.FULL_NAME, "

"P.LAST_NAME,P.CREATED_ON, P.UPDATED_ON " +

"FROM PERSON P, ACCOUNT A " +

"INNER JOIN DEPARTMENT D on D.ID = P.DEPARTMENT_ID " +

"INNER JOIN COMPANY C on D.COMPANY_ID = C.ID " +

"WHERE (P.ID = A.ID AND P.FIRST_NAME like ?) " +

"OR (P.LAST_NAME like ?) " +

"GROUP BY P.ID " +

"HAVING (P.LAST_NAME like ?) " +

"OR (P.FIRST_NAME like ?) " +

"ORDER BY P.ID, P.FULL_NAME";

如果您喜欢上面这种语法,那您仍然可以使用它,但是它很容易出错,要注意在每行结束的

地方都加一个空格。即使您喜欢上面的语法,下面的例子毫无争辩地比在java代码中连接字符串更简单:

private String selectPersonLike(Person p){

BEGIN(); // Clears ThreadLocal variable

SELECT("P.ID, P.USERNAME, P.PASSWORD, P.FIRST_NAME, P.LAST_NAME"); FROM("PERSON P"); if (p.id != null) {

WHERE("P.ID like #{id}");

}

if (p.firstName != null) {

WHERE("P.FIRST_NAME like #{firstName}");

}

if (p.lastName != null) {

WHERE("P.LAST_NAME like #{lastName}");

}

ORDER_BY("P.LAST_NAME"); return SQL();

}

上面的例子有什么特别之处呢?如果您仔细观察的话,您会发现您不需要再担心是否会遇外地加入了重复的“AND”关键字,或者考虑要使用“WHERE”还是“AND”或者两者都使用。上面的例子生成一个查询所有PERSON记录的语句,查出与ID匹配的、或者与firstName匹配的、或者与 lastName匹配的记录,或者是与上述三种任何组合条件相匹配的记录。SelectBuilder 会知道哪里需要添加“WHERE”,哪里需要使用“AND”,以及会注意字符串的连接。最重要的是,它几乎不管您调用这些方法的顺序(只有一个例外就是使用OR()方法)。

您可能已经关注到这两个方法:BEGIN() and SQL()。概括地说,每一个SelectBuilder方法都要以BEGIN()开头,以SQL()结束,这也是生成SQL的范围。在BEGIN()和SQL()之间,您可以提取SelectBuilder的方法来一步一步分解您的逻辑。BEGIN()方法清除了ThreadLocal变量以保证不会有先前残留的内容,SQL()方法装配自从调用BEGIN()后所有SelectBuilder方法生成的

SQL语句。注意:BEGIN()方法有一个同义词方法RESET(),它们做相同的事,在某些情况下使用

RESET()可读性会更强。

要像上面那样使用SelectBuilder 的方法,您简单地只要使用静态导入就可以了,如:

import static org.mybatis.jdbc.SelectBuilder.*;

一旦被导入,您就能够使用SelectBuilder类的所有方法。SelectBuilder类的方法如下:

Method

Description

BEGIN() / RESET()

这个方法清空SelectBuilder 的ThreadLocal 状态,并准备好生成

一个新的语句。BEGIN()方法放在句子的开头可读性比较好。RESET() 方法在执行过程中由于一些原因(可能在某些情况下,执行逻辑需要一个完全不同的语句)清空语句后使用,可读性比较好

SELECT(String)

开始或者追加一个SELECT 子句。也可被多次调用,参数将会被追加到SELECT 语句中。参数通常是用一个逗号分隔的列名和别名字符串,或者是任何驱动器支持的语法。

SELECT_DISTINCT(String)

开始或者追加一个SELECT 子句,在生成的查询语句中增加关键字 “DISTINCT”。也可被多次调用,参数将会被追加到SELECT 语句中。参数通常是用一个逗号分隔的列名和别名字符串,或者是任何驱动器支持的语法。

FROM(String)

开始或者追加一个FROMT 子句。也可被多次调用,参数将会被追加

到FROM 语句中。参数通常是一个table名或者别名,或者是驱动器支持的语法。

JOIN(String)

INNER_JOIN(String)

LEFT_OUTER_JOIN(String)

增加一个恰当类型的JOIN子句,这由调用的方法决定。参数是标准的由列和条件连接在一起的JOIN子句。

Method

Description

RIGHT_OUTER_JOIN(String)

 

WHERE(String)

追加一个新的条件子句,可以被多次调用,每个子句使用AND来连结。如果使用 OR() ,则使用OR来分隔。

OR()

使用OR来分隔WHERE条件子句,可以被多次调用,但是对某一行调用超过一次则可能生成错误的SQL语句。

AND()

使用AND来分隔WHERE条件子句,可以被多次调用,但是对某一行

调用超过一次会产生错误的SQL语句。因此WHERE和HAVING都会自动的添加AND来连结语句。这个方法的使用非常罕见,包含这个方法的原因可能就为了生成SQL语句语法上的完整性。

GROUP_BY(String)

追加一个GROUP BY子句,可以被多次调用,每个子句由逗号连接。

HAVING(String)

追加一个HAING条件子句,可以被多次调用,每个子句由AND连接。如果使用 OR() ,则使用OR来分隔。

ORDER_BY(String)

追加一个ORDER BY子句,可以被多次调用,每个子句由逗号分隔。

SQL()

SQL()方法返回生成的SQL语句,并且重新设置SelectBuilder 状态(只有BEGIN() 或者RESET()已经被调用时)。因此,这个方法只能被调用一次。

SqlBuilder

与SelectBuilder类似,MyBatis 也包含了一个通用的SqlBuilder类,它包含了

SelectBuilder的所有方法,同时也有一些针对inserts, updates, 和deletes 的方法。这个类在 DeleteProvider 、nsertProvider和UpdateProvider (以及SelectProvider )里生成SQL 语句时非常有用。

要像上面的例子那样使用SqlBuilder,同样只要使用静态导入就可了,例如:

import static org.mybatis.jdbc.SqlBuilder.*;

SqlBuilder 包含了SelectBuilder的所有方法, 同时还有一些额外的方法:

Method

Description

DELETE_FROM(String)

开始一个delete语句,同时指定要删除的表。通常它应该在WHERE 语句前被调用。

INSERT_INTO(String)

开始一个插入语句,同时指定插入的表。它应该在一个或者多个

VALUES()方法之前被调用。

SET(String)

追加“set”列表到更新语句中。

UPDATE(String)

开始一个更新语句,同时指定更新的表。在一个或者多个SET()方法前被调用,通常情况下,WHERE()方法连接在它后面。

VALUES(String, String)

追加一个插入语句,第一个参数是要插入的一列名,第二个参数是要插入的值。

下面是些例子:

public String deletePersonSql() {

BEGIN(); // Clears ThreadLocal variable

DELETE_FROM("PERSON"); WHERE("ID = ${id}"); return SQL();

}

public String insertPersonSql() {

BEGIN(); // Clears ThreadLocal variable

INSERT_INTO("PERSON");

VALUES("ID, FIRST_NAME", "${id}, ${firstName}"); VALUES("LAST_NAME", "${lastName}"); return SQL();

}

public String updatePersonSql() {

BEGIN(); // Clears ThreadLocal variable UPDATE("PERSON");

SET("FIRST_NAME = ${firstName}"); WHERE("ID = ${id}"); return SQL();

}

 

结束语

MyBatis是一款方便、简单又不失灵活的持久层框架。至此,官方英文版用户指南也结束了,感谢GOOGLE,因为在翻译本文档时,GOOGLE翻译帮了大忙。

但是,用户指南对MyBatis的介绍还远远不够。接下来要靠我们继续去学习、使用和挖掘了,这样才能真正地了解或掌握MyBatis。

以后的附录章节是译者学习和使用MyBatis的一些笔记记录,供大家参考,与大家共勉。

附录1 对象模型

为保持与官方用户指南的连贯性,并且可以使用文中的例子,帮助理解,仍然使用官方文档中的对象模型:包括由一个作者写的一个博客,一个博客有许多文章(Post,帖子),每个文章由0个或者多个评论和标签组成。

首先建立对象模型。

Author.java

package org.mybatis.model; public class Author { private Integer id; private String username; private String password; private String email; private String bio;

//省略get和set方法

}

Blog.java

package org.mybatis.model; public class Blog { private Integer id; private String title; private Integer authorId;

//省略get和set方法

}

MyBatis 3 - User Guide

Post.java

package org.mybatis.model; public class Post { private Integer id; private Integer blogId; private Integer authorId; private String createdOn; private String section; private String subject; private String body;

//省略get和set方法}

Comment.java

package org.mybatis.model; public class Comment { private Integer id; private Integer postId; private String name; private String comment;

//省略get和set方法

}

Tag.java

package org.mybatis.model; public class Tag {

71

 

private Integer id; private String name; public Integer getId() {

//省略get和set方法

}

要使用MyBatis,首先要有一个数据库,这里选用Java6自带的JavaDB,也就是derby数据库。使用JavaDB内嵌模式,可以不需要独立维护一个数据库服务器。

附录2就是使用JavaDB来搭建数据库环境的过程。

附录2创建数据库

JavaDB作为JDK 6内嵌的数据库,使得程序员不再需要耗费大量精力安装和配置数据库,就能进行安全、易用、标准、并且免费的数据库编程。

在安装JDK6的时候,也同时会安装JavaDB,并包含了一些实用工具,我们可以直接使用

JavaDB自带的工具来启动、查看数据库等。特别是ij工具,可以用来创建数据库、创建表、执行SQL语句等等操作。JavaDB有两种工作模式:内嵌模式和网络服务器模式。这里选用网络服务器模式,独立启动JavaDB,并用ij工具进行连接操作。但是在程序中使用derby数据库时是选用内嵌模式,这样就不用启动derby服务器了,请留意后面章节的连接数据库配置。

对于JavaDB的介绍可参考其它相关资料。下面开始创建数据库、创建表和插入准备数据…… 第一步,双击JavaDB\bin\startNetworkServer.bat,启动JavaDB数据库。

第二步,双击JavaDB\bin\ ij.bat,启动ij工具。

第三步,连接并创建数据库。数据库名:blogDB。

ij> connect

'jdbc:derby://localhost:1527/blogDB; create=true';

使用参数create=true表明,连接数据库服务器并创建一个数据库。创建数据库以后连接就不再需要参数create=true了,但是使用也无妨。

ij> connect

'jdbc:derby://localhost:1527/blogDB';

第四步,执行SQL脚本文件,创建表、插入数据。

ij> connect 'jdbc:derby://localhost:1527/blogDB; user=lory';

这次使用了参数user=lory,本次连接使用用户名lory来进行连接。指定用户名也就是指定了模式(schema),那么可以直接使用表名来访问,而不需要用schema.tablename的形式,例如,不指定用户名来连接服务器,要访问lory用户创建的blog表,就需要通过lory.blog的方式来访问。这里的用户名也是MyBatis配置数据库连接参数所指定的用户名。

执行SQL脚本。如执行事先准备好创建作者的脚本Author.sql:

ij> run 'E:\blogDB\Author.sql';

Author.sql:

drop table author; create table author

( id int not null,

username varchar(10) not null, password varchar(30) not null, email varchar(30), bio varchar(30)

);

insert into author(id,username,password,email,bio) values(1,'user1','user1','[email protected]','guy');

insert into author(id,username,password,email,bio) values(2,'user2','user2','[email protected]','guy');

insert into author(id,username,password,email,bio) values(3,'user3','user3','[email protected]','guy');

insert into author(id,username,password,email,bio) values(4,'user4','user4','[email protected]','guy');

insert into author(id,username,password,email,bio) values(5,'user5','user5','[email protected]','guy');

insert into author(id,username,password,email,bio) values(6,'user6','user6','[email protected]','guy');

同理,我们再创建四张表:blog,post,comment,tag。

Blog.sql:

drop table blog; create table blog

(

id int not null primary key, title varchar(30) default 'My Blog', author_id int not null

);

insert into blog(id,title,author_id) values(1,'just fun',1);

MyBatis 3 - User Guide

insert into blog(id,title,author_id) values(2,'just funny',2); insert into blog(id,author_id) values(3,3); insert into blog(id,author_id) values(4,4); insert into blog(id,title,author_id) values(5,'hello one',5); insert into blog(id,title,author_id) values(6,'hello two',6);

Post.sql:

drop table post; create table post

( id int not null, blog_id int not null, author_id int not null, created_on varchar(30), section varchar(30), subject varchar(30), body varchar(100)

);

insert into post(id,blog_id,author_id,created_on,section,subject,body) values(1,1,1,'2010-08-04','photo','ddd','nothing');

insert into post(id,blog_id,author_id,created_on,section,subject,body) values(2,1,1,'2010-08-05','photo','hello','nothing too');

insert into post(id,blog_id,author_id,created_on,section,subject,body) values(3,1,1,'2010-08-06','photo','ddfdfdd','also nothing'); insert into post(id,blog_id,author_id,created_on,section,subject,body) values(4,2,2,'2010-08-06','photo','hi','nothing more');

Comment.sql:

drop table comment; create table comment

75

( id int not null, post_id int not null, name varchar(30), comment varchar(50)

)

Tag.sql:

drop table tag; create table tag

( id int not null, name varchar(30)

)

至此,一个可供使用的数据库已经建成。上面创建的表都比较简单,没有创建外键、索

引等。接下来就开始MyBatis之旅。

附录3 MyBatis实例

简单select

一切从简单开始,下面是一个Mapper XML配置,包含了基本的部分。

Sqlconfig.xml

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="org/mybatis/data/sqlconfig.properties" /> <typeAliases>

<typeAlias alias="Author"type="org.mybatis.model.Author"/>

<typeAlias alias="Blog"type="org.mybatis.model.Blog"/>

<typeAlias alias="Comment"type="org.mybatis.model.Comment"/>

<typeAlias alias="Post"type="org.mybatis.model.Post"/>

<typeAlias alias="Tag"type="org.mybatis.model.Tag"/>

typeAliases>

<environments default="development">

<environment id="development">

<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="org/mybatis/data/BlogMapper.xml"/>

<mapper resource="org/mybatis/data/AuthorMapper.xml"/>

<mapper resource="org/mybatis/data/PostMapper.xml"/>

mappers>

configuration>

sqlconfig.properties

#derby Network Server

#driver=org.apache.derby.jdbc.ClientDriver #url=jdbc:derby://localhost:1527/blogDB;

#derby embedded model

driver=org.apache.derby.jdbc.EmbeddedDriver

#url=jdbc:derby:blogDB;

url=jdbc:derby:D:\\Program Files\\Java\\JavaDB\\bin\\blogDB;

username=lory password=lorry

说明:这里使用derby数据库,它有两种配置方式。

网络服务器模式使用配置:

driver=org.apache.derby.jdbc.ClientDriver url=jdbc:derby://localhost:1527/blogDB;

内嵌模式使用配置:

driver=org.apache.derby.jdbc.EmbeddedDriver

#url=jdbc:derby:blogDB;

url=jdbc:derby:D:\\Program Files\\Java\\JavaDB\\bin\\blogDB;

url直接指向本地已经建好的blogDB数据库。程序中使用内嵌模式就不需要独立启动derby数

据库服务器了。此时的derby数据库只接受一个连接,但作为练习,是没有问题的。

BlogMapper.xml

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<select id="selectBlog_by_id"parameterType="int"resultType="Blog"> select * from Blog where id = #{id}

select>

<select id="selectBlog_by_id_Map" parameterType="HashMap"resultType="Blog"> select * from Blog where id = #{id}

select>

 

接着创建XML配置管理器。

SqlMapperManager

package org.mybatis.service;

import java.io.IOException; import java.io.Reader;

import org.apache.ibatis.io.Resources;

import org.apache.ibatis.session.SqlSessionFactory; import org.apache.ibatis.session.SqlSessionFactoryBuilder;

public class SqlMapperManager

{ private static SqlSessionFactory factory= null;

private static String fileName= "org/mybatis/data/Sqlconfig.xml";

private SqlMapperManager()

{ }

public static void initMapper(String sqlMapperFileName)

{ fileName= sqlMapperFileName;

}

public static SqlSessionFactory getFactory()

{ try { if (factory== null)

{

Reader reader = Resources

.getResourceAsReader(fileName);

SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();

factory= builder.build(reader);

builder = null;

}

}

catch (IOException e)

{

e.printStackTrace(); return null;

}

return factory;

}

}

创建测试类:

SimpleMapper.java

package org.mybatis.action; import java.util.HashMap;

import org.apache.ibatis.session.SqlSession; import org.apache.log4j.BasicConfigurator; import org.mybatis.model.Blog;

import org.mybatis.service.SqlMapperManager; public class SimpleMapper {

/**

* @param args

*/ public static void main(String[] args)

{

BasicConfigurator.configure();

SqlSession session = null; Blog blog = null; try {

SqlSessionFactory factory = SqlMapperManager.getFactory(); if (factory == null)

{

System.out.println("get SqlSessionFactory failed."); return;

} session = factory.openSession();

HashMap paramMap = new HashMap();

paramMap.put("id", 2);

Blog myBlog = new Blog(); myBlog.setId(3);

blog = (Blog) session.selectOne(

"selectBlog_by_id", 1);

pringBlog(blog);

blog = (Blog) session.selectOne(

"selectBlog_by_id_Map", paramMap);

pringBlog(blog);

blog = (Blog) session.selectOne(

"selectBlog_by_bean", myBlog);

pringBlog(blog);

}

catch(Exception e)

{

e.printStackTrace(); } finally

{ session.close();

}

}

public static void pringBlog(Blog blog)

{ if (blog != null)

{

System.out.println("ID:" + blog.getId());

System.out.println("title:" + blog.getTitle());

System.out.println("authorID:" + blog.getAuthorId());

} else

{

System.out.println("blog=null"); }

}

}

运行SimpleMapper.java结果,控制台输出信息:

D:\Programs\workspace\MyBatis3

0 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource

- PooledDataSource

forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource

- PooledDataSource

forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource

- PooledDataSource

forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource

- PooledDataSource forc

efully closed/removed all connections.

6787 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Created connection 24585668.

6866 [main] DEBUG java.sql.Connection - ooo Connection Opened

10158 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select * from Blog where id = ?

10158 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 2(Integer)

10517 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

10517 [main] DEBUG java.sql.ResultSet - <== Row: 2, just funny, 2

ID:2

title:just funny authorID:null

11266 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select * from Blog where id = ?

11266 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 1(Integer)

11266 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

11266 [main] DEBUG java.sql.ResultSet - <== Row: 1, just fun, 1

ID:1 title:just fun authorID:null

11283 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select * from Blog where id = ?

11283 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer)

11283 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

11298 [main] DEBUG java.sql.ResultSet - <== Row: 3, My Blog, 3

ID:3 title:My Blog authorID:null

11298 [main] DEBUG java.sql.Connection - xxx Connection Closed

11298 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection 24585668 to pool.

从上面可以看出,要传递什么参数有多种选择。如果是要传递多个参数,如insert或 update,在没有模型(javaBean)的情况下,使用HashMap应该是比较方便的。HashMap同样可以保存映射结果。.虽然这对许多场合下有用,但是HashMap 却不是非常好的域模型。

从结果可以看到,authorID:null,因为数据库中的author_id与Blog没有匹配的字段,因此结果为null,并且使用“*”来代替查询的字段也是很拙劣的做法。解决字段不匹配的问题有两种方式,如把BlogMapper.xml修改如下:

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResultMap"type="Blog">

<id property="id"column="id"/>

<result property="title"column="title"/>

<result property="authorId"column="author_id"/> resultMap>

<select id="selectBlog_use_as"parameterType="HashMap"resultType="Blog"> select id , title, author_id as authorid from Blog where id = #{id}

select>

<select id="selectBlog_use_resultMap"parameterType="HashMap"resultMap="blogResultMap"> select id , title, author_id from Blog where id = #{id}

select>

mapper>

然后在代码里调用:

HashMap paramMap = new HashMap(); paramMap.put("id", 2);

Blog myBlog = new Blog(); myBlog.setId(3);

blog = (Blog) session.selectOne("selectBlog_use_as", myBlog);

pringBlog(blog);

blog = (Blog) session.selectOne("selectBlog_use_resultMap", paramMap); pringBlog(blog);

控制台输出结果:

9611 [main] DEBUG java.sql.Connection - ooo Connection Opened

12903 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select id , title, author_id as authorid from Blog where id = ?

12903 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 9(Integer)

13169 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHORID

13169 [main] DEBUG java.sql.ResultSet - <== Row: 9, I Love Photh too, 3

ID:9

title:I Love Photh too authorID:3

13294 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select id , title, author_id from Blog where id = ?

13294 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 2(Integer)

13294 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

13294 [main] DEBUG java.sql.ResultSet - <== Row: 2, just funny, 2

ID:2

title:just funny authorID:2

13294 [main] DEBUG java.sql.Connection - xxx Connection Closed

13294 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection 9936523 to pool.

其实,parameterType="HashMap"不是必须的,MyBatis有足够的信息可以知道输入的对

象是什么。注意selectBlog_use_resultMap,虽然设置的是parameterType="HashMap",但实际传入的是一个Blog对象。

update,delete,insert

对于update,delete,insert,看下面的例子:

映射配置文件:

<update id="updateBlog_use_bean"statementType="PREPARED"parameterType="Blog"> update blog set title= #{title}, author_id=#{authorId} where id = #{id}

update>

<delete id="deleteBlog_use_bean"statementType="PREPARED"parameterType="Blog"> delete from blog where id = #{id}

delete>

<insert id="insertBlog_user_bean"statementType="PREPARED"parameterType="Blog"> insert into blog(id, title, author_id) values(#{id}, #{title}, #{authorId}) insert>

调用代码:

Blog myBlog = new Blog(); myBlog.setId(3); myBlog.setTitle("I Love Photh"); myBlog.setAuthorId(3);

session.update("updateBlog_use_bean", myBlog); session.delete("deleteBlog_use_bean", myBlog); session.insert("insertBlog_user_bean", myBlog);

控制台输出结果:

2476 [main] DEBUG java.sql.PreparedStatement - ==> Executing: update blog set title= ?, author_id=? where id = ?

2476 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: I Love Photh(String),

3(Integer), 3(Integer)

2731 [main] DEBUG java.sql.PreparedStatement - ==> Executing: delete from blog where id = ?

2731 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer)

2792 [main] DEBUG java.sql.PreparedStatement - ==> Executing: insert into blog(id, title, author_id) values(?, ?, ?)

2797 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer), I Love

Photh(String), 3(Integer)

默认地MyBatis不会自动提交,最后还需要使用session.commit()来提交修改。但可以调用下面的方法来自动提交。

SqlSession openSession(boolean autoCommit)

自动生成主键

下面是自动生成blog的id的值,生成规则是:当前表中最大的id值加1。映射配置文件:

<insert id="insertBlog_user_autokey"statementType="PREPARED"parameterType="Blog">

<selectKey keyProperty="id"resultType="int"order="BEFORE"> select max(id)+1 from blog

selectKey> insert into blog(id, title, author_id) values(#{id}, #{title}, #{authorId}) insert>

调用代码:

Blog myBlog1 = new Blog(); myBlog1.setTitle("I Love Photh");

myBlog1.setAuthorId(3);

session.insert("insertBlog_user_autokey", myBlog1); session.insert("insertBlog_user_autokey", myBlog1); session.insert("insertBlog_user_autokey", myBlog1);

虽然插入同样的myBlog1,但是myBlog1.id是不一样的。

控制台输出结果:

D:\Programs\workspace\MyBatis3

0 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

16 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

1560 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Created connection 17007273.

1592 [main] DEBUG java.sql.Connection - ooo Connection Opened

2014 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select max(id)+1 from blog 2014 [main] DEBUG java.sql.PreparedStatement - ==> Parameters:

2435 [main] DEBUG java.sql.ResultSet - <== Columns: 1

2435 [main] DEBUG java.sql.ResultSet - <== Row: 7

2575 [main] DEBUG java.sql.PreparedStatement - ==> Executing: insert into blog(id, title, author_id) values(?, ?, ?)

2575 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 7(Integer), I Love

Photh(String), 3(Integer)

2591 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select max(id)+1 from blog 2591 [main] DEBUG java.sql.PreparedStatement - ==> Parameters:

2607 [main] DEBUG java.sql.ResultSet - <== Columns: 1

2607 [main] DEBUG java.sql.ResultSet - <== Row: 8

2607 [main] DEBUG java.sql.PreparedStatement - ==> Executing: insert into blog(id, title, author_id) values(?, ?, ?)

2607 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 8(Integer), I Love

Photh(String), 3(Integer)

2607 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select max(id)+1 from blog 2607 [main] DEBUG java.sql.PreparedStatement - ==> Parameters:

2623 [main] DEBUG java.sql.ResultSet - <== Columns: 1 2623 [main] DEBUG java.sql.ResultSet - <== Row: 9

2623 [main] DEBUG java.sql.PreparedStatement - ==> Executing: insert into blog(id, title, author_id) values(?, ?, ?)

2623 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 9(Integer), I Love

Photh(String), 3(Integer)

2701 [main] DEBUG java.sql.Connection - xxx Connection Closed

2701 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection 17007273 to pool.

默认地MyBatis不会自动提交,最后还需要使用session.commit()来提交修改

处理NULL值

像上面的例子,如果myBlog.setTitle(null)情况会怎样呢?那么程序会报错:

org.apache.ibatis.exceptions.PersistenceException:

### Error updating database. Cause: org.apache.ibatis.type.TypeException: Error setting null parameter. Most JDBC drivers require that the JdbcType must be specified for all nullable parameters. Cause: java.sql.SQLDataException: 尝试从类型“OTHER”的数据值获取类型“VARCHAR” 的数据值。

### The error may involve org.mybatis.model.BlogMapper.insertBlog_user_autokey-Inline ### The error occurred while setting parameters

Note:如果传递了一个空值,那这个JDBC Type 对于所有JDBC 允许为空的列来说是必须的。您可以研读一下关于PreparedStatement.setNull()的JavaDocs 文档。

解决这个问题就需要在参数中指定jdbcType属性,这个属性只在insert,update 或delete 的时候针对允许空的列有用。如:

<insert id="insertBlog_user_autokey"statementType="PREPARED"parameterType="Blog" flushCache="true">

<selectKey keyProperty="id"resultType="int"order="BEFORE"> select max(id)+1 from blog

selectKey> insert into blog(id, title, author_id) values(#{id}, #{title,jdbcType=VARCHAR}, #{authorId})

insert>

然后我们使用ij工具查看一下插入的数据,title值被设置为NULL:

ij 版本 10.5

ij> connect 'jdbc:derby:blogDB;user=lory'; ij> select * from blog;

ID |TITLE |AUTHOR_ID

------------------------------------------------------

  1. |nothing title |1
  2. |just funny |2
  3. |My Blog |3
  4. |My Blog |4
  5. |hello one |5
  6. |hello two |6
  7. |I Love Photh |3
  8. |I Love Photh |3
  9. |I Love Photh too |3
  10. |NULL |3
  11. |NULL |3
  12. |NULL |3

已选择 12 行 ij>

使用接口映射类

BlogMapper.class

对给定的映射语句,使用一个正确描述参数与返回值的接口(如BlogMapper.class),您就能更清晰地执行类型安全的代码,从而避免错误和异常。下在是使用接口的例子:定义接口映射类

package org.mybatis.model;

public interface BlogMapper

{

// public Blog selectBlog_by_bean(Blog blog);

// public void updateBlog_use_bean(Blog blog);

// public void insertBlog_user_bean(Blog blog);

// public void insertBlog_user_autokey(Blog blog); }

调用代码:

/**

* user mapper interface

*/ public static void userMapper(SqlSession session)

{

Blog blog = new Blog(); blog.setTitle("nothing title"); blog.setId(1); blog.setAuthorId(1);

BlogMapper blogMapper = session.getMapper(BlogMapper.class);

Blog blog1 = blogMapper.selectBlog_by_id(1); pringBlog(blog1); blogMapper.updateBlog_use_bean(blog);

blog1 = blogMapper.selectBlog_by_id(1);

pringBlog(blog1);

}

控制台输出结果:

D:\Programs\workspace\MyBatis3

0 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

15 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

15 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

15 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - PooledDataSource forcefully closed/removed all connections.

1217 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Created connection 20474136.

1233 [main] DEBUG java.sql.Connection - ooo Connection Opened

1640 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select * from Blog where id = ?

1640 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 1(Integer)

1796 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

1796 [main] DEBUG java.sql.ResultSet - <== Row: 1, just fun, 1

ID:1 title:just fun authorID:null

1842 [main] DEBUG java.sql.PreparedStatement - ==> Executing: update blog set title= ?, author_id=? where id = ?

1842 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: nothing title(String), 1(Integer), 1(Integer)

1920 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select * from Blog where id = ?

1920 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 1(Integer)

1920 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

1920 [main] DEBUG java.sql.ResultSet - <== Row: 1, nothing title, 1

ID:1

title:nothing title authorID:null

1920 [main] DEBUG java.sql.Connection - xxx Connection Closed

1920 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection 20474136 to pool.

BlogMapper.class

要注意的是,是XML映射配置文件BlogMapper.xml里命名空间指向

的接口:"org.mybatis.model.BlogMapper">

一般来说,一个XML映射配置文件对应一个命名空间,而这个命名空间又对应一个接

口,这样我们就可以定义这样一个接口,并类型安全地执行SQL语句,如:

BlogMapper blogMapper = session.getMapper(BlogMapper.class);

Blog blog1 = blogMapper.selectBlog_by_id(1);

MyBatis启动时首先会根据XML配置文件中的命名空间查找是否存在这个接口类,如果存在则保存在一个名为knownMappers的HashSet中。当我们使用session的方法:

T getMapper(Class type)

的时候,如果type是已知的接口类(knownMappers.contains(type)==true),OK,继续执

行,如果不是已知接口类(knownMappers.contains(type)=false),则抛出如下异常:

org.apache.ibatis.binding.BindingException: Type interface org.mybatis.model.BlogMapperr is not known to the MapperRegistry.

使用Constructor元素

接下来的几节都是resultMap元素下的子元素的应用例子,为了简单,不把所有子元素都应

用在一起,而是一个子元素一个子元素地进行学习使用。

使用Constructor元素是将数据库查询的结果通过构造器注入到结果映射类(JavaBean)

中,可以理解为spring中的构造器注入。实际上在一个映射结果类中很可能没有构造器,或者在一个resultMap中既有构造器映射,又有指定属性映射,会造成配置不统一,反而不清晰,因此这个元素可能会很少使用。

首先在Blog类中增加两个构造函数:

public Blog()

{

}

public Blog(Integer id, String title, Integer authorId)

{ this.id = id; this.title = title;

this.authorId = authorId; }

BlogMapper.xml进行如下配置:

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResultMap"type="Blog">

<constructor>

<idArg column="id"javaType="int"/>

<arg column="title"javaType="String"/>

<arg column="author_id"javaType="int"/>

constructor>

resultMap>

<select id="selectBlog_use_constructor" resultMap="blogResultMap"> select id , title, author_id from Blog where id = #{id}

select>

mapper>

调用代码是:

Blog blog = (Blog)session.selectOne("selectBlog_use_constructor", 3); pringBlog(blog);

控制台输出:

3089 [main] DEBUG java.sql.Connection - ooo Connection Opened

4494 [main] DEBUG java.sql.PreparedStatement - ==> Executing: select id , title, author_id from Blog where id = ?

4494 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer)

4667 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

4667 [main] DEBUG java.sql.ResultSet - <== Row: 3, My Blog, 3

ID:3 title:My Blog authorID:3

4729 [main] DEBUG java.sql.Connection - xxx Connection Closed

4729 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection

15558189 to pool.

使用Association元素

一个作者有一个博客,这是种“has-a”的一对一关系,现在我们使用Association元素把博

客及作者的信息查询出来。这里重现官方教程的例子:修改Blog的域模型:

package org.mybatis.model;

public class Blog { private Integer id; private String title; private Author author; //将原来的authorId换成author

//省略get和set方法

}

BlogMapper.xml配置(1):

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResult"type="Blog"> <id property="id"column="blog_id"/>

<result property="title"column="blog_title"/>

<association property="author"javaType="Author">

<id property="id"column="author_id"/>

<result property="username"column="author_username"/>

<result property="password"column="author_password"/>

<result property="email"column="author_email"/>

<result property="bio"column="author_bio"/> association> resultMap>

<select id="selectBlog_use_association"parameterType="int"resultMap="blogResult"> select

B.id as blog_id,

B.title as blog_title,

A.id as author_id,

A.username as author_username,

A.password as author_password,

A.email as author_email,

A.bio as author_bio

from Blog B left outer join Author A on B.author_id = A.id where B.id = #{id}

select>

mapper>

BlogMapper.xml配置(2):

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResult"type="Blog">

<id property="id"column="blog_id"/>

<result property="title"column="blog_title"/>

<association property="author"column="blog_author_id"javaType="Author" resultMap="authorResult"/>

resultMap>

<resultMap id="authorResult"type="Author">

<id property="id"column="author_id"/>

<result property="username"column="author_username"/>

<result property="password"column="author_password"/>

<result property="email"column="author_email"/>

<result property="bio"column="author_bio"/> resultMap>

<select id="selectBlog_use_association"parameterType="int"resultMap="blogResult"> select

B.id as blog_id,

B.title as blog_title,

A.id as author_id,

A.username as author_username,

A.password as author_password,

A.email as author_email,

A.bio as author_bio

from Blog B left outer join Author A on B.author_id = A.id where B.id = #{id}

select>

mapper>

BlogMapper.xml配置(3):

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResult"type="Blog">

<association property="author"column="author_id"javaType="Author"

select="selectAuthor"/>

resultMap>

<select id="selectAuthor"parameterType="int"resultType="Author">

SELECT * FROM AUTHOR WHERE ID = #{id}

select>

<select id="selectBlog_use_association"parameterType="int"resultMap="blogResult"> SELECT * FROM BLOG WHERE ID = #{id}

select>

mapper>

调用代码:

Blog blog = (Blog)session.selectOne("selectBlog_use_association", 3); printBlogAuthor(blog);

public static void printBlogAuthor(Blog blog)

{

System.out.println("ID:" + blog.getId());

System.out.println("title:" + blog.getTitle());

System.out.println("authorID:" + blog.getAuthor().getId());

System.out.println("authorName:" + blog.getAuthor().getUsername());

System.out.println("authorPassword:" + blog.getAuthor().getPassword());

System.out.println("authorEmail:" + blog.getAuthor().getEmail());

System.out.println("authorBio:" + blog.getAuthor().getBio()); }

控制台输出信息:

ID:3 title:My Blog authorID:3 authorName:user3 authorPassword:user3 authorEmail:[email protected] authorBio:guy

BlogMapper.xml配置(1)和BlogMapper.xml配置(2)的配置是等效的,谁优谁劣见仁见智,或者看具体的项目要求。BlogMapper.xml配置(3)使用了两条查询语句,加载博客信息后再加载作者信息。

对于配置(3),如果"selectBlog_use_association"查询返回N条博客记录,如修改一下 SQL语句:

注:如果SQL语句有特殊符号,需要用括起来。这里的小于号<被认为

是特殊符号,如果不用括起来是执行不了的。而调用代码也相应的改变为:

List blogList = (List)session.selectList("selectBlog_use_association"); printBlogAuthorList(blogList);

public static void printBlogAuthorList(List blogList)

{ for (Blog blog : blogList)

{

System.out.println("===========================");

System.out.println("ID:" + blog.getId());

System.out.println("title:" + blog.getTitle());

System.out.println("authorID:" + blog.getAuthor().getId());

System.out.println("authorName:" + blog.getAuthor().getUsername());

System.out.println("authorPassword:" + blog.getAuthor().getPassword());

System.out.println("authorEmail:" + blog.getAuthor().getEmail());

System.out.println("authorBio:" + blog.getAuthor().getBio());

}

}

控制台输出结果:

3510 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Created connection 22377952.

3510 [main] DEBUG java.sql.Connection - ooo Connection Opened

4931 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM BLOG WHERE ID > 0 and ID < 7

4946 [main] DEBUG java.sql.PreparedStatement - ==> Parameters:

 

5149 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, TITLE, AUTHOR_ID

5149 [main]

DEBUG

java.sql.ResultSet - <==

Row: 1, nothing title, 1

5227 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5227 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 1(Integer)

5227 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5227 [main]

DEBUG

java.sql.ResultSet - <==

Row: 1, user1, user1, [email protected], guy

5258 [main]

DEBUG

java.sql.ResultSet - <==

Row: 2, just funny, 2

5258 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5258 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 2(Integer)

5258 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5258 [main]

DEBUG

java.sql.ResultSet - <==

Row: 2, user2, user2, [email protected], guy

5274 [main]

DEBUG

java.sql.ResultSet - <==

Row: 3, My Blog, 3

5274 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5274 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 3(Integer)

5274 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5274 [main]

DEBUG

java.sql.ResultSet - <==

Row: 3, user3, user3, [email protected], guy

5274 [main]

DEBUG

java.sql.ResultSet - <==

Row: 4, My Blog, 4

5274 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5274 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 4(Integer)

5289 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5289 [main]

DEBUG

java.sql.ResultSet - <==

Row: 4, user4, user4, [email protected], guy

5289 [main]

DEBUG

java.sql.ResultSet - <==

Row: 5, hello one, 5

5289 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5289 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 5(Integer)

5289 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5289 [main]

DEBUG

java.sql.ResultSet - <==

Row: 5, user5, user5, [email protected], guy

5289 [main]

DEBUG

java.sql.ResultSet - <==

Row: 6, hello two, 6

5289 [main]

DEBUG

java.sql.PreparedStatement

- ==> Executing: SELECT * FROM AUTHOR WHERE ID

= ?

5289 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters: 6(Integer)

5289 [main]

DEBUG

java.sql.ResultSet - <==

Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5289 [main]

DEBUG

java.sql.ResultSet - <==

Row: 6, user6, user6, [email protected], guy

=========================== ID:1

title:nothing title authorID:1 authorName:user1 authorPassword:user1 authorEmail:[email protected] authorBio:guy

=========================== ID:2

title:just funny authorID:2

 

authorName:user2 authorPassword:user2 authorEmail:[email protected] authorBio:guy

=========================== ID:3 title:My Blog authorID:3 authorName:user3 authorPassword:user3 authorEmail:[email protected] authorBio:guy

=========================== ID:4 title:My Blog authorID:4 authorName:user4 authorPassword:user4 authorEmail:[email protected] authorBio:guy

=========================== ID:5 title:hello one authorID:5 authorName:user5 authorPassword:user5 authorEmail:[email protected] authorBio:guy

=========================== ID:6 title:hello two authorID:6 authorName:user6 authorPassword:user6 authorEmail:[email protected] authorBio:guy

5305 [main] DEBUG java.sql.Connection

- xxx Connection Closed

修改后的结果是"selectAuthor"的SQL语句会被执行6次来加载每个博客的作者信息,这样就产生N+1问题。但恰恰,配置(1)或配置(2)就是解决这个N+1问题的一种方案,这叫联合嵌套结果集(Nested Results for Association)。

另外一种方案就是使用延迟加载,在SQL配置文件中设置:

<settings>

<setting name="lazyLoadingEnabled"value="true"/> settings>

再次执行的结果是:

3371 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Created connection 701508.

3386 [main] DEBUG java.sql.Connection - ooo Connection Opened

4745 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM BLOG WHERE ID > 0 and ID < 7

4745 [main] DEBUG java.sql.PreparedStatement - ==> Parameters:

4886 [main] DEBUG java.sql.ResultSet - <== Columns: ID, TITLE, AUTHOR_ID

4886 [main] DEBUG java.sql.ResultSet - <== Row: 1, nothing title, 1

5120 [main] DEBUG java.sql.ResultSet - <== Row: 2, just funny, 2

5120 [main] DEBUG java.sql.ResultSet - <== Row: 3, My Blog, 3

5120 [main] DEBUG java.sql.ResultSet - <== Row: 4, My Blog, 4

5136 [main] DEBUG java.sql.ResultSet - <== Row: 5, hello one, 5

5136 [main] DEBUG java.sql.ResultSet - <== Row: 6, hello two, 6

===========================

5214 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5214 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 1(Integer)

5214 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5214 [main] DEBUG java.sql.ResultSet - <== Row: 1, user1, user1, [email protected], guy

ID:1

title:nothing title authorID:1 authorName:user1 authorPassword:user1 authorEmail:[email protected] authorBio:guy

===========================

5307 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5307 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 2(Integer)

5323 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5323 [main] DEBUG java.sql.ResultSet - <== Row: 2, user2, user2, [email protected], guy

ID:2

title:just funny authorID:2 authorName:user2 authorPassword:user2 authorEmail:[email protected] authorBio:guy

===========================

5323 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5323 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer)

5323 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5323 [main] DEBUG java.sql.ResultSet - <== Row: 3, user3, user3, [email protected], guy

ID:3 title:My Blog

 

authorID:3 authorName:user3 authorPassword:user3 authorEmail:[email protected] authorBio:guy

===========================

5323 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5338 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 4(Integer)

5338 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5338 [main] DEBUG java.sql.ResultSet - <== Row: 4, user4, user4, [email protected], guy

ID:4 title:My Blog authorID:4 authorName:user4 authorPassword:user4 authorEmail:[email protected] authorBio:guy

===========================

5338 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5338 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 5(Integer)

5338 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5338 [main] DEBUG java.sql.ResultSet - <== Row: 5, user5, user5, [email protected], guy

ID:5 title:hello one authorID:5 authorName:user5 authorPassword:user5 authorEmail:[email protected] authorBio:guy

===========================

5338 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM AUTHOR WHERE ID = ?

5338 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 6(Integer)

5354 [main] DEBUG java.sql.ResultSet - <== Columns: ID, USERNAME, PASSWORD, EMAIL, BIO

5354 [main] DEBUG java.sql.ResultSet - <== Row: 6, user6, user6, [email protected], guy

ID:6 title:hello two authorID:6 authorName:user6 authorPassword:user6 authorEmail:[email protected] authorBio:guy

5354 [main] DEBUG java.sql.Connection - xxx Connection Closed

MyBatis首先只是加载了Blog信息,Author信息并没有加载。但我们执行下面的方法:

printBlogAuthorList(blogList);

Author信息才被加载进来,并且是处理一条才加载一条。正如官方指南说的:

MyBatis可以使用延迟加载这些查询,因此这些查询立马可节省开销。然而,如果您加载一个列表后立即迭代访问嵌套的数据,这将会调用所有的延迟加载,因此性能会变得非常糟糕。

一般地,获得一个数据库列表后,都需要进行迭代处理,要不然获得数据就没有太大意义了,因此使用延迟加载在总的时间上并没有节省。

因此,在“has-a”一对一情况下,使用联合嵌套结果集是较好的做法,但可能会产生能大量

冗余的、重复的数据。

使用Collection元素

Collection元素用来处理“一对多”的数据模型,例如,一个博客有许多文章(Posts)。

在博客类里,应该有一个文章的列表,其属性定义如下:

private List posts;

这个例子,查询id排在前3位的博客,并列出这3个博客的所有文章。

重新修改Blog域模型:

package org.mybatis.model;

public class Blog { private Integer id; private String title; private Integer authorId; private List posts;

//省略get和set方法

}

BlogMapper.xml配置(1),使用集合嵌套选择(Nested Select for Collection)

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResult"type="Blog">

<id property="id"column="id"/>

<result property="title"column="title"/>

<result property="authorId"column="authorid"/>

<collection property="posts"javaType="ArrayList"column="id" ofType="Post"select="selectPostsForBlog"/>

resultMap>

<select id="selectPostsForBlog"parameterType="int"resultType="Post">

SELECT * FROM POST WHERE BLOG_ID = #{id}

select>

<select id="selectBlog_use_collection"resultMap="blogResult">

SELECT id , title, author_id as authorid FROM BLOG WHERE ID > 0 and ID < 4]]>

select>

mapper>

BlogMapper.xml配置(2),使用集合的嵌套结果集(Nested Results for Collection)

xml version="1.0"encoding="UTF-8"?>

DOCTYPE mapper

PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

"http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="org.mybatis.model.BlogMapper">

<resultMap id="blogResult"type="Blog"> <id property="id"column="blog_id"/>

<result property="title"column="blog_title"/>

<result property="authorId"column="authorid"/>

<collection property="posts"ofType="Post">

<id property="id"column="post_id"/>

<result property="subject"column="post_subject"/>

<result property="section"column="post_section"/>

<result property="body"column="post_body"/> collection>

resultMap>

<select id="selectBlog_use_collection" resultMap="blogResult">

select

B.id as blog_id,

B.title as blog_title,

B.author_id as authorid,

P.id as post_id,

P.subject as post_subject,

P.section as post_section,

P.body as post_body from Blog B

left outer join Post P on B.id = P.blog_id where B.id > 0 and B.id < 4]]>

select>

mapper>

调用代码:

List blogList = (List)session.selectList("selectBlog_use_collection"); printBlogPosts(blogList);

public static void printBlogPosts(List blogList)

{ for (Blog blog : blogList)

{

System.out.println("\n==========================="); System.out.println("ID:" + blog.getId());

System.out.println("blog_title:" + blog.getTitle());

System.out.println("authorID:" + blog.getAuthorId()); System.out.println("===========posts=============");

for (Post post : blog.getPosts())

{

System.out.println("subject:" + post.getSubject());

System.out.println("section:" + post.getSection());

System.out.println("body:" + post.getBody());

}

}

}

BlogMapper.xml配置(1)和BlogMapper.xml配置(2)打印的结果都是一样的,只是查询的方式不一样:

3340 [main] DEBUG java.sql.Connection - ooo Connection Opened

4620 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT id , title, author_id as authorid FROM BLOG WHERE ID > 0 and ID < 4

 

4620 [main]

DEBUG

java.sql.PreparedStatement

- ==> Parameters:

 

4777 [main]

DEBUG

java.sql.ResultSet -

<==

Columns: ID, TITLE, AUTHORID

 

4777 [main]

DEBUG

java.sql.ResultSet -

<==

Row: 1,

nothing title, 1

 

4995 [main]

DEBUG

java.sql.ResultSet -

<==

Row: 2,

just funny, 2

 

4995 [main]

DEBUG

java.sql.ResultSet -

<==

Row: 3,

My Blog, 3

 

================= 5058 [main] DEBUG

========== java.sql.PreparedState

ment

- ==> Execu

ting: SELECT * FROM

POST WHERE

BLOG_ID = ?

5058 [main] DEBUG

java.sql.PreparedStatement

- ==> Parameters: 1(Integer)

 

5058 [main] DEBUG

java.sql.ResultSet -

<==

Columns: ID, BLOG_ID, AUTHOR_ID, CREATED_ON,

SECTION, SUBJECT,

BODY

 

 

5058 [main] DEBUG

java.sql.ResultSet -

<==

Row: 1,

1,

1,

2010-08-04, photo, ddd,

nothing

5182 [main] DEBUG

java.sql.ResultSet -

<==

Row: 2,

1,

1,

2010-08-05, photo, df,

nothing too

5182 [main] DEBUG

java.sql.ResultSet -

<==

Row: 3,

1,

1,

2010-08-06, photo, ddfdfdd,

also nothing ID:1

blog_title:nothing title authorID:1

===========posts============= subject:ddd section:photo body:nothing subject:df section:photo body:nothing too subject:ddfdfdd section:photo body:also nothing

===========================

5182 [main] DEBUG java.sql.PreparedState

ment

- ==> Execu

ting:

 

SELECT * FROM POST WHERE

BLOG_ID = ?

5182 [main] DEBUG

java.sql.PreparedStatement

- ==> Parameters:

2(Integer)

5198 [main] DEBUG

java.sql.ResultSet - <==

Columns: ID, BLOG_ID, AUTHOR_ID, CREATED_ON,

SECTION, SUBJECT,

BODY

 

5198 [main] DEBUG

java.sql.ResultSet - <==

Row: 4, 2, 2, 2010-08-06, photo, fd,

nothing more ID:2

blog_title:just funny authorID:2

===========posts============= subject:fd section:photo body:nothing more

===========================

 

5198 [main] DEBUG java.sql.PreparedStatement - ==> Executing: SELECT * FROM POST WHERE BLOG_ID = ?

5198 [main] DEBUG java.sql.PreparedStatement - ==> Parameters: 3(Integer) ID:3

blog_title:My Blog authorID:3

===========posts=============

5198 [main] DEBUG java.sql.Connection - xxx Connection Closed

5198 [main] DEBUG org.apache.ibatis.datasource.pooled.PooledDataSource - Returned connection 3403998 to pool.

从输出结果看,可以知道是使用了延迟加载。对于collection元素的学习暂时到这里。如果一篇文章还有N个评论,那还可以继续嵌套下去,嵌套的深度是没有限制的,但要考虑一下性能的因素。

附录4 XML中的特殊字符

如果MyBatis使用XML配置,那不可避免地会遇到一些对XML来说是特殊的字符。如小于号

“<”,因为XML解析器会认为是一个新元素的开始,因此要进行转义。这里有两个方法:

使用转义实体

下面是五个在XML文档中预定义好的转义实体:

<

 

< 小于号

>

 

> 大于号

&

 

& 和

'

 

' 单引号

"

 

" 双引号

如果是小于等于“<=”,其转义实体为:<= 同理,大小等于“>=”,其转义实体为:>= 使用CDATA部件

一个 CDATA 部件以""标记结束。在""之间的特殊字符的意义都不起作用,而转变为普通字符串内容。

一般地,在MyBatis 的XML映射语句配置文件中,如果SQL语句有特殊字符,那么使用CDTA

部件括起来,如:

<select id="selectBlog_use_collection"resultMap="blogResult">

SELECT id , title, author_id as authored

FROM BLOG

WHERE ID > 0 and ID < 10

]]> select>

当然使用转义实体也行:

<select id="selectBlog_use_collection"resultMap="blogResult">

SELECT id , title, author_id as authorid

FROM BLOG

WHERE ID > 0 and ID < 10

select>

而在动态SQL各元素的测试语句中,在元素的属性中不能再嵌套其它元素或包含CDATA部

件,因此只能使用转义实体,如:

<select id="selectAuthor_use_where"parameterType="Blog"resultType="Author"> select * from author

<where>

<if test="authorId!=null andauthorId>=1 andauthorId<=5"> id = #{authorId}

if>

where> select>

你可能感兴趣的:(My,thoughts,Java架构,mybatis3,夜光,Java)