mybatis 缓存

打印 上一主题 下一主题

主题 819|帖子 819|积分 2457

本地缓存

Mybatis 利用到了两种缓存:本地缓存(local cache)和二级缓存(second level cache)。
每当一个新 session 被创建,MyBatis 就会创建一个与之相干联的本地缓存。任何在 session 执行过的查询结果都会被保存在本地缓存中,以是,当再次执行参数类似的类似查询时,就不必要现实查询数据库了。本地缓存将会在做出修改、事务提交或回滚,以及关闭 session 时清空。
默认环境下,本地缓存数据的生命周期等同于整个 session 的周期。由于缓存会被用来解决循环引用问题和加快重复嵌套查询的速度,以是无法将其完全禁用。但是你可以通过设置 localCacheScope=STATEMENT 来只在语句执行时利用缓存。
注意,如果 localCacheScope 被设置为 SESSION,对于某个对象,MyBatis 将返回在本地缓存中唯一对象的引用。对返回的对象(比方 list)做出的任何修改将会影响本地缓存的内容,进而将会影响到在本次 session 中从缓存返回的值。因此,不要对 MyBatis 所返回的对象作出更改,以防后患。
你可以随时调用以下方法来清空本地缓存:

  1. void clearCache()
复制代码

确保 SqlSession 被关闭

  1. void close()
复制代码
对于你打开的任何 session,你都要保证它们被妥善关闭,这很紧张。保证妥善关闭的最佳代码模式是这样的:
  1. SqlSession session = sqlSessionFactory.openSession();
  2. try (SqlSession session = sqlSessionFactory.openSession()) {
  3.     // 假设下面三行代码是你的业务逻辑
  4.     session.insert(...);
  5.     session.update(...);
  6.     session.delete(...);
  7.     session.commit();
  8. }
复制代码
提示 和 SqlSessionFactory 一样,你可以调用当前利用的 SqlSession 的 getConfiguration 方法来得到 Configuration 实例。
  1. Configuration getConfiguration()
复制代码

利用映射器

  1. <T> T getMapper(Class<T> type)
复制代码
上述的各个 insert、update、delete 和 select 方法都很强盛,但也有些繁琐,它们并不符合范例安全,对你的 IDE 和单元测试也不是那么友爱。因此,利用映射器类来执行映射语句是更常见的做法。
我们已经在之前的入门章节中见到过一个利用映射器的示例。一个映射器类就是一个仅需声明与 SqlSession 方法相匹配方法的接口。下面的示例展示了一些方法签名以及它们是如何映射到 SqlSession 上的。
  1. public interface AuthorMapper {
  2.   // (Author) selectOne("selectAuthor",5);
  3.   Author selectAuthor(int id);
  4.   // (List<Author>) selectList(“selectAuthors”)
  5.   List<Author> selectAuthors();
  6.   // (Map<Integer,Author>) selectMap("selectAuthors", "id")
  7.   @MapKey("id")
  8.   Map<Integer, Author> selectAuthors();
  9.   // insert("insertAuthor", author)
  10.   int insertAuthor(Author author);
  11.   // updateAuthor("updateAuthor", author)
  12.   int updateAuthor(Author author);
  13.   // delete("deleteAuthor",5)
  14.   int deleteAuthor(int id);
  15. }
复制代码
总之,每个映射器方法签名应该匹配相干联的 SqlSession 方法,字符串参数 ID 无需匹配。而是由方法名匹配映射语句的 ID。
此外,返回范例必须匹配期望的结果范例,返回单个值时,返回范例应该是返回值的类,返回多个值时,则为数组或集合类,另外也可以是游标(Cursor)。所有常用的范例都是支持的,包括:原始范例、Map、POJO 和 JavaBean。
提示 映射器接口不必要去实现任何接口或继承自任何类。只要方法签名可以被用来唯一识别对应的映射语句就可以了。
提示 映射器接口可以继承自其他接口。在利用 XML 来绑定映射器接口时,保证语句处于合适的命名空间中即可。唯一的限定是,不能在两个具有继承关系的接口中拥有类似的方法签名(这是潜伏的危险做法,不可取)。
你可以传递多个参数给一个映射器方法。在多个参数的环境下,默认它们将会以 param 加上它们在参数列表中的位置来命名,比如:#{param1}、#{param2}等。如果你想(在有多个参数时)自定义参数的名称,那么你可以在参数上利用 @Param("paramName") 注解。
你也可以给方法传递一个 RowBounds 实例来限定查询结果。

