欧美性猛交xxxx免费看_牛牛在线视频国产免费_天堂草原电视剧在线观看免费_国产粉嫩高清在线观看_国产欧美日本亚洲精品一5区

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

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

3天內(nèi)不再提示

類隔離的使用場景

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

前言

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

類隔離是什么?

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

使用場景

比如業(yè)務服務A和業(yè)務服務B均需要消息通知等,均依賴消息中間件,但所引用版本不一致,導致最終只有一個版本加載到JVM,在某一個服務調(diào)用時會出現(xiàn) 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 >

業(yè)務調(diào)用流程:

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

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

解決方案

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

實現(xiàn)思路

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

涉及的知識點

  • JVM加載過程:加載-》鏈接-》初始化(具體后續(xù)介紹)
  • 雙親委派機制:委托父加載器查詢;如果父加載器查詢不到,則調(diào)用自身的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() {
   // 業(yè)務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[]{};
  }

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

  • 預期結果:業(yè)務A的MessageServiceImpl與MessageUtil由CustomerFindClass加載
  • 實際結果:業(yè)務A的MessageServiceImpl由CustomerFindClass加載,而MessageUtil由sun.misc.AppClassLoader加載。
  • 分析:由于JVM類加載的雙親委托機制,業(yè)務A調(diào)用消息工具類時,類加載器(CustomerFindClass)會委托父類加載器(AppClassLoader)加載類,如果存在,則不再執(zhí)行自身的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);
    }

滿足業(yè)務A的MessageServiceImpl與MessageUtil由CustomerFindClass加載

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

總結

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

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

    關注

    33

    文章

    8712

    瀏覽量

    151998
  • 函數(shù)
    +關注

    關注

    3

    文章

    4346

    瀏覽量

    63006
  • 微服務
    +關注

    關注

    0

    文章

    143

    瀏覽量

    7443
  • 類加載器
    +關注

    關注

    0

    文章

    6

    瀏覽量

    941
收藏 人收藏

    評論

    相關推薦

    AG32VF-MIPI應用場景

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

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

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

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

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

    MOS管的應用場景

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

    this的使用場景及與C,Java中的this的區(qū)別

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

    CP-OFMD調(diào)制波形應用場景

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

    =>的使用場景有哪些

    使用場景
    發(fā)表于 10-27 13:25

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

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

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

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

    FPGA的應用場景

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

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

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

    MS9331的應用場景是什么?

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

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

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

    labview 和 wincc 的區(qū)別 使用場景

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

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

    隔離驅(qū)動器的一大應用場景是取代數(shù)字隔離器+MOS驅(qū)動器的分立設計,高絕緣耐壓要求的半橋驅(qū)動應用中,部分系統(tǒng)采用了雙通道數(shù)字隔離器配合高低邊MOS驅(qū)動器的設計。
    的頭像 發(fā)表于 04-01 15:42 ?2277次閱讀