1、前言
前面介紹了MyBaits中兩種使用遊標的方式來避免搜查內容過大致使JVM OOM,這兩種方式被稱爲是客戶端side的遊標,由於mysql client每次從mysqlserver獲取一條記錄,這雖然解決了OOM,可是會形成IO網絡比較頻繁,那麼有沒有一種方式可讓mysql server一次返回fetchsize個記錄那?答案是確定的。java
2、MySQL Server Side Cursor
2.1 使用
要使用MySQL Server Side遊標須要知足下面條件:mysql
必須是select語句sql
設置了fetchSize>0
在mapper文件裏面設置數據庫設置了useCursorFetch=true數組
數據集類型爲ResultSet.TYPE_FORWARD_ONLY服務器
數據集併發設置爲ResultSet.CONCUR_READ_ONLY
在數據庫連接後面設置:jdbc:mysql://localhost:3306/test?useCursorFetch=true」微信Server versions 5.0.5 or newer網絡
這是由於代碼層面作了下面判斷:併發
// we only create cursor-backed result sets if// a) The query is a SELECT// b) The server supports it// c) We know it is forward-only (note this doesn't preclude updatable result sets)// d) The user has set a fetch sizeif (this.resultFields != null && this.useCursorFetch && getResultSetType() == ResultSet.TYPE_FORWARD_ONLY
&& getResultSetConcurrency() == ResultSet.CONCUR_READ_ONLY && getFetchSize() > 0) {
packet.writeByte(OPEN_CURSOR_FLAG); // usingCursor = true;} else {
packet.writeByte((byte) 0); // placeholder for flags}//// Server versions 5.0.5 or newer will only open a cursor and set this flag if they can, otherwise they punt and go back to mysql_store_results()// behavior//if (this.connection.versionMeetsMinimum(5, 0, 5)) {
usingCursor = (this.serverStatus & SERVER_STATUS_CURSOR_EXISTS) != 0;
}
2.2 原理簡單介紹
服務器邊的遊標是mysqlclient先從mysqlserver獲取fetchSize個記錄放到mysqlclient的遊標內部的數組裏面,遊標獲取的時候是從數組裏面獲取數據,若是數組爲空了,在向mysql server獲取fetchSize個記錄。app
3、總結對比
服務器邊的遊標的使用的確能夠減小網絡IO,可是這是使用佔用MySQL服務器資源來實現的,由於服務器這邊確定要作維護每次返回fetchSize的事情,因此並不見的使用服務器端遊標比客戶端的就好。
本文分享自微信公衆號 - 技術原始積累(gh_805ebfd2deb0)。
若有侵權,請聯繫 support@oschina.cn 刪除。
本文參與「OSC源創計劃」,歡迎正在閱讀的你也加入,一塊兒分享。