映射器注解

设计初期的 MyBatis 是一个 XML 驱动的框架。配置信息是基于 XML 的,映射语句也是定义在 XML 中的。而在 MyBatis 3 中,我们提供了其它的配置方式。MyBatis 3 构建在全面且强盛的基于 Java 语言的配置 API 之上。它是 XML 和注解配置的底子。注解提供了一种简朴且低成本的方式来实现简朴的映射语句。
提示 不幸的是,Java 注解的表达能力和灵活性非常有限。只管我们花了很多时间在观察、设计和试验上,但最强盛的 MyBatis 映射并不能用注解来构建——我们真没开顽笑。而 C# 属性就没有这些限定,因此 MyBatis.NET 的配置会比 XML 有更大的选择余地。虽说云云,基于 Java 注解的配置照旧有它的长处的。
注解如下表所示:
                                                                                                                               
注解利用对象XML 等价情势形貌@CacheNamespace类<cache>为给定的命名空间(比如类)配置缓存。属性:implemetation、eviction、flushInterval、size、readWrite、blocking、properties。@PropertyN/A<property>指定参数值或占位符(placeholder)(该占位符能被 mybatis-config.xml 内的配置属性更换)。属性:name、value。(仅在 MyBatis 3.4.2 以上可用)@CacheNamespaceRef类<cacheRef>引用另外一个命名空间的缓存以供利用。注意,即使共享类似的全限定类名,在 XML 映射文件中声明的缓存仍被识别为一个独立的命名空间。属性:value、name。如果你利用了这个注解,你应设置 value 或者 name 属性的此中一个。value 属性用于指定能够表示该命名空间的 Java 范例(命名空间名就是该 Java 范例的全限定类名),name 属性(这个属性仅在 MyBatis 3.4.2 以上可用)则直接指定了命名空间的名字。@ConstructorArgs方法<constructor>收集一组结果以传递给一个结果对象的构造方法。属性:value,它是一个 Arg 数组。@ArgN/A     

  • <arg>
  • <idArg>
ConstructorArgs 集合的一部分,代表一个构造方法参数。属性:id、column、javaType、jdbcType、typeHandler、select、resultMap。id 属性和 XML 元素 <idArg> 相似,它是一个布尔值,表示该属性是否用于唯一标识和比较对象。从版本 3.5.4 开始,该注解变为可重复注解。@TypeDiscriminator方法<discriminator>决定利用何种结果映射的一组取值(case)。属性:column、javaType、jdbcType、typeHandler、cases。cases 属性是一个 Case 的数组。@CaseN/A<case>表示某个值的一个取值以及该取值对应的映射。属性:value、type、results。results 属性是一个 Results 的数组,因此这个注解现实上和 ResultMap 很相似,由下面的 Results 注解指定。@Results方法<resultMap>一组结果映射,指定了对某个特定结果列,映射到某个属性或字段的方式。属性:value、id。value 属性是一个 Result 注解的数组。而 id 属性则是结果映射的名称。从版本 3.5.4 开始,该注解变为可重复注解。@ResultN/A     

  • <result>
  • <id>
