吴忠躺衫网络科技有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

類隔離的使用場景

科技綠洲 ? 來源:Java技術指北 ? 作者:Java技術指北 ? 2023-10-08 15:20 ? 次閱讀

前言

由于微服務的快速迭代、持續集成等特性,越來越多的團隊更傾向于它。但是也體現出了一些問題,比如在基礎設施建設過程中,需要把通用功能下沉,把現有大而全的基礎設施按領域拆分,考慮需要兼容現有生產服務,會產生不同的依賴版本,有時不注意就可以引發問題。比如本文遇到的依賴包版本沖突問題,以及如何利用類隔離技術解決的分析。

類隔離是什么?

類隔離是一種通過類加載器實現加載所需類的實現方式,使得不同版本類間隔離,避免了使用沖突問題,最終的效果就是不同模塊的內容被不同的類加載器加載,滿足同一環境下同時兼容不同接口實現類。

使用場景

比如業務服務A和業務服務B均需要消息通知等,均依賴消息中間件,但所引用版本不一致,導致最終只有一個版本加載到JVM,在某一個服務調用時會出現 NoSuchMethodError或NoSuchClassError問題,這就很難排查出來,沒準會影響項目進度,最終月度的績效(“雞腿”)不保。

服務A pom.xml:

< !-- common-message-- >
        < dependency >
            < groupId >com.lgy< /groupId >
            < artifactId >spring-common-message< /artifactId >
            < version >1.0.0< version >
        < /dependency >

服務B pom.xml:

< !-- common-message-- >
        < dependency >
            < groupId >com.lgy< /groupId >
            < artifactId >spring-common-message< /artifactId >
            < version >2.0.0< version >
        < /dependency >

業務調用流程:

// 業務A調用微信服務通知
 MessageUtil.sendMessage(content,peopleId,templateId,"wechat");
 // 業務B調用微信服務通知
 MessageUtil.sendToWechat(content,peopleId,templateId);

JVM最終加載的為 2.0.0 版本的依賴,導致業務A在調用時拋異常java.lang.NoSuchMethodError。

解決方案

大體的解決思路就是,在不改變業務代碼的前提下, 業務A調用 1.0.0 版本的消息工具類, 業務B調用2.0.0版本的消息工具類,因此需要JVM能夠利用自定義類加載器加載所需的類或關聯的類。

實現思路

  • 重寫類加載器,實現自定義類加載(java.lang.ClassLoader)
  • 重寫類加載函數
    • 重寫 findClass(String name)
    • 重寫 loadClass(String name)

涉及的知識點

  • JVM加載過程:加載-》鏈接-》初始化(具體后續介紹)
  • 雙親委派機制:委托父加載器查詢;如果父加載器查詢不到,則調用自身的findClass加載

重寫findClass:

