本文主要講解Mybatis的如下知識點:php
緩存的意義java
mybatis提供一級緩存和二級緩存mysql
看完上面對Mybatis的緩存的解釋,咱們發現Mybatis的緩存和Hibernate的緩存是極爲類似的..程序員
Mybatis的一級緩存原理:spring
第一次發出一個查詢sql,sql查詢結果寫入sqlsession的一級緩存中,緩存使用的數據結構是一個map<key,value>sql
同一個sqlsession再次發出相同的sql,就從緩存中取不走數據庫。若是兩次中間出現commit操做(修改、添加、刪除),本sqlsession中的一級緩存區域所有清空,下次再去緩存中查詢不到因此要從數據庫查詢,從數據庫查詢到再寫入緩存。數據庫
Mybatis一級緩存值得注意的地方:apache
二級緩存原理:編程
二級緩存的範圍是mapper級別(mapper同一個命名空間),mapper以命名空間爲單位建立緩存數據結構,結構是map<key、value>。緩存
須要咱們在Mybatis的配置文件中配置二級緩存
<!-- 全局配置參數 -->
<settings>
<!-- 開啓二級緩存 -->
<setting name="cacheEnabled" value="true"/>
</settings>
複製代碼
上面已經說了,二級緩存的範圍是mapper級別的,所以咱們的Mapper若是要使用二級緩存,還須要在對應的映射文件中配置..
<cache/>
複製代碼
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的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 – 弱引用:更積極地移除基於垃圾收集器狀態和弱引用規則的對象。
複製代碼
ehcache是專門用於管理緩存的,Mybatis的緩存交由ehcache管理會更加得當..
在mybatis中提供一個cache接口,只要實現cache接口就能夠把緩存數據靈活的管理起來。
ehcache對cache接口的實現類:
這個xml配置文件是配置全局的緩存管理方案
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
<!--diskStore:緩存數據持久化的目錄 地址 -->
<diskStore path="F:\develop\ehcache" />
<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二級緩存技術下降數據庫訪問量,提升訪問速度
業務場景好比:
實現方法以下:經過設置刷新間隔時間,由mybatis每隔一段時間自動清空緩存,根據數據變化頻率設置緩存刷新間隔flushInterval,好比設置爲30分鐘、60分鐘、24小時等,根據需求而定。
mybatis侷限性
mybatis二級緩存對細粒度的數據級別的緩存實現很差,好比以下需求:對商品信息進行緩存,因爲商品信息查詢訪問量大,可是要求用戶每次都能查詢最新的商品信息,此時若是使用mybatis的二級緩存就沒法實現當一個商品變化時只刷新該商品的緩存信息而不刷新其它商品的信息,由於mybaits的二級緩存區域以mapper爲單位劃分,當一個商品信息變化會將全部商品信息的緩存數據所有清空。解決此類問題須要在業務層根據需求對數據有針對性緩存。
Mapper代理方式的意思就是:程序員只須要寫dao接口,dao接口實現對象由mybatis自動生成代理對象。
通過咱們上面的幾篇博文,咱們能夠發現咱們的DaoImpl是十分重複的...
1 dao的實現類中存在重複代碼,整個mybatis操做的過程代碼模板重複(先建立sqlsession、調用sqlsession的方法、關閉sqlsession)
二、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);
}
}
複製代碼
想要Mybatis幫咱們自動生成Mapper代理的話,咱們須要遵循如下的規範:
一、mapper.xml中namespace指定爲mapper接口的全限定名
二、mapper.xml中statement的id就是mapper.java中方法名
三、mapper.xml中statement的parameterType和mapper.java中方法輸入參數類型一致
四、mapper.xml中statement的resultType和mapper.java中方法返回值類型一致.
再次說明:statement就是咱們在mapper.xml文件中命名空間+sql指定的id
mapper接口方法返回值:
一、數據庫連接建立、釋放頻繁形成系統資源浪費從而影響系統性能,若是使用數據庫連接池可解決此問題。
二、Sql語句寫在代碼中形成代碼不易維護,實際應用sql變化的可能較大,sql變更須要改變java代碼。
三、向sql語句傳參數麻煩,由於sql語句的where條件不必定,可能多也可能少,佔位符須要和參數一一對應。
四、對結果集解析麻煩,sql變化致使解析代碼變化,且解析前須要遍歷,若是能將數據庫記錄封裝成pojo對象解析比較方便。
在Intellij idea下,沒有學習Maven的狀況下使用Mybatis的逆向工程好像有點複雜,資料太少了...找到的資料好像也行不通...
因而學完Maven以後,我就再來更新Idea下使用Mybatis的逆向工程配置...
借鑑博文:blog.csdn.net/for_my_life…
向該工程添加逆向工程插件..
<?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>
複製代碼
<?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:\mybatisMaven\lib\mysql-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代理的方式來簡化開發
若是文章有錯的地方歡迎指正,你們互相交流。習慣在微信看技術文章,想要獲取更多的Java資源的同窗,能夠關注微信公衆號:Java3y