在列和属性或字段之间的单个结果映射。属性:id、column、javaType、jdbcType、typeHandler、one、many。id 属性和 XML 元素 <id> 相似,它是一个布尔值,表示该属性是否用于唯一标识和比较对象。one 属性是一个关联,和 <association> 类似,而 many 属性则是集合关联,和 <collection> 类似。这样命名是为了制止产生名称辩论。@OneN/A<association>复杂范例的单个属性映射。属性: select,指定可加载合适范例实例的映射语句(也就是映射器方法)全限定名; fetchType,指定在该映射中覆盖全局配置参数 lazyLoadingEnabled; resultMap(3.5.5以上可用), 结果集的完全限定名,该结果映射到查询结果中的集合对象; columnPrefix(3.5.5以上可用),在嵌套的结果会合对所查询的列举行分组的列前缀。 提示 注解 API 不支持联合映射。这是由于 Java 注解不允许产生循环引用。@ManyN/A<collection>复杂范例的集合属性映射。属性: select,指定可加载合适范例实例集合的映射语句(也就是映射器方法)全限定名; fetchType,指定在该映射中覆盖全局配置参数 lazyLoadingEnabled ; resultMap(3.5.5以上可用),结果集的完全限定名,该结果映射到查询结果中的集合对象; columnPrefix(3.5.5以上可用),在嵌套的结果会合对所查询的列举行分组的列前缀。 提示 注解 API 不支持联合映射。这是由于 Java 注解不允许产生循环引用。@MapKey方法供返回值为 Map 的方法利用的注解。它利用对象的某个属性作为 key,将对象 List 转化为 Map。属性:value,指定作为 Map 的 key 值的对象属性名。@Options方法映射语句的属性该注解允许你指定大部分开关和配置选项,它们通常在映射语句上作为属性出现。与在注解上提供大量的属性相比,Options 注解提供了同等、清晰的方式来指定选项。属性:useCache=true、flushCache=FlushCachePolicy.DEFAULT、resultSetType=DEFAULT、statementType=PREPARED、fetchSize=-1、timeout=-1、useGeneratedKeys=false、keyProperty=""、keyColumn=""、resultSets="", databaseId=""。注意,Java 注解无法指定 null 值。因此,一旦你利用了 Options 注解,你的语句就会被上述属性的默认值所影响。要注意制止默认值带来的非预期行为。 databaseId(3.5.5以上可用), 如果有一个配置好的 DatabaseIdProvider, MyBatis 会加载不带 databaseId 属性和带有匹配当前数据库 databaseId 属性的所有语句。如果同时存在带 databaseId 和不带 databaseId 属性的类似语句,则后者会被舍弃。

       注意:keyColumn 属性只在某些数据库中有效(如 Oracle、PostgreSQL 等)。要了解更多关于 keyColumn 和 keyProperty 可选值信息,请检察“insert, update 和 delete”一节。     

  • @Insert
  • @Update
  • @Delete
  • @Select
方法     

  • <insert>
  • <update>
  • <delete>
  • <select>
每个注解分别代表将会被执行的 SQL 语句。它们用字符串数组(或单个字符串)作为参数。如果传递的是字符串数组,字符串数组会被毗连成单个完整的字符串,每个字符串之间加入一个空格。这有效地制止了用 Java 代码构建 SQL 语句时产生的“丢失空格”问题。固然,你也可以提前手动毗连好字符串。属性:value,指定用来构成单个 SQL 语句的字符串数组。 databaseId(3.5.5以上可用), 如果有一个配置好的 DatabaseIdProvider, MyBatis 会加载不带 databaseId 属性和带有匹配当前数据库 databaseId 属性的所有语句。如果同时存在带 databaseId 和不带 databaseId 属性的类似语句,则后者会被舍弃。     

  • @InsertProvider
  • @UpdateProvider
  • @DeleteProvider
  • @SelectProvider
方法     

  • <insert>
  • <update>
  • <delete>
  • <select>
