Mybatis的缺陷

来源:http://www.cnblogs.com/linjisong/archive/2016/10/29/6010787.html
-Advertisement-
Play Games

Mybatis是業界非常流行的持久層框架,輕量級、易用,在金融IT領域完全是領軍地位,比Hibernate更受歡迎,優勢非常多,也是非常值得我們學習的。但Mybatis並不盡善盡美,其自身的設計、編碼也還有許多不足,甚至是缺陷,這篇文章來簡要討論一下這些缺陷: 1.Mybatis使用DTD作為XML ...


Mybatis是業界非常流行的持久層框架,輕量級、易用,在金融IT領域完全是領軍地位,比Hibernate更受歡迎,優勢非常多,也是非常值得我們學習的。但Mybatis並不盡善盡美,其自身的設計、編碼也還有許多不足,甚至是缺陷,這篇文章來簡要討論一下這些缺陷:

1.Mybatis使用DTD作為XML配置文件的校驗文件,但是很明顯,DTD差不多是快被淘汰的技術了,功能非常有限,擴展性非常差,擴展性非常差,擴展性非常差,可讀性也不好,Spring能夠從DTD到XSD華麗轉身,但Mybatis始終沒這個魄力。

2.版本相容性做的不好,就拿3.3.0—>3.4.0來說,按業界通用規範,第2級版本號升級,可以添加功能,但是要保證向下相容性,然而Mybatis的做法並不完全是這樣的,看一下關鍵介面StatementHandler的關鍵方法prepare:

// 3.3.0
Statement prepare(Connection connection)
      throws SQLException;

// 3.4.0
Statement prepare(Connection connection, Integer transactionTimeout)
      throws SQLException;

這裡沒有添加一個方法,而是直接在原方法中添加了一個參數!類似例子還有不少,就不一一列舉了。

3.Mybatis的插件,採用一個通用的Interceptor介面,配以@Intercepts、@Signature等註解,實現對多個組件的多種方法的攔截,看似非常靈活,在我看來其實是結構不夠清晰,實際開發時,你會把對StatementHandler和ResultSetHandler的攔截增強放在一個類裡面嗎?不會是吧(會?你當單一職責原則、開閉原則都是狗屎嗎),那有什麼必要強制使用同一個介面呢?

另外,使用@Signature註解來設別需要被攔截的組件方法,如果註解有錯,編譯也是不會報錯的,而只能等到運行時才能發現,再看上面的例子:

假設我針對3.3.0版本實現了一個插件:

@Intercepts({ 
    @Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class })
})
public class StatementHandlerInterceptor implements Interceptor {

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        return invocation.proceed();
    }

    @Override
    public Object plugin(Object target) {
        return Plugin.wrap(target, this);
    }

    @Override
    public void setProperties(Properties properties) {
        
    }
}

然後,升級為3.4.0,結果呢,編譯一直正常,但是等到運行,卻拋出異常了。

4.Mybatis的緩存簡直就是雞肋,而且不管有沒有配置需要使用緩存、是否更新緩存,都要去計算CacheKey,不使用緩存、也不更新緩存的情況下,這種計算完全是浪費。

5.Mybatis的批量執行,看下麵的一個JDBC例子:

public void testJdbcBatch(Connection conn) throws Exception {try{
    conn.setAutoCommit(false);
    batchUpdate(conn);
    clearTestData(conn);
    conn.commit();
    conn.setAutoCommit(true);
  }catch(Exception e){
    conn.rollback();
    throw e;
  }
}

private void clearTestData(Connection conn) throws SQLException {
  PreparedStatement ps = null;
  try{
    ps = conn.prepareStatement("delete TABLE_NAME1 where FIELD_NAME1 = ? ");
    ps.setString(1, "TEST");
    int d = ps.executeUpdate();
    System.out.println("delete counts : " + d);
  }finally{
    try{
      ps.close();
    }catch(Exception e){}
  }
}

