虽然我们都知道有26个设计模式,但是大多停留在概念层面,真实开发中很少遇到,Mybatis源码中使用了大量的设计模式,阅读源码并观察设计模式在其中的应用,能够更深入的理解设计模式。
Mybatis至少遇到了以下的设计模式的使用:
接下来挨个模式进行解读,先介绍模式自身的知识,然后解读在Mybatis中怎样应用了该模式。
Builder模式的定义是“将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。”,它属于创建类模式,一般来说,如果一个对象的构建比较复杂,超出了构造函数所能包含的范围,就可以使用工厂模式和Builder模式,相对于工厂模式会产出一个完整的产品,Builder应用于更加复杂的对象的构建,甚至只会构建产品的一个部分。
在Mybatis环境的初始化过程中,SqlSessionFactoryBuilder会调用XMLConfigBuilder读取所有的MybatisMapConfig.xml和所有的*Mapper.xml文件,构建Mybatis运行的核心对象Configuration对象,然后将该Configuration对象作为参数构建一个SqlSessionFactory对象。
其中XMLConfigBuilder在构建Configuration对象时,也会调用XMLMapperBuilder用于读取*Mapper文件,而XMLMapperBuilder会使用XMLStatementBuilder来读取和build所有的SQL语句。
在这个过程中,有一个相似的特点,就是这些Builder会读取文件或者配置,然后做大量的XpathParser解析、配置或语法的解析、反射生成对象、存入结果缓存等步骤,这么多的工作都不是一个构造函数所能包括的,因此大量采用了Builder模式来解决。
对于builder的具体类,方法都大都用build*开头,比如SqlSessionFactoryBuilder为例,它包含以下方法:
即根据不同的输入参数来构建SqlSessionFactory这个工厂对象。
在Mybatis中比如SqlSessionFactory使用的是工厂模式,该工厂没有那么复杂的逻辑,是一个简单工厂模式。
简单工厂模式(Simple Factory Pattern):又称为静态工厂方法(Static Factory Method)模式,它属于类创建型模式。在简单工厂模式中,可以根据参数的不同返回不同类的实例。简单工厂模式专门定义一个类来负责创建其他类的实例,被创建的实例通常都具有共同的父类。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
private
SqlSession
openSessionFromDataSource
(
ExecutorType
execType
,
TransactionIsolationLevel
level
,
boolean
autoCommit
)
{
Transaction
tx
=
null
;
try
{
final
Environment
environment
=
configuration
.
getEnvironment
(
)
;
final
TransactionFactory
transactionFactory
=
getTransactionFactoryFromEnvironment
(
environment
)
;
tx
=
transactionFactory
.
newTransaction
(
environment
.
getDataSource
(
)
,
level
,
autoCommit
)
;
final
Executor
executor
=
configuration
.
newExecutor
(
tx
,
execType
)
;
return
new
DefaultSqlSession
(
configuration
,
executor
,
autoCommit
)
;
}
catch
(Exception
e
)
{
closeTransaction
(
tx
)
;
// may have fetched a connection so lets call
// close()
throw
ExceptionFactory
.
wrapException
(
"Error opening session. Cause: "
+
e
,
e
)
;
}
finally
{
ErrorContext
.
instance
(
)
.
reset
(
)
;
}
}
|
在这里其实也可以看到端倪,SqlSession的执行,其实是委托给对应的Executor来进行的。
而对于LogFactory,它的实现代码:
1
2
3
4
5
6
7
8
9
10
|
public
final
class
LogFactory
{
private
static
Constructor
extends
Log
>
logConstructor
;
private
LogFactory
(
)
{
// disable construction
}
public
static
Log getLog
(
Class
>
aClass
)
{
return
getLog
(
aClass
.
getName
(
)
)
;
}
|
这里有个特别的地方,是Log变量的的类型是Constructor extends Log>,也就是说该工厂生产的不只是一个产品,而是具有Log公共接口的一系列产品,比如Log4jImpl、Slf4jImpl等很多具体的Log。
单例模式(Singleton Pattern):单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类称为单例类,它提供全局访问的方法。
单例模式的要点有三个:一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。单例模式是一种对象创建型模式。单例模式又名单件模式或单态模式。
在Mybatis中有两个地方用到单例模式,ErrorContext和LogFactory,其中ErrorContext是用在每个线程范围内的单例,用于记录该线程的执行环境错误信息,而LogFactory则是提供给整个Mybatis使用的日志工厂,用于获得针对项目配置好的日志对象。
ErrorContext的单例实现代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public
class
ErrorContext
{
private
static
final
ThreadLocal
private
ErrorContext
(
)
{
}
public
static
ErrorContext
instance
(
)
{
ErrorContext
context
=
LOCAL
.
get
(
)
;
if
(
context
==
null
)
{
context
=
new
ErrorContext
(
)
;
LOCAL
.
set
(
context
)
;
}
return
context
;
}
|
构造函数是private修饰,具有一个static的局部instance变量和一个获取instance变量的方法,在获取实例的方法中,先判断是否为空如果是的话就先创建,然后返回构造好的对象。
只是这里有个有趣的地方是,LOCAL的静态实例变量使用了ThreadLocal修饰,也就是说它属于每个线程各自的数据,而在instance()方法中,先获取本线程的该实例,如果没有就创建该线程独有的ErrorContext。
代理模式可以认为是Mybatis的核心使用的模式,正是由于这个模式,我们只需要编写Mapper.java接口,不需要实现,由Mybatis后台帮我们完成具体SQL的执行。
代理模式(Proxy Pattern) :给某一个对象提供一个代 理,并由代理对象控制对原对象的引用。代理模式的英 文叫做Proxy或Surrogate,它是一种对象结构型模式。
这里有两个步骤,第一个是提前创建一个Proxy,第二个是使用的时候会自动请求Proxy,然后由Proxy来执行具体事务;
当我们使用Configuration的getMapper方法时,会调用mapperRegistry.getMapper方法,而该方法又会调用mapperProxyFactory.newInstance(sqlSession)来生成一个具体的代理:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
/**
* @author Lasse Voss
*/
public
class
MapperProxyFactory
private
final
Class
private
final
Map
<
Method
,
MapperMethod
>
methodCache
=
new
ConcurrentHashMap
<
Method
,
MapperMethod
>
(
)
;
public
MapperProxyFactory
(
Class
this
.
mapperInterface
=
mapperInterface
;
}
public
Class
return
mapperInterface
;
}
public
Map
<
Method
,
MapperMethod
>
getMethodCache
(
)
{
return
methodCache
;
}
@SuppressWarnings
(
"unchecked"
)
protected
T
newInstance
(
MapperProxy
return
(
T
)
Proxy
.
newProxyInstance
(
mapperInterface
.
getClassLoader
(
)
,
new
Class
[
]
{
mapperInterface
}
,
mapperProxy
)
;
}
public
T
newInstance
(
SqlSession
sqlSession
)
{
final
MapperProxy
return
newInstance
(
mapperProxy
)
;
}
}
|
在这里,先通过T newInstance(SqlSession sqlSession)方法会得到一个MapperProxy对象,然后调用T newInstance(MapperProxy
而查看MapperProxy的代码,可以看到如下内容:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
public
class
MapperProxy
@Override
public
Object
invoke
(
Object
proxy
,
Method
method
,
Object
[
]
args
)
throws
Throwable
{
try
{
if
(
Object
.
class
.
equals
(
method
.
getDeclaringClass
(
)
)
)
{
return
method
.
invoke
(this
,
args
)
;
}
else
if
(
isDefaultMethod
(
method
)
)
{
return
invokeDefaultMethod
(
proxy
,
method
,
args
)
;
}
}
catch
(Throwable
t
)
{
throw
ExceptionUtil
.
unwrapThrowable
(
t
)
;
}
final
MapperMethod
mapperMethod
=
cachedMapperMethod
(
method
)
;
return
mapperMethod
.
execute
(
sqlSession
,
args
)
;
}
|
非常典型的,该MapperProxy类实现了InvocationHandler接口,并且实现了该接口的invoke方法。
通过这种方式,我们只需要编写Mapper.java接口类,当真正执行一个Mapper接口的时候,就会转发给MapperProxy.invoke方法,而该方法则会调用后续的sqlSession.cud>executor.execute>prepareStatement等一系列方法,完成SQL的执行和返回。
组合模式组合多个对象形成树形结构以表示“整体-部分”的结构层次。
组合模式对单个对象(叶子对象)和组合对象(组合对象)具有一致性,它将对象组织到树结构中,可以用来描述整体与部分的关系。同时它也模糊了简单元素(叶子对象)和复杂元素(容器对象)的概念,使得客户能够像处理简单元素一样来处理复杂元素,从而使客户程序能够与复杂元素的内部结构解耦。
在使用组合模式中需要注意一点也是组合模式最关键的地方:叶子对象和组合对象实现相同的接口。这就是组合模式能够将叶子节点和对象节点进行一致处理的原因。
Mybatis支持动态SQL的强大功能,比如下面的这个SQL:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
UPDATE users
name = #{name}
, age = #{age}
, birthday = #{birthday}
where id = ${id}
|
在这里面使用到了trim、if等动态元素,可以根据条件来生成不同情况下的SQL;
在DynamicSqlSource.getBoundSql方法里,调用了rootSqlNode.apply(context)方法,apply方法是所有的动态节点都实现的接口:
1
2
3
|
public
interface
SqlNode
{
boolean
apply
(
DynamicContext
context
)
;
}
|
对于实现该SqlSource接口的所有节点,就是整个组合模式树的各个节点:
组合模式的简单之处在于,所有的子节点都是同一类节点,可以递归的向下执行,比如对于TextSqlNode,因为它是最底层的叶子节点,所以直接将对应的内容append到SQL语句中:
1
2
3
4
5
6
|
@Override
public
boolean
apply
(
DynamicContext
context
)
{
GenericTokenParser
parser
=
createParser
(new
BindingTokenParser
(
context
,
injectionFilter
)
)
;
context
.
appendSql
(
parser
.
parse
(
text
)
)
;
return
true
;
}
|
但是对于IfSqlNode,就需要先做判断,如果判断通过,仍然会调用子元素的SqlNode,即contents.apply方法,实现递归的解析。
1
2
3
4
5
6
7
8
|
@Override
public
boolean
apply
(
DynamicContext
context
)
{
if
(
evaluator
.
evaluateBoolean
(
test
,
context
.
getBindings
(
)
)
)
{
contents
.
apply
(
context
)
;
return
true
;
}
return
false
;
}
|
模板方法模式是所有模式中最为常见的几个模式之一,是基于继承的代码复用的基本技术。
模板方法模式需要开发抽象类和具体子类的设计师之间的协作。一个设计师负责给出一个算法的轮廓和骨架,另一些设计师则负责给出这个算法的各个逻辑步骤。代表这些具体逻辑步骤的方法称做基本方法(primitive method);而将这些基本方法汇总起来的方法叫做模板方法(template method),这个设计模式的名字就是从此而来。
模板类定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
在Mybatis中,sqlSession的SQL执行,都是委托给Executor实现的,Executor包含以下结构:
其中的BaseExecutor就采用了模板方法模式,它实现了大部分的SQL执行逻辑,然后把以下几个方法交给子类定制化完成:
1
2
3
4
5
6
|
protected
abstract
int
doUpdate
(
MappedStatement
ms
,
Object
parameter
)
throws
SQLException
;
protected
abstract
List
protected
abstract
ResultHandler
resultHandler
,
BoundSql
boundSql
)
throws
SQLException
;
|
该模板方法类有几个子类的具体实现,使用了不同的策略:
比如在SimpleExecutor中这样实现update方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
@Override
public
int
doUpdate
(
MappedStatement
ms
,
Object
parameter
)
throws
SQLException
{
Statement
stmt
=
null
;
try
{
Configuration
configuration
=
ms
.
getConfiguration
(
)
;
StatementHandler
handler
=
configuration
.
newStatementHandler
(this
,
ms
,
parameter
,
RowBounds
.DEFAULT
,
null
,
null
)
;
stmt
=
prepareStatement
(
handler
,
ms
.
getStatementLog
(
)
)
;
return
handler
.
update
(
stmt
)
;
}
finally
{
closeStatement
(
stmt
)
;
}
}
|
适配器模式(Adapter Pattern) :将一个接口转换成客户希望的另一个接口,适配器模式使接口不兼容的那些类可以一起工作,其别名为包装器(Wrapper)。适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。
在Mybatsi的logging包中,有一个Log接口:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
/**
* @author Clinton Begin
*/
public
interface
Log
{
boolean
isDebugEnabled
(
)
;
boolean
isTraceEnabled
(
)
;
void
error
(
String
s
,
Throwable
e
)
;
void
error
(
String
s
)
;
void
debug
(
String
s
)
;
void
trace
(
String
s
)
;
void
warn
(
String
s
)
;
}
|
该接口定义了Mybatis直接使用的日志方法,而Log接口具体由谁来实现呢?Mybatis提供了多种日志框架的实现,这些实现都匹配这个Log接口所定义的接口方法,最终实现了所有外部日志框架到Mybatis日志包的适配:
比如对于Log4jImpl的实现来说,该实现持有了org.apache.log4j.Logger的实例,然后所有的日志方法,均委托该实例来实现。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
public
class
Log4jImpl
implements
Log
{
private
static
final
String
FQCN
=
Log4jImpl
.
class
.
getName
(
)
;
private
Logger
log
;
public
Log4jImpl
(
String
clazz
)
{
log
=
Logger
.
getLogger
(
clazz
)
;
}
@Override
public
boolean
isDebugEnabled
(
)
{
return
log
.
isDebugEnabled
(
)
;
}
@Override
public
boolean
isTraceEnabled
(
)
{
return
log
.
isTraceEnabled
(
)
;
}
@Override
public
void
error
(
String
s
,
Throwable
e
)
{
log
.
log
(
FQCN
,
Level
.
ERROR
,
s
,
e
)
;
}
@Override
public
void
error
(
String
s
)
{
log
.
log
(
FQCN
,
Level
.
ERROR
,
s
,
null
)
;
}
@Override
public
void
debug
(
String
s
)
{
log
.
log
(
FQCN
,
Level
.
DEBUG
,
s
,
null
)
;
}
@Override
public
void
trace
(
String
s
)
{
log
.
log
(
FQCN
,
Level
.
TRACE
,
s
,
null
)
;
}
@Override
public
void
warn
(
String
s
)
{
log
.
log
(
FQCN
,
Level
.
WARN
,
s
,
null
)
;
}
}
|
装饰模式(Decorator Pattern) :动态地给一个对象增加一些额外的职责(Responsibility),就增加对象功能来说,装饰模式比生成子类实现更为灵活。其别名也可以称为包装器(Wrapper),与适配器模式的别名相同,但它们适用于不同的场合。根据翻译的不同,装饰模式也有人称之为“油漆工模式”,它是一种对象结构型模式。
在mybatis中,缓存的功能由根接口Cache(org.apache.ibatis.cache.Cache)定义。整个体系采用装饰器设计模式,数据存储和缓存的基本功能由PerpetualCache(org.apache.ibatis.cache.impl.PerpetualCache)永久缓存实现,然后通过一系列的装饰器来对PerpetualCache永久缓存进行缓存策略等方便的控制。如下图:
用于装饰PerpetualCache的标准装饰器共有8个(全部在org.apache.ibatis.cache.decorators包中):
另外,还有一个特殊的装饰器TransactionalCache:事务性的缓存
正如大多数持久层框架一样,mybatis缓存同样分为一级缓存和二级缓存
二级缓存对象的默认类型为PerpetualCache,如果配置的缓存是默认类型,则mybatis会根据配置自动追加一系列装饰器。
Cache对象之间的引用顺序为:
SynchronizedCache–>LoggingCache–>SerializedCache–>ScheduledCache–>LruCache–>PerpetualCache
迭代器(Iterator)模式,又叫做游标(Cursor)模式。GOF给出的定义为:提供一种方法访问一个容器(container)对象中各个元素,而又不需暴露该对象的内部细节。
Java的Iterator就是迭代器模式的接口,只要实现了该接口,就相当于应用了迭代器模式:
比如Mybatis的PropertyTokenizer是property包中的重量级类,该类会被reflection包中其他的类频繁的引用到。这个类实现了Iterator接口,在使用时经常被用到的是Iterator接口中的hasNext这个函数。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
public
class
PropertyTokenizer
implements
Iterator
private
String
name
;
private
String
indexedName
;
private
String
index
;
private
String
children
;
public
PropertyTokenizer
(
String
fullname
)
{
int
delim
=
fullname
.
indexOf
(
'.'
)
;
if
(
delim
>
-
1
)
{
name
=
fullname
.
substring
(
0
,
delim
)
;
children
=
fullname
.
substring
(
delim
+
1
)
;
}
else
{
name
=
fullname
;
children
=
null
;
}
indexedName
=
name
;
delim
=
name
.
indexOf
(
'['
)
;
if
(
delim
>
-
1
)
{
index
=
name
.
substring
(
delim
+
1
,
name
.
length
(
)
-
1
)
;
name
=
name
.
substring
(
0
,
delim
)
;
}
}
public
String
getName
(
)
{
return
name
;
}
public
String
getIndex
(
)
{
return
index
;
}
public
String
getIndexedName
(
)
{
return
indexedName
;
}
public
String
getChildren
(
)
{
return
children
;
}
@Override
public
boolean
hasNext
(
)
{
return
children
!=
null
;
}
@Override
public
PropertyTokenizer
next
(
)
{
return
new
PropertyTokenizer
(
children
)
;
}
@Override
public
void
remove
(
)
{
throw
new
UnsupportedOperationException
(
"Remove is not supported, as it has no meaning in the context of properties."
)
;
}
}
|
可以看到,这个类传入一个字符串到构造函数,然后提供了iterator方法对解析后的子串进行遍历,是一个很常用的方法类。