允许构建动态 SQL。这些备选的 SQL 注解允许你指定返回 SQL 语句的类和方法,以供运行时执行。(从 MyBatis 3.4.6 开始,可以利用 CharSequence 代替 String 来作为返回范例)。当执行映射语句时,MyBatis 会实例化注解指定的类,并调用注解指定的方法。你可以通过 ProviderContext 传递映射方法接收到的参数、"Mapper interface type" 和 "Mapper method"(仅在 MyBatis 3.4.5 以上支持)作为参数。(MyBatis 3.4 以上支持传入多个参数) 属性:value、type、method、databaseId。 value and type 属性用于指定类名 (type 属性是 value 的别名, 你必须指定恣意一个。 但是你如果在全局配置中指定 defaultSqlProviderType ,两个属性都可以忽略)。 method 用于指定该类的方法名(从版本 3.5.1 开始,可以省略 method 属性,MyBatis 将会利用 ProviderMethodResolver 接口解析方法的详细实现。如果解析失败,MyBatis 将会利用名为 provideSql 的降级实现)。提示 接下来的“SQL 语句构建器”一章将会讨论该话题,以帮助你以更清晰、更便于阅读的方式构建动态 SQL。 databaseId(3.5.5以上可用), 如果有一个配置好的 DatabaseIdProvider, MyBatis 会加载不带 databaseId 属性和带有匹配当前数据库 databaseId 属性的所有语句。如果同时存在带 databaseId 和不带 databaseId 属性的类似语句,则后者会被舍弃。@Param参数N/A如果你的映射方法继承多个参数,就可以利用这个注解自定义每个参数的名字。否则在默认环境下,除 RowBounds 以外的参数会以 "param" 加参数位置被命名。比方 #{param1}, #{param2}。如果利用了 @Param("person"),参数就会被命名为 #{person}。@SelectKey方法<selectKey>这个注解的功能与 <selectKey> 标签完全同等。该注解只能在 @Insert 或 @InsertProvider 或 @Update 或 @UpdateProvider 标注的方法上利用,否则将会被忽略。如果标注了 @SelectKey 注解,MyBatis 将会忽略掉由 @Options 注解所设置的天生主键或设置(configuration)属性。属性:statement 以字符串数组情势指定将会被执行的 SQL 语句,keyProperty 指定作为参数传入的对象对应属性的名称,该属性将会更新成新的值,before 可以指定为 true 或 false 以指明 SQL 语句应被在插入语句的之前照旧之后执行。resultType 则指定 keyProperty 的 Java 范例。statementType 则用于选择语句范例,可以选择 STATEMENT、PREPARED 或 CALLABLE 之一,它们分别对应于 Statement、PreparedStatement 和 CallableStatement。默认值是 PREPARED。 databaseId(3.5.5以上可用), 如果有一个配置好的 DatabaseIdProvider, MyBatis 会加载不带 databaseId 属性和带有匹配当前数据库 databaseId 属性的所有语句。如果同时存在带 databaseId 和不带 databaseId 属性的类似语句,则后者会被舍弃。@ResultMap方法N/A这个注解为 @Select 或者 @SelectProvider 注解指定 XML 映射中 <resultMap> 元素的 id。这使得注解的 select 可以复用已在 XML 中定义的 ResultMap。如果标注的 select 注解中存在 @Results 或者 @ConstructorArgs 注解,这两个注解将被此注解覆盖。@ResultType方法N/A在利用告终果处置惩罚器的环境下,必要利用此注解。由于此时的返回范例为 void,以是 Mybatis 必要有一种方法来判断每一行返回的对象范例。如果在 XML 有对应的结果映射,请利用 @ResultMap 注解。如果结果范例在 XML 的 <select> 元素中指定了,就不必要利用其它注解了。否则就必要利用此注解。比如,如果一个标注了 @Select 的方法想要利用结果处置惩罚器,那么它的返回范例必须是 void,并且必须利用这个注解(或者 @ResultMap)。这个注解仅在方法返回范例是 void 的环境下生效。@Flush方法N/A如果利用了这个注解,定义在 Mapper 接口中的方法就能够调用 SqlSession#flushStatements() 方法。(Mybatis 3.3 以上可用)
映射注解示例

这个例子展示了如何利用 @SelectKey 注解来在插入前读取数据库序列的值:
  1. @Insert("insert into table3 (id, name) values(#{nameId}, #{name})")
  2. @SelectKey(statement="call next value for TestSequence", keyProperty="nameId", before=<strong>true</strong>, resultType=<strong>int.class</strong>)
  3. <strong>int</strong> insertTable3(Name name);
复制代码
这个例子展示了如何利用 @SelectKey 注解来在插入后读取数据库自增列的值:
  1. @Insert("insert into table2 (name) values(#{name})")
  2. @SelectKey(statement="call identity()", keyProperty="nameId", before=<strong>false</strong>, resultType=<strong>int.class</strong>)
  3. <strong>int</strong> insertTable2(Name name);
复制代码
这个例子展示了如何利用 @Flush 注解来调用 SqlSession#flushStatements():
  1. @Flush
  2. List<BatchResult> flush();
复制代码
这些例子展示了如何通过指定 @Result 的 id 属性来命名结果集:
  1. @Results(id = "userResult", value = {
  2.   @Result(property = "id", column = "uid", id = <strong>true</strong>),
  3.   @Result(property = "firstName", column = "first_name"),
  4.   @Result(property = "lastName", column = "last_name")
  5. })
  6. @Select("select * from users where id = #{id}")
  7. User getUserById(Integer id);
  8. @Results(id = "companyResults")
  9. @ConstructorArgs({
  10.   @Arg(column = "cid", javaType = Integer.class, id = <strong>true</strong>),
  11.   @Arg(column = "name", javaType = String.class)
  12. })
  13. @Select("select * from company where id = #{id}")
  14. Company getCompanyById(Integer id);