private void batchUpdate(Connection conn) throws SQLException {
  PreparedStatement ps = null;
  try{
    String sql = "INSERT INTO TABLE_NAME2(FIELD_NAME1, FIELD_NAME2, FIELD_NAME2)VALUES(?,?,?)";
    ps = conn.prepareStatement(sql);
    for(int i = 0; i < 10; i++){
      String random = RandomStringUtils.randomAlphabetic(8);
      ps.setString(1, "TEST");//FIELD_NAME1
      ps.setString(2, "數據" + random);//FIELD_NAME2
      ps.setString(3, "參數" + random);//FIELD_NAME3
      ps.addBatch();
    }
    int[] rs = ps.executeBatch();
  }finally{
    try{
      ps.close();
    }catch(Exception e){}
  }
}

代碼沒有什麼違和感,能夠執行正常,也可以按預期的回滾,也就是說同一個事務中的同一個connection,可以同時運行普通sql和batch,但是你在同一個事務的SqlSession中試試,反饋給你的是——不能在同一個事務中切換執行方式!

6、資料庫產品的相容性:Mybatis把SQL的控制權交給了開發人員,於是從道德上占據了制高點——你寫的不相容,那是你自己的水平不行!但,這是一個真正的優秀框架的正確姿勢嗎?為什麼就不能提供一些輔助性的相容實施?比如說在Oracle中被奉為神明的DECODE函數,是否可以在SqlMapper中提供一個<decode>標簽,在後面默默的修改成CASE WHEN?或者說,官方不提供沒有關係,但你得提供擴展方式啊,於是又回到了:擴展性非常差,擴展性非常差,擴展性非常差。重要的事說三遍,但,我已經說六遍了。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • UEditor For ASP.Net Core Use Qiniu 此項目為UEditor提供文件管理; 後端服務使用 ASP.Net Core; 使用七牛提供的雲存儲; 項目地址 http://git.oschina.net/joesjiang/BaiduEditorUseQiniu.NetCo ...
  • 列印一個介面、方法的運行時間在程式中是很容易遇到的一件事情;現在,我就分享一個我在工作中使用的臨時列印日誌的方法和結合 Stopwatch 列印測量某個時間間隔的運行時間的方法。 Stopwatch 實例可以很好的測量一個時間間隔的運行時間;以下例子是比較常用到的: 引用命名空間: using Sy ...
  • using System; using System.Collections.Generic; using System.Linq; using System.Text; //本人英語不太好,嘿嘿。。 //其實介面多繼承和單繼承的形式是一樣的。 //博客里介面單繼承,單繼承會了,看這個就很輕鬆了 n ...
  • 上一章筆者講到關於Asp.NET的知識點。瞭解Asp.NET基本的知識點之後,我們在來學習關於C#的MVC框架就簡單多了。顯然本章就是來介紹一下關於Asp.NET MVC。對於MVC的思想筆者相信大家已經很清楚了。筆者在這裡就不多說了。我有個同學做Asp.NET開發的。他跟我講過一句話——我都快忘了 ...
  • 只要根據錯誤提示,找到相應文件夾下的\obj\Debug文件夾下的 相應名字.csproj.FileListAbsolute.txt, 打開並刪除含有'<<<<<<< .mine'或類似的欄位,重新編譯即可。 項目的.csproj 下相關'<<<<<<< .mine'或類似的欄位也需要刪除! ...
  • 在項目中遇到列印日誌或者處理異常這再常見不過了,通常service層處理日誌每個方法都寫一遍logger.info()太麻煩了。而aop恰好可以解決這個問題,寫一個aop的類就能處理這個問題了,spring更是擁有強大的aop功能。那就看看springaop到底多麼簡便吧! ...
  • 測試環境:php5.3.29 sort_flags 可以用以下值改變排序的行為: 排序類型標記: SORT_REGULAR - 正常比較單元(不改變類型) SORT_NUMERIC - 單元被作為數字來比較 SORT_STRING - 單元被作為字元串來比較 SORT_LOCALE_STRING - ...
  • 在我心中類載入一直是很迷人的,主要的原因是覺得為何能夠這個過程是如何運作的。最近在總結自己以前學過的東西,所以也就總結一下類載入器這方面的東西。 對於為何將載入類這樣一個那麼重要的事放在JVM外部在實現,我覺的最主要的原因是為了提供更好的擴展性吧,讓程式員自己去決定如何載入一個類。根據書上的說法是這 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...