MyBatis 的Mapper文件配置详解.docx
- 文档编号:22785181
- 上传时间:2023-04-27
- 格式:DOCX
- 页数:33
- 大小:36.45KB
MyBatis 的Mapper文件配置详解.docx
《MyBatis 的Mapper文件配置详解.docx》由会员分享,可在线阅读,更多相关《MyBatis 的Mapper文件配置详解.docx(33页珍藏版)》请在冰豆网上搜索。
MyBatis的Mapper文件配置详解
MyBatis真正的力量是在映射语句中。
这里是奇迹发生的地方。
对于所有的力量,SQL映射的XML文件是相当的简单。
当然如果你将它们和对等功能的JDBC代码来比较,你会发现映射文件节省了大约95%的代码量。
MyBatis的构建就是聚焦于SQL的,使其远离于普通的方式。
SQL映射文件有很少的几个顶级元素(按照它们应该被定义的顺序):
∙cache–配置给定命名空间的缓存。
∙cache-ref–从其他命名空间引用缓存配置。
∙resultMap–最复杂,也是最有力量的元素,用来描述如何从数据库结果集中来加载你的对象。
∙parameterMap–已经被废弃了!
老式风格的参数映射。
内联参数是首选,这个元素可能在将来被移除。
这里不会记录。
∙sql–可以重用的SQL块,也可以被其他语句引用。
∙insert–映射插入语句
∙update–映射更新语句
∙delete–映射删除语句
∙select–映射查询语句
下一部分将从语句本身开始来描述每个元素的细节。
select
查询语句是使用MyBatis时最常用的元素之一。
直到你从数据库取出数据时才会发现将数据存在数据库中是多么的有价值,所以许多应用程序查询要比更改数据多的多。
对于每次插入,更新或删除,那也会有很多的查询。
这是MyBatis的一个基本原则,也是将重心和努力放到查询和结果映射的原因。
对简单类别的查询元素是非常简单的。
比如:
这个语句被称作selectPerson,使用一个int(或Integer)类型的参数,并返回一个HashMap类型的对象,其中的键是列名,值是列对应的值。
注意参数注释:
#{id}
这就告诉MyBatis创建一个预处理语句参数。
使用JDBC,这样的一个参数在SQL中会由一个“?
”来标识,并被传递到一个新的预处理语句中,就像这样:
//SimilarJDBCcode,NOTMyBatis…StringselectPerson="SELECT*FROMPERSONWHEREID=?
";PreparedStatementps=conn.prepareStatement(selectPerson);ps.setInt(1,id);
当然,这需要很多单独的JDBC的代码来提取结果并将它们映射到对象实例中,这就是MyBatis节省你时间的地方。
我们需要深入了解参数和结果映射。
那些细节部分我们下面来了解。
select元素有很多属性允许你配置,来决定每条语句的作用细节。
SelectAttributes
属性
描述
id
在命名空间中唯一的标识符,可以被用来引用这条语句。
parameterType
将会传入这条语句的参数类的完全限定名或别名。
parameterMap
这是引用外部parameterMap的已经被废弃的方法。
使用内联参数映射和parameterType属性。
resultType
从这条语句中返回的期望类型的类的完全限定名或别名。
注意集合情形,那应该是集合可以包含的类型,而不能是集合本身。
使用resultType或resultMap,但不能同时使用。
resultMap
命名引用外部的resultMap。
返回map是MyBatis最具力量的特性,对其有一个很好的理解的话,许多复杂映射的情形就能被解决了。
使用resultMap或resultType,但不能同时使用。
flushCache
将其设置为true,不论语句什么时候被带哦用,都会导致缓存被清空。
默认值:
false。
useCache
将其设置为true,将会导致本条语句的结果被缓存。
默认值:
true。
timeout
这个设置驱动程序等待数据库返回请求结果,并抛出异常时间的最大等待值。
默认不设置(驱动自行处理)
fetchSize
这是暗示驱动程序每次批量返回的结果行数。
默认不设置(驱动自行处理)。
statementType
STATEMENT,PREPARED或CALLABLE的一种。
这会让MyBatis使用选择使用Statement,PreparedStatement或CallableStatement。
默认值:
PREPARED。
resultSetType
FORWARD_ONLY|SCROLL_SENSITIVE|SCROLL_INSENSITIVE中的一种。
默认是不设置(驱动自行处理)。
databaseId
IncasethereisaconfigureddatabaseIdProvider,MyBatiswillloadallstatementswithnodatabaseIdattributeorwithadatabaseIdthatmatchesthecurrentone.IfcasethesamestatementiffoundwithandwithoutthedatabaseIdthelatterwillbediscarded.
insert,updateanddelete
数据变更语句insert,update和delete在它们的实现中非常相似:
Insert,UpdateandDeleteAttributes
属性
描述
id
在命名空间中唯一的标识符,可以被用来引用这条语句。
parameterType
将会传入这条语句的参数类的完全限定名或别名。
parameterMap
这是引用外部parameterMap的已经被废弃的方法。
使用内联参数映射和parameterType属性。
flushCache
将其设置为true,不论语句什么时候被带哦用,都会导致缓存被清空。
默认值:
false。
timeout
这个设置驱动程序等待数据库返回请求结果,并抛出异常时间的最大等待值。
默认不设置(驱动自行处理)。
statementType
STATEMENT,PREPARED或CALLABLE的一种。
这会让MyBatis使用选择使用Statement,PreparedStatement或CallableStatement。
默认值:
PREPARED。
useGeneratedKeys
(仅对insert有用)这会告诉MyBatis使用JDBC的getGeneratedKeys方法来取出由数据(比如:
像MySQL和SQLServer这样的数据库管理系统的自动递增字段)内部生成的主键。
默认值:
false。
keyProperty
(仅对insert有用)标记一个属性,MyBatis会通过getGeneratedKeys或者通过insert语句的selectKey子元素设置它的值。
默认:
不设置。
keyColumn
(仅对insert有用)标记一个属性,MyBatis会通过getGeneratedKeys或者通过insert语句的selectKey子元素设置它的值。
默认:
不设置。
下面就是insert,update和delete语句的示例:
values(#{id},#{username},#{password},#{email},#{bio})
username=#{username},
password=#{password},
email=#{email},
bio=#{bio}
whereid=#{id}
如前所述,插入语句有一点多,它有一些属性和子元素用来处理主键的生成。
首先,如果你的数据库支持自动生成主键的字段(比如MySQL和SQLServer),那么你可以设置useGeneratedKeys=”true”,而且设置keyProperty到你已经做好的目标属性上。
例如,如果上面的Author表已经对id使用了自动生成的列类型,那么语句可以修改为:
values(#{username},#{password},#{email},#{bio})
MyBatis有另外一种方法来处理数据库不支持自动生成类型,或者可能JDBC驱动不支持自动生成主键时的主键生成问题。
这里有一个简单(甚至很傻)的示例,它可以生成一个随机ID(可能你不会这么做,但是这展示了MyBatis处理问题的灵活性,因为它并不真的关心ID的生成):
insertintoAuthor
(id,username,password,email,bio,favourite_section)
values
(#{id},#{username},#{password},#{email},#{bio},#{favouriteSection,jdbcType=VARCHAR})
在上面的示例中,selectKey元素将会首先运行,Author的id会被设置,然后插入语句会被调用。
这给你了一个简单的行为在你的数据库中来处理自动生成的主键,而不需要使你的Java代码变得复杂。
selectKey元素描述如下:
selectKeyAttributes
属性
描述
keyProperty
selectKey语句结果应该被设置的目标属性。
resultType
结果的类型。
MyBatis通常可以算出来,但是写上也没有问题。
MyBatis允许任何简单类型用作主键的类型,包括字符串。
order
这可以被设置为BEFORE或AFTER。
如果设置为BEFORE,那么它会首先选择主键,设置keyProperty然后执行插入语句。
如果设置为AFTER,那么先执行插入语句,然后是selectKey元素-这和如Oracle数据库相似,可以在插入语句中嵌入序列调用。
statementType
和前面的相同,MyBatis支持STATEMENT,PREPARED和CALLABLE语句的映射类型,分别代表PreparedStatement和CallableStatement类型。
sql
这个元素可以被用来定义可重用的SQL代码段,可以包含在其他语句中。
比如:
这个SQL片段可以被包含在其他语句中,例如:
whereid=#{id}
Parameters
在之前的语句中,你已经看到了一些简单参数的示例。
MyBatis中参数是非常强大的在元素。
对于简单的做法,大概90%的情况,是不用太多的,比如:
fromusers
whereid=#{id}
上面的这个示例说明了一个非常简单的命名参数映射。
参数类型被设置为“int”,因此这个参数可以被设置成任何内容。
原生的类型或简单数据类型,比如整型和没有相关属性的字符串,因此它会完全用参数来替代。
然而,如果你传递了一个复杂的对象,那么MyBatis的处理方式就会有一点不同。
比如:
values(#{id},#{username},#{password})
如果User类型的参数对象传递到了语句中,username和password属性将会被查找,id、然后它们的值就被传递到预处理语句的参数中。
这点对于传递参数到语句中非常好。
但是对于参数映射也有一些其他的特性。
首先,像MyBatis的其他部分,参数可以指定一个确定的数据类型。
#{property,javaType=int,jdbcType=NUMERIC}
像MyBatis的剩余部分,javaType通常可以从参数对象中来去顶,除非对象是一个HashMap。
那么javaType应该被确定来保证使用正确类型处理器。
注意 如果null被当作值来传递,对于所有可能为空的列,JDBCType是需要的。
以可以自己通过阅读预处理语句的setNull()方法的JavaDocs文档来研究这个。
为了自定义类型处理器,你可以指定一个确定的类型处理器类(或别名),比如:
#{age,javaType=int,jdbcType=NUMERIC,typeHandler=MyTypeHandler}
尽管它看起来繁琐,但是实际上是你很少设置它们其中之一。
对于数值类型,对于决定有多少数字是相关的,有一个数值范围。
#{height,javaType=double,jdbcType=NUMERIC,numericScale=2}
最后,mode属性允许你指定IN,OUT或INOUT参数。
如果参数为OUT或INOUT,参数对象属性的真实值将会被改变,就像你期望你需要你个输出参数。
如果mode为OUT(或INOUT),而且jdbcType为CURSOR(也就是Oracle的REFCURSOR),你必须指定一个resultMap来映射结果集到参数类型。
要注意这里的javaType属性是可选的,如果左边的空白是jdbcType的CURSOR类型,它会自动地被设置为结果集。
#{department,mode=OUT,jdbcType=CURSOR,javaType=ResultSet,resultMap=departmentResultMap}
MyBatis也支持很多高级的数据类型,比如结构体,但是当注册out参数时你必须告诉语句类型名称。
比如(再次提示,在实际中不要像这样换行):
#{middleInitial,mode=OUT,jdbcType=STRUCT,jdbcTypeName=MY_TYPE,resultMap=departmentResultMap}
尽管所有这些强大的选项很多时候你只简单指定属性名,MyBatis会自己计算剩余的。
最多的情况是你为jdbcType指定可能为空的列名。
#{firstName}#{middleInitial,jdbcType=VARCHAR}#{lastName}
字符串替换
默认情况下,使用#{}格式的语法会导致MyBatis创建预处理语句属性并以它为背景设置安全的值(比如?
)。
这样做很安全,很迅速也是首选做法,有时你只是想直接在SQL语句中插入一个不改变的字符串。
比如,像ORDERBY,你可以这样来使用:
ORDERBY${columnName}
这里MyBatis不会修改或转义字符串。
重要 接受从用户输出的内容并提供给语句中不变的字符串,这样做是不安全的。
这会导致潜在的SQL注入攻击,因此你不应该允许用户输入这些字段,或者通常自行转义并检查。
ResultMaps
resultMap元素是MyBatis中最重要最强大的元素。
它就是让你远离90%的需要从结果集中取出数据的JDBC代码的那个东西,而且在一些情形下允许你做一些JDBC不支持的事情。
事实上,编写相似于对复杂语句联合映射这些等同的代码,也许可以跨过上千行的代码。
ResultMap的设计就是简单语句不需要明确的结果映射,而很多复杂语句确实需要描述它们的关系。
你已经看到简单映射语句的示例了,但没有明确的resultMap。
比如:
fromsome_table
whereid=#{id}
这样一个语句简单作用于所有列被自动映射到HashMap的键上,这由resultType属性指定。
这在很多情况下是有用的,但是HashMap不能很好描述一个领域模型。
那样你的应用程序将会使用JavaBeans或POJOs(PlainOldJavaObjects,普通Java对象)来作为领域模型。
MyBatis对两者都支持。
看看下面这个JavaBean:
packagecom.someapp.model;
publicclassUser{
privateintid;
privateStringusername;
privateStringhashedPassword;
publicintgetId(){
returnid;
}
publicvoidsetId(intid){
this.id=id;
}
publicStringgetUsername(){
returnusername;
}
publicvoidsetUsername(Stringusername){
this.username=username;
}
publicStringgetHashedPassword(){
returnhashedPassword;
}
publicvoidsetHashedPassword(StringhashedPassword){
this.hashedPassword=hashedPassword;
}}
基于JavaBean的规范,上面这个类有3个属性:
id,username和hashedPassword。
这些在select语句中会精确匹配到列名。
这样的一个JavaBean可以被映射到结果集,就像映射到HashMap一样简单。
fromsome_table
whereid=#{id}
要记住类型别名是你的伙伴。
使用它们你可以不用输入类的全路径。
比如:
--Inmybatis-config.xmlfile-->
--InSQLMappingXMLfile-->
fromsome
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- MyBatis 的Mapper文件配置详解 Mapper 文件 配置 详解