复制代码
这个例子展示了如何利用单个参数的 @SqlProvider 注解:
  1. @SelectProvider(type = UserSqlBuilder.class, method = "buildGetUsersByName")
  2. List<User> getUsersByName(String name);
  3. class UserSqlBuilder {
  4.   public static String buildGetUsersByName(final String name) {
  5.     return new SQL(){{
  6.       SELECT("*");
  7.       FROM("users");
  8.       if (name != null) {
  9.         WHERE("name like #{value} || '%'");
  10.       }
  11.       ORDER_BY("id");
  12.     }}.toString();
  13.   }
  14. }
复制代码
这个例子展示了如何利用多个参数的 @SqlProvider 注解:
  1. @SelectProvider(type = UserSqlBuilder.class, method = "buildGetUsersByName")
  2. List<User> getUsersByName(
  3.     @Param("name") String name, @Param("orderByColumn") String orderByColumn);
  4. class UserSqlBuilder {
  5.   // 如果不使用 @Param,就应该定义与 mapper 方法相同的参数
  6.   public static String buildGetUsersByName(
  7.       final String name, final String orderByColumn) {
  8.     return new SQL(){{
  9.       SELECT("*");
  10.       FROM("users");
  11.       WHERE("name like #{name} || '%'");
  12.       ORDER_BY(orderByColumn);
  13.     }}.toString();
  14.   }
  15.   // 如果使用 @Param,就可以只定义需要使用的参数
  16.   public static String buildGetUsersByName(@Param("orderByColumn") final String orderByColumn) {
  17.     return new SQL(){{
  18.       SELECT("*");
  19.       FROM("users");
  20.       WHERE("name like #{name} || '%'");
  21.       ORDER_BY(orderByColumn);
  22.     }}.toString();
  23.   }
  24. }
复制代码
这是一个在全局配置下让所有映射方法在同一个 sql provider 类里面的例子(3.5.6 后可用):
  1. Configuration configuration = new Configuration();
  2. configuration.setDefaultSqlProviderType(TemplateFilePathProvider.class); // 让所有映射方法在同一个 sql provider 类里面
  3. // ...
复制代码
  1. // 在 sql provider 注解上可以省略 type/value 属性
  2. // 如果省略,MyBatis 将使用 defaultSqlProviderType 所指定的类
  3. public interface UserMapper {
  4.   @SelectProvider // 等价于 @SelectProvider(TemplateFilePathProvider.class)
  5.   User findUser(int id);
  6.   @InsertProvider // 等价于 @InsertProvider(TemplateFilePathProvider.class)
  7.   void createUser(User user);
  8.   @UpdateProvider // 等价于 @UpdateProvider(TemplateFilePathProvider.class)
  9.   void updateUser(User user);
  10.   @DeleteProvider // 等价于 @DeleteProvider(TemplateFilePathProvider.class)
  11.   void deleteUser(int id);
  12. }
复制代码
以下例子展示了 ProviderMethodResolver(3.5.1 后可用)的默认实现利用方法:
  1. @SelectProvider(UserSqlProvider.class)
  2. List<User> getUsersByName(String name);
  3. // 在你的 provider 类中实现 ProviderMethodResolver 接口
  4. class UserSqlProvider implements ProviderMethodResolver {
  5.   // 默认实现中,会将映射器方法的调用解析到实现的同名方法上
  6.   public static String getUsersByName(final String name) {
  7.     return new SQL(){{
  8.       SELECT("*");
  9.       FROM("users");
  10.       if (name != null) {
  11.         WHERE("name like #{value} || '%'");
  12.       }
  13.       ORDER_BY("id");
  14.     }}.toString();
  15.   }
  16. }
复制代码
这个例子显现了如何在声明注解时利用 databaseId 属性(3.5.5后可用):
  1. @Select(value = "SELECT SYS_GUID() FROM dual", databaseId = "oracle") // 如果 DatabaseIdProvider 提供的是 "oracle",使用这条语句
  2. @Select(value = "SELECT uuid_generate_v4()", databaseId = "postgres") // 如果 DatabaseIdProvider 提供的是 "postgres",使用这条语句
  3. @Select("SELECT RANDOM_UUID()") // 如果 DatabaseIdProvider 没有配置或者没有对应的 databaseId, 使用这条语句
  4. String generateId();
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

风雨同行

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表