Weblogic環境下hibernate、antlr類載入衝突問題分析及解決方案

来源:http://www.cnblogs.com/bingjava/archive/2016/03/31/5342802.html
-Advertisement-
Play Games

公司應用項目在客戶部署時經常遇到此類問題,為避免實施部署時增加配置量,花了點時間找到了此問題的終極解決辦法(方案二、修改org.hibernate.hql.ast.HqlLexer的源代碼)。在此進行記錄本問題的分析解決方案。 一、問題現象描述: 1、異常信息: 'weblogic.kernel.D ...


公司應用項目在客戶部署時經常遇到此類問題,為避免實施部署時增加配置量,花了點時間找到了此問題的終極解決辦法(方案二、修改org.hibernate.hql.ast.HqlLexer的源代碼)。在此進行記錄本問題的分析解決方案。

一、問題現象描述:

1、異常信息:

'weblogic.kernel.Default (self-tuning)']…

org.hibernate.QueryException: ClassNotFoundException: org.hibernate.hql.ast.HqlToken [

 

    at org.hibernate.hql.ast.HqlLexer.panic(HqlLexer.java:57)

    at antlr.CharScanner.setTokenObjectClass(CharScanner.java:340)

    at org.hibernate.hql.ast.HqlLexer.setTokenObjectClass(HqlLexer.java:31)

    at antlr.CharScanner.<init>(CharScanner.java:51)

    at antlr.CharScanner.<init>(CharScanner.java:60)

    at org.hibernate.hql.antlr.HqlBaseLexer.<init>(HqlBaseLexer.java:56)

    at org.hibernate.hql.antlr.HqlBaseLexer.<init>(HqlBaseLexer.java:53)

    at org.hibernate.hql.antlr.HqlBaseLexer.<init>(HqlBaseLexer.java:50)

    at org.hibernate.hql.ast.HqlLexer.<init>(HqlLexer.java:26)

    at org.hibernate.hql.ast.HqlParser.getInstance(HqlParser.java:44)

    at org.hibernate.hql.ast.QueryTranslatorImpl.parse(QueryTranslatorImpl.java:242)

    atorg.hibernate.hql.ast.QueryTranslatorImpl.doCompile(QueryTranslatorImpl.java:157)

    at org.hibernate.hql.ast.QueryTranslatorImpl.compile(QueryTranslatorImpl.java:111)

    at org.hibernate.engine.query.HQLQueryPlan.<init>(HQLQueryPlan.java:77)

    at org.hibernate.engine.query.HQLQueryPlan.<init>(HQLQueryPlan.java:56)

    at org.hibernate.engine.query.QueryPlanCache.getHQLQueryPlan(QueryPlanCache.java:72)

    at org.hibernate.impl.SessionFactoryImpl.checkNamedQueries(SessionFactoryImpl.java:402)

    at org.hibernate.impl.SessionFactoryImpl.<init>(SessionFactoryImpl.java:352)

    at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:1294)

2、查詢weblogic安裝目錄下的antlr包:

3、應用中引用的是hibernate3和antlr_2.7.6的jar

 

二、原因分析

根據以上異常信息查看hibernate及antlr的源代碼:

org.hibernate.hql.ast.HqlLexer的部分代碼:

public void setTokenObjectClass(String cl) {        

super.setTokenObjectClass( HqlToken.class.getName() );

        }

以上super.setTokenObjectClass 方法就是antlr.CharScanner類中定義的方法:

public void setTokenObjectClass(String paramString) {

    try {

      this.tokenObjectClass = Utils.loadClass(paramString);

    } catch (ClassNotFoundException localClassNotFoundException) {

      panic("ClassNotFoundException: " + paramString);

    }

}

此方法的關鍵部分:Utils.loadClass(paramString);即在hibernate在解析hql是會採用此工具載入org.hibernate.hql.ast.HqlToken類(即HqlLexer類中的setTokenObjectClass方法)。此處會發生什麼情況呢,請看Utils.loadClass的源代碼:

  static {

          if ("true".equalsIgnoreCase(System.getProperty("ANTLR_DO_NOT_EXIT", "false")))

              useSystemExit = false;

          if ("true".equalsIgnoreCase(System.getProperty("ANTLR_USE_DIRECT_CLASS_LOADING", "false")))

              useDirectClassLoading = true;

  }

 

    /** Thanks to Max Andersen at JBOSS and Scott Stanchfield */

    public static Class loadClass(String name) throws ClassNotFoundException {

        try {

            ClassLoader contextClassLoader =Thread.currentThread(). getContextClassLoader();

            if (!useDirectClassLoading && contextClassLoader!=null ) {

                return contextClassLoader.loadClass(name);

            }

            return Class.forName(name);

        }

        catch (Exception e) {

            return Class.forName(name);

        }

    }

 