import java.io.*;
 import java.util.HashMap;
 import java.util.Map;

 public class CustomerFindClass extends ClassLoader {
  private Map< String, String > classPathMap = new HashMap<  >();
  public CustomerFindClass() {
   // 業務A的自定義類加載器
   classPathMap.put("com.lgy.businessA.service.impl.MessageServiceImpl", "E:/dataway-demo/example/target/classes/com/lgy/businessA/service/impl/MessageServiceImpl.class");
   classPathMap.put("com.lgy.v1.message.util.MessageUtil", "E:/dataway-demo/example/target/classes/com/lgy/v1/message/util/MessageUtil.class");
  }
  
  /**
  * findClass方式加載類
  */
  @Override
  protected Class< ? > findClass(String name) throws ClassNotFoundException {
   String classPath = classPathMap.get(name);
   File file = new File(classPath);
   if (!file.exists()) {
    throw new ClassNotFoundException();
   }
   byte[] bytes = getClassData(file);
   if (null == bytes || 0 == bytes.length) {
    throw new ClassNotFoundException();
   }
   return defineClass(bytes, 0, bytes.length);
  }
  
  private byte[] getClassData(File file) {
   try (InputStream ins = new FileInputStream(file); 
     ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
    byte[] buffer = new byte[4096];
    int bytesNumRead = 0;
    while ((bytesNumRead = ins.read(buffer)) != -1) {
     baos.write(buffer, 0, bytesNumRead);
    }
    return baos.toByteArray();
   } catch (FileNotFoundException e) {
    e.printStackTrace();
   } catch (IOException e) {
    e.printStackTrace();
   }
   return new byte[]{};
  }

最終結果與預期的結果不一致

  • 預期結果:業務A的MessageServiceImpl與MessageUtil由CustomerFindClass加載
  • 實際結果:業務A的MessageServiceImpl由CustomerFindClass加載,而MessageUtil由sun.misc.AppClassLoader加載。
  • 分析:由于JVM類加載的雙親委托機制,業務A調用消息工具類時,類加載器(CustomerFindClass)會委托父類加載器(AppClassLoader)加載類,如果存在,則不再執行自身的findClass方法加載,導致結果不理想。(main 方法類默認情況下都是由 JDK 自帶的 AppClassLoader 加載的)。

重寫loadClass

private ClassLoader classLoader;
 
 /**
 * 重新loadClass方法
 */
 @Override
    protected Class< ? > loadClass(String name, boolean resolve) throws ClassNotFoundException {
        Class result = null;
        try {
            //這里要使用 JDK 的類加載器加載 java.lang 包里面的類
            result = classLoader.loadClass(name);
        } catch (Exception e) {
            // ignore error
        }
        if (null != result) {
            return result;
        }
        String classPath = classPathMap.get(name);
        File file = new File(classPath);
        if (!file.exists()) {
            throw new ClassNotFoundException();
        }
        byte[] bytes = getClassData(file);
        if (null == bytes || 0 == bytes.length) {
            throw new ClassNotFoundException();
        }
        return defineClass(bytes, 0, bytes.length);
    }

滿足業務A的MessageServiceImpl與MessageUtil由CustomerFindClass加載

注意:這種方式破壞了雙親委托機制,但由于重寫了loadClass方法,所有類均會有CustomerFindClass加載器加載,需要過濾出不需要隔離的類,如java.lang包下的類,需要由ExtClassLoader 來加載。

總結

本文分享的方式是從類加載器方向出發,實現最終的類隔離,避免了不同模塊間不同類的沖突,其中順便也簡單帶過了jvm類加載相關的知識點,也算是一勞多得,后續會結合實際使用場景進一步分析。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 接口
    +關注

    關注

    33

    文章

    8694

    瀏覽量

    151929
  • 函數
    +關注

    關注

    3

    文章

    4346

    瀏覽量

    62979
  • 微服務
    +關注

    關注

    0

    文章

    142

    瀏覽量

    7433
  • 類加載器
    +關注

    關注

    0

    文章

    6

    瀏覽量

    941
收藏 人收藏

    評論

    相關推薦

    AG32VF-MIPI應用場景

    MIPI接口技術在圖像和視頻傳輸中的應用越來越廣泛,應用場景也在不斷拓展,而不僅限于移動設備。MIPI接口在物聯網、智能家居、智能監控、智能電視、智能汽車等領域也得到廣泛應用。 MIPI還可
    發表于 01-22 08:56

    太陽膜測試儀的技術原理和應用場景

    太陽膜測試儀的技術原理和應用場景可以詳細闡述如下:技術原理太陽膜測試儀的技術原理主要基于光學測量和物理定律。具體來說,它通過模擬太陽光中的各種波長(主要是紫外線、可見光和紅外線)的輻射,來檢測太陽膜
    發表于 09-29 14:18

    可展示三種RS485應用場景的半雙工參考設計包括BOM及層圖

    描述 RS485(隔離式、非隔離式)是適用于電網基礎設施空間的常用接口,也是新設計的設備上最重要的選項之一。TIDA-00308 允許客戶針對三種不同應用場景通過 TI RS485 器件以及該
    發表于 09-21 09:15

    MOS管的應用場景

    mos管的應用場景,你了解么?低壓MOS管可稱為金屬氧化物半導體場效應管,因為低壓MOS管具有良好的開關特性,廣泛應用在電子開關的電路中。如開關電源,電動馬達、照明調光等!下面銀聯寶科技就跟大家一起
    發表于 11-14 09:24

    this的使用場景及與C,Java中的this的區別

    【JS】this有哪些使用場景?跟C,Java中的this有什么區別?如何改變this的值?
    發表于 03-11 10:17

    CP-OFMD調制波形應用場景

    圖1、5G的應用場景5G使用5G多載波波形來為智能手機,辦公室,工廠自動化,智能電網,智慧城市,物聯網,M2M,M2X等多種設備提供應用平臺。5G新無線電(5G NR)根據應用場景可分為三大服務
    發表于 06-18 06:51

    =>的使用場景有哪些

    使用場景
    發表于 10-27 13:25

    運放電路有哪些應用場景?

    運放電路的七大應用場景
    發表于 03-11 07:49

    藍牙低功耗常見的應用場景及架構

    淺談藍牙低功耗(BLE)的幾種常見的應用場景及架構
    發表于 06-15 09:51

    FPGA的應用場景

    目錄文章目錄目錄FPGAFPGA 的應用場景FPGA 的技術難點FPGA 的工作原理FPGA 的體系結構FPGA 的開發FPGA 的使用FPGA 的優缺點參考文檔FPGAFPGA(Field
    發表于 07-28 08:43

    ARM的技術特征是什么?應用場景有哪些?

    ARM的技術特征是什么?應用場景有哪些?
    發表于 11-05 07:32

    MS9331的應用場景是什么?

    MS9331的應用場景是什么?
    發表于 02-11 06:41

    RK3308的特點及應用場景是什么?

    RK3308的特點及應用場景是什么?
    發表于 03-09 08:04

    labview 和 wincc 的區別 使用場景

    labview 和 wincc 的區別 使用場景 都是上位機軟件,都可以做監控軟件 wincc的名氣也比較大 對比的資料較少 寫這些文章的人,從自己的從事的行業出發,帶有自己的思維 使用的場景 肯定
    發表于 10-27 18:01

    榮湃隔離驅動器的應用場景有哪些?

    隔離驅動器的一大應用場景是取代數字隔離器+MOS驅動器的分立設計,高絕緣耐壓要求的半橋驅動應用中,部分系統采用了雙通道數字隔離器配合高低邊MOS驅動器的設計。
    的頭像 發表于 04-01 15:42 ?2268次閱讀
    澳门百家乐官网赢钱| 永利高平台网址| 宝马会百家乐官网娱乐城| 玩百家乐官网怎么能赢吗| 一直对百家乐官网很感兴趣.zibo太阳城娱乐城| 百家乐官网庄闲的冷热| 壹贰博网站| 大发888xp缺少 casino| KK百家乐的玩法技巧和规则| 威尼斯人娱乐城送钱| 大发888 软件| 真人百家乐体验金| 大发888娱乐捕鱼游戏| 盛京棋牌网| 七乐国际| 百家乐官网好不好玩| 缅甸百家乐官网赌博现场下载| 万龙百家乐官网的玩法技巧和规则 | 大发888国际娱乐| 百家乐官网风云论坛| 神人百家乐官网赌场| 24山风水实例| 欢乐谷百家乐的玩法技巧和规则 | 娱乐城百家乐官网可以代理吗 | 玩百家乐的高手| 大发888官方下| 健康| 什么是24山风水| 菲律宾百家乐开户| 香港六合彩彩图| 百家乐官网韩泰阁| 圆梦城百家乐娱乐城| 百家乐博彩,| 赌王百家乐官网的玩法技巧和规则| 百家乐路子技巧| 青鹏棋牌游戏大厅v3.0| 缅甸百家乐官网赌博有假吗| 百家乐游戏作弊| 三门县| 百家乐登封代理| 百家乐平注常赢玩法|