前言
本文主要講解Mybatis的以下知識點:
-
Mybatis快取
- 一級快取
- 二級快取
- 與Ehcache整合
-
Mapper代理
- 使用Mapper代理就不用寫實現類了
-
逆向工程
- 自動生成程式碼
Mybatis快取
快取的意義
- 將使用者經常查詢的資料放在快取(記憶體)中,使用者去查詢資料就不用從磁碟上(關係型資料庫資料檔案)查詢,從快取中查詢,從而提高查詢效率,解決了高併發系統的效能問題。
mybatis提供一級快取和二級快取
- mybatis一級快取是一個SqlSession級別,sqlsession只能訪問自己的一級快取的資料
- 二級快取是跨sqlSession,是mapper級別的快取,對於mapper級別的快取不同的sqlsession是可以共享的。
看完上面對Mybatis的快取的解釋,我們發現Mybatis的快取和Hibernate的快取是極為相似的..
Mybatis一級快取
Mybatis的一級快取原理:
第一次發出一個查詢sql,sql查詢結果寫入sqlsession的一級快取中,快取使用的資料結構是一個map<key,value>
- key:hashcode+sql+sql輸入引數+輸出引數(sql的唯一標識)
- value:使用者資訊
同一個sqlsession再次發出相同的sql,就從快取中取不走資料庫。如果兩次中間出現commit操作(修改、新增、刪除),本sqlsession中的一級快取區域全部清空,下次再去快取中查詢不到所以要從資料庫查詢,從資料庫查詢到再寫入快取。
Mybatis一級快取值得注意的地方:
- Mybatis預設就是支援一級快取的,並不需要我們配置.
- mybatis和spring整合後進行mapper代理開發,不支援一級快取,mybatis和spring整合,spring按照mapper的模板去生成mapper代理物件,模板中在最後統一關閉sqlsession。
Mybatis二級快取
二級快取原理:
二級快取的範圍是mapper級別(mapper同一個名稱空間),mapper以名稱空間為單位建立快取資料結構,結構是map<key、value>。
Mybatis二級快取配置
需要我們在Mybatis的配置檔案中配置二級快取
<!-- 全域性配置引數 -->
<settings>
<!-- 開啟二級快取 -->
<setting name="cacheEnabled" value="true"/>
</settings>
上面已經說了,二級快取的範圍是mapper級別的,因此我們的Mapper如果要使用二級快取,還需要在對應的對映檔案中配置..
<cache/>
查詢結果對映的pojo序列化
mybatis二級快取需要將查詢結果對映的pojo實現 java.io.serializable介面,如果不實現則丟擲異常:
org.apache.ibatis.cache.CacheException: Error serializing object. Cause: java.io.NotSerializableException: cn.itcast.mybatis.po.User
二級快取可以將記憶體的資料寫到磁碟,存在物件的序列化和反序列化,所以要實現java.io.serializable介面。
如果結果對映的pojo中還包括了pojo,都要實現java.io.serializable介面。
禁用二級快取
對於變化頻率較高的sql,需要禁用二級快取:
在statement中設定useCache=false可以禁用當前select語句的二級快取,即每次查詢都會發出sql去查詢,預設情況是true,即該sql使用二級快取。、、
<select id="findOrderListResultMap" resultMap="ordersUserMap" useCache="false">
重新整理快取
有的同學到這裡可能會有一個疑問:為什麼快取我們都是在查詢語句中配置??而使用增刪改的時候,快取預設就會被清空【重新整理了】???
快取其實就是為我們的查詢服務的,對於增刪改而言,如果我們的快取儲存了增刪改後的資料,那麼再次讀取時就會讀到髒資料了!
我們在特定的情況下,還可以單獨配置重新整理快取【但不建議使用】flushCache,預設是的true
<update id="updateUser" parameterType="cn.itcast.mybatis.po.User" flushCache="false">
update user set username=#{username},birthday=#{birthday},sex=#{sex},address=#{address} where id=#{id}
</update>
瞭解Mybatis快取的一些引數
mybatis的cache引數只適用於mybatis維護快取。
flushInterval(重新整理間隔)可以被設定為任意的正整數,而且它們代表一個合理的毫秒形式的時間段。預設情況是不設定,也就是沒有重新整理間隔,快取僅僅呼叫語句時重新整理。
size(引用數目)可以被設定為任意正整數,要記住你快取的物件數目和你執行環境的可用記憶體資源數目。預設值是1024。
readOnly(只讀)屬性可以被設定為true或false。只讀的快取會給所有呼叫者返回快取物件的相同例項。因此這些物件不能被修改。這提供了很重要的效能優勢。可讀寫的快取會返回快取物件的拷貝(通過序列化)。這會慢一些,但是安全,因此預設是false。
如下例子:
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
這個更高階的配置建立了一個 FIFO 快取,並每隔 60 秒重新整理,存數結果物件或列表的 512 個引用,而且返回的物件被認為是隻讀的,因此在不同執行緒中的呼叫者之間修改它們會導致衝突。可用的收回策略有, 預設的是 LRU:
1.LRU – 最近最少使用的:移除最長時間不被使用的物件。
2.FIFO – 先進先出:按物件進入快取的順序來移除它們。
3.SOFT – 軟引用:移除基於垃圾回收器狀態和軟引用規則的物件。
4.WEAK – 弱引用:更積極地移除基於垃圾收集器狀態和弱引用規則的物件。
mybatis和ehcache快取框架整合
ehcache是專門用於管理快取的,Mybatis的快取交由ehcache管理會更加得當..
在mybatis中提供一個cache介面,只要實現cache介面就可以把快取資料靈活的管理起來。
整合jar包
- mybatis-ehcache-1.0.2.jar
- ehcache-core-2.6.5.jar
ehcache對cache介面的實現類:
ehcache.xml配置資訊
這個xml配置檔案是配置全域性的快取管理方案
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
<!--diskStore:快取資料持久化的目錄 地址 -->
<diskStore path="F:developehcache" />
<defaultCache
maxElementsInMemory="1000"
maxElementsOnDisk="10000000"
eternal="false"
overflowToDisk="false"
diskPersistent="true"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU">
</defaultCache>
</ehcache>
如果我們Mapper想單獨擁有一些特性,需要在mapper.xml中單獨配置
<!-- 單位:毫秒 -->
<cache type="org.mybatis.caches.ehcache.EhcacheCache">
<property name="timeToIdleSeconds" value="12000"/>
<property name="timeToLiveSeconds" value="3600"/>
<!-- 同ehcache引數maxElementsInMemory -->
<property name="maxEntriesLocalHeap" value="1000"/>
<!-- 同ehcache引數maxElementsOnDisk -->
<property name="maxEntriesLocalDisk" value="10000000"/>
<property name="memoryStoreEvictionPolicy" value="LRU"/>
</cache>
應用場景與侷限性
應用場景
對查詢頻率高,變化頻率低的資料建議使用二級快取。
對於訪問多的查詢請求且使用者對查詢結果實時性要求不高,此時可採用mybatis二級快取技術降低資料庫訪問量,提高訪問速度
業務場景比如:
- 耗時較高的統計分析sql、
- 電話賬單查詢sql等。
實現方法如下:通過設定重新整理間隔時間,由mybatis每隔一段時間自動清空快取,根據資料變化頻率設定快取重新整理間隔flushInterval,比如設定為30分鐘、60分鐘、24小時等,根據需求而定。
侷限性
mybatis侷限性
mybatis二級快取對細粒度的資料級別的快取實現不好,比如如下需求:對商品資訊進行快取,由於商品資訊查詢訪問量大,但是要求使用者每次都能查詢最新的商品資訊,此時如果使用mybatis的二級快取就無法實現當一個商品變化時只重新整理該商品的快取資訊而不重新整理其它商品的資訊,因為mybaits的二級快取區域以mapper為單位劃分,當一個商品資訊變化會將所有商品資訊的快取資料全部清空。解決此類問題需要在業務層根據需求對資料有針對性快取。
Mapper代理方式
Mapper代理方式的意思就是:程式設計師只需要寫dao介面,dao介面實現物件由mybatis自動生成代理物件。
經過我們上面的幾篇博文,我們可以發現我們的DaoImpl是十分重複的…
1 dao的實現類中存在重複程式碼,整個mybatis操作的過程程式碼模板重複(先建立sqlsession、呼叫sqlsession的方法、關閉sqlsession)
2、dao的實現 類中存在硬編碼,呼叫sqlsession方法時將statement的id硬編碼。
以前的重複程式碼和硬編碼如下:
public class StudentDao {
public void add(Student student) throws Exception {
//得到連線物件
SqlSession sqlSession = MybatisUtil.getSqlSession();
try{
//對映檔案的名稱空間.SQL片段的ID,就可以呼叫對應的對映檔案中的SQL
sqlSession.insert("StudentID.add", student);
sqlSession.commit();
}catch(Exception e){
e.printStackTrace();
sqlSession.rollback();
throw e;
}finally{
MybatisUtil.closeSqlSession();
}
}
public static void main(String[] args) throws Exception {
StudentDao studentDao = new StudentDao();
Student student = new Student(3, "zhong3", 10000D);
studentDao.add(student);
}
}
Mapper開發規範
想要Mybatis幫我們自動生成Mapper代理的話,我們需要遵循以下的規範:
1、mapper.xml中namespace指定為mapper介面的全限定名
- 此步驟目的:通過mapper.xml和mapper.java進行關聯。
2、mapper.xml中statement的id就是mapper.java中方法名
3、mapper.xml中statement的parameterType和mapper.java中方法輸入引數型別一致
4、mapper.xml中statement的resultType和mapper.java中方法返回值型別一致.
再次說明:statement就是我們在mapper.xml檔案中名稱空間+sql指定的id
Mapper代理返回值問題
mapper介面方法返回值:
- 如果是返回的單個物件,返回值型別是pojo型別,生成的代理物件內部通過selectOne獲取記錄
- 如果返回值型別是集合物件,生成的代理物件內部通過selectList獲取記錄。
Mybatis解決JDBC程式設計的問題
1、資料庫連結建立、釋放頻繁造成系統資源浪費從而影響系統效能,如果使用資料庫連結池可解決此問題。
- 解決:在SqlMapConfig.xml中配置資料連結池,使用連線池管理資料庫連結。
2、Sql語句寫在程式碼中造成程式碼不易維護,實際應用sql變化的可能較大,sql變動需要改變java程式碼。
- 解決:將Sql語句配置在XXXXmapper.xml檔案中與java程式碼分離。
3、向sql語句傳引數麻煩,因為sql語句的where條件不一定,可能多也可能少,佔位符需要和引數一一對應。
- 解決:Mybatis自動將java物件對映至sql語句,通過statement中的parameterType定義輸入引數的型別。
4、對結果集解析麻煩,sql變化導致解析程式碼變化,且解析前需要遍歷,如果能將資料庫記錄封裝成pojo物件解析比較方便。
- 解決:Mybatis自動將sql執行結果對映至java物件,通過statement中的resultType定義輸出結果的型別。
Mybatis逆向工程
在Intellij idea下,沒有學習Maven的情況下使用Mybatis的逆向工程好像有點複雜,資料太少了…找到的資料好像也行不通…
於是學完Maven之後,我就再來更新Idea下使用Mybatis的逆向工程配置…
借鑑博文:http://blog.csdn.net/for_my_life/article/details/51228098
修改pom.xml檔案
向該工程新增逆向工程外掛..
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>asdf</groupId>
<artifactId>asdf</artifactId>
<version>1.0-SNAPSHOT</version>
<build>
<finalName>zhongfucheng</finalName>
<plugins>
<plugin>
<groupId>org.mybatis.generator</groupId>
<artifactId>mybatis-generator-maven-plugin</artifactId>
<version>1.3.2</version>
<configuration>
<verbose>true</verbose>
<overwrite>true</overwrite>
</configuration>
</plugin>
</plugins>
</build>
</project>
generatorConfig.xml配置檔案
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration
PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
"http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">
<generatorConfiguration>
<!--
<properties resource="conn.properties" />
-->
<!-- 處理1,這裡的jar包位置可能需要修改 -->
<classPathEntry location="C:mybatisMavenlibmysql-connector-java-5.1.7-bin.jar"/>
<!-- 指定執行環境是mybatis3的版本 -->
<context id="testTables" targetRuntime="MyBatis3">
<commentGenerator>
<!-- 是否取消註釋 -->
<property name="suppressAllComments" value="true" />
<!-- 是否生成註釋代時間戳 -->
<property name="suppressDate" value="true" />
</commentGenerator>
<!-- 處理2 jdbc 連線資訊,看看庫是否存在 -->
<jdbcConnection driverClass="com.mysql.jdbc.Driver"
connectionURL="jdbc:mysql://localhost:3306/scm?useUnicode=true&characterEncoding=UTF-8" userId="root" password="root">
</jdbcConnection>
<!--處理3 targetPackage指定模型在生成在哪個包 ,targetProject指定專案的src,-->
<javaModelGenerator targetPackage="zhongfucheng.entity"
targetProject="src/main/java">
<!-- 去除欄位前後空格 -->
<property name="trimStrings" value="false" />
</javaModelGenerator>
<!--處理4 配置SQL對映檔案生成資訊 -->
<sqlMapGenerator targetPackage="zhongfucheng.dao"
targetProject="src/main/java" />
<!-- 處理5 配置dao介面生成資訊-->
<javaClientGenerator type="XMLMAPPER" targetPackage="zhongfucheng.dao" targetProject="src/main/java" />
<table tableName="account" domainObjectName="Account"/>
<table tableName="supplier" domainObjectName="Supplier"/>
</context>
</generatorConfiguration>
使用外掛步驟
最後生成程式碼
如果對我們上面generatorConfig.xml配置的包資訊不清楚的話,那麼可以看一下我們的完整專案結構圖…
因為我們在Idea下是不用寫對應的工程名字的,而在eclipse是有工程名字的。
總結
- Mybatis的一級快取是sqlSession級別的。只能訪問自己的sqlSession內的快取。如果Mybatis與Spring整合了,Spring會自動關閉sqlSession的。所以一級快取會失效的。
- 一級快取的原理是map集合,Mybatis預設就支援一級快取
- 二級快取是Mapper級別的。只要在Mapper名稱空間下都可以使用二級快取。需要我們自己手動去配置二級快取
- Mybatis的快取我們可以使用Ehcache框架來進行管理,Ehcache實現Cache介面就代表使用Ehcache來環境Mybatis快取。
-
由於之前寫的DaoImpl是有非常多的硬編碼的。可以使用Mapper代理的方式來簡化開發
- 名稱空間要與JavaBean的全類名相同
- sql片段語句的id要與Dao介面的方法名相同
- 方法的引數和返回值要與SQL片段的接收引數型別和返回型別相同。
> 如果文章有錯的地方歡迎指正,大家互相交流。**習慣在微信看技術文章,想要獲取更多的Java資源的同學,可以關注微信公眾號:Java3y**