從以上的代碼可看處,載入org.hibernate.hql.ast.HqlToken類的類載入器是weblogic啟動類載入器(不管是Thread.currentThread().getContextClassLoader()還是Class.forName,其中Class.forName採用的是Reflection.getCallerClass()的類載入器,即antlr的類載入器),並非應用類載入器。Weblogic類路徑下已經存在antlrjar包了,系統會優先使用weblogic下的antlr包,而weblogic類路徑下並沒有hibnatejar包,所以在載入org.hibernate.hql.ast.HqlToken類是會拋出ClassNotFoundException: org.hibernate.hql.ast.HqlToken異常。

 

三、解決方案

    方案一、修改weblogic類載入器中antlr載入的優先順序

  1. 拷貝應用中的包antlr-2.7.6.jar%WL_HOME%\server\lib 
  2. 修改% mydomain%\startWebLogic.cmd(.sh)  
    set CLASSPATH之前加上下麵一句: 
    set PRE_CLASSPATH=%WL_HOME%\server\lib\antlr-2.7.6.jar; 
    set CLASSPATH之後加上下麵一句: 
    set CLASSPATH=%PRE_CLASSPATH%;%CLASSPATH% 

此方案並不總是有效(尤其是在osgi類型項目或者同一個weblogic域下部署多個項目的情況),當然根據筆者遇到的情況成功率也在95%以上。當此方案無效時可以採用方案二。

    方案二、修改org.hibernate.hql.ast.HqlLexer的源代碼:

        載入org.hibernate.hql.ast.HqlToken類是,直接用hibernate所在類classload載入即可:

    將原來的代碼:

  public void setTokenObjectClass(String cl) {        

super.setTokenObjectClass( HqlToken.class.getName() );

        }

 

    修改為:直接將hqltoken類賦值給this.tokenObjectClass

public void setTokenObjectClass(String cl) {

    this.tokenObjectClass = HqlToken.class;

}

    


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

-Advertisement-
Play Games
更多相關文章
  • 自己項目中使用到了 結果在不同的windows 操作系統中,程式的運行不一致,在windows server 2008上可以很好的運行,但是到了windows7上去卡死了!!!!!!!!!!!!!!!!!!!!!! p.waitFor() 卡死了或者報錯: 如果改為: 則直接卡死了。如果將 p.wa ...
  • 歷屆試題 核桃的數量 時間限制:1.0s 記憶體限制:256.0MB 時間限制:1.0s 記憶體限制:256.0MB 問題描述 小張是軟體項目經理,他帶領3個開發組。工期緊,今天都在加班呢。為鼓舞士氣,小張打算給每個組發一袋核桃(據傳言能補腦)。他的要求是: 1. 各組的核桃數量必須相同 2. 各組內必 ...
  • 廢話不多說,上來貼代碼最實在,哈哈! 以下代碼量有點多,不過這都是在下一手一手敲出來的,小巧好用,把以下代碼複製出來,放到相應的hpp文件即可,VS,GCC下均能編譯通過 接下來是traits庫的完整代碼 ...
  • 主題:構建一個基於SpringMVC的HelloWord Web 項目 目的:快速體驗什麼是SpringMVC 方案: 1、創建工程,命名:SpringMVC 2、導包 3、在SRC下添加spring-mvc.xml配置文件 (註意:名字可以隨便取,最好就是看上就知道是什麼) 4、在web.xml配 ...
  • 如何實現類型名跟類型的對應, 我們很容易想到map, 沒錯, 就是使用map實現的. std::map<std::string, .....>, 等下, 第二部分該填什麼類型, 一個函數指針, auto create()? auto只是占位符, 編譯器好像不會讓你通過吧. 我們需要一種容器, 可以存 ...
  • <!--?xml version="1.0" encoding="UTF-8" standalone="no"?--> 當你在swift中編寫一個類時,預設其中任何屬性,方法都能被外部訪問的。有的時候我們不希望屬性或方法被外部訪問,希望私有化。 在swift中有三個關鍵字 public : 公共訪問 ...
  • 重定向和轉發有一個重要的不同:當使用轉發時,JSP容器將使用一個內部的方法來調用目標頁面,新的頁面繼續處理同一個請求,而瀏覽器將不會知道這個過程。 與之相反,重定向方式的含義是第一個頁面通知瀏覽器發送一個新的頁面請求。因為,當你使用重定向時,瀏覽器中所顯示的URL會變成新頁面的URL, 而當使用轉發 ...
  • java中equal和==的區別: equal方法除非進行了重寫(比如String類和基本類型包裝類),否則都是進行對象引用的比較。而==除了比較基本類型時是比較值,其他情況均是比較引用地址。 java適配器模式? 將原來的實現介面所有方法變成了繼承實現了介面方法的抽象類(adapter),然後再實 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...