女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

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

類隔離的使用場景

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

前言

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

類隔離是什么?

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

使用場景

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

服務(wù)A pom.xml:

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

服務(wù)B pom.xml:

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

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

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

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

解決方案

大體的解決思路就是,在不改變業(yè)務(wù)代碼的前提下, 業(yè)務(wù)A調(diào)用 1.0.0 版本的消息工具類, 業(yè)務(wù)B調(diào)用2.0.0版本的消息工具類,因此需要JVM能夠利用自定義類加載器加載所需的類或關(guān)聯(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è)務(wù)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[]{};
  }

最終結(jié)果與預(yù)期的結(jié)果不一致

  • 預(yù)期結(jié)果:業(yè)務(wù)A的MessageServiceImpl與MessageUtil由CustomerFindClass加載
  • 實際結(jié)果:業(yè)務(wù)A的MessageServiceImpl由CustomerFindClass加載,而MessageUtil由sun.misc.AppClassLoader加載。
  • 分析:由于JVM類加載的雙親委托機制,業(yè)務(wù)A調(diào)用消息工具類時,類加載器(CustomerFindClass)會委托父類加載器(AppClassLoader)加載類,如果存在,則不再執(zhí)行自身的findClass方法加載,導(dǎo)致結(jié)果不理想。(main 方法類默認(rèn)情況下都是由 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è)務(wù)A的MessageServiceImpl與MessageUtil由CustomerFindClass加載

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

總結(jié)

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

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

    關(guān)注

    33

    文章

    8933

    瀏覽量

    153191
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4368

    瀏覽量

    64182
  • 微服務(wù)
    +關(guān)注

    關(guān)注

    0

    文章

    145

    瀏覽量

    7671
  • 類加載器
    +關(guān)注

    關(guān)注

    0

    文章

    6

    瀏覽量

    973
收藏 人收藏

    評論

    相關(guān)推薦
    熱點推薦

    太陽膜測試儀的技術(shù)原理和應(yīng)用場景

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

    敏捷合成器的技術(shù)原理和應(yīng)用場景

    敏捷合成器,作為一種高性能的信號發(fā)生器,其技術(shù)原理和應(yīng)用場景值得深入探討。技術(shù)原理敏捷合成器的技術(shù)原理主要基于先進的頻率合成技術(shù)和數(shù)字信號處理技術(shù)。它通常具有寬頻率范圍、快速建立和可編程的相位、頻率
    發(fā)表于 02-20 15:25

    可展示三種RS485應(yīng)用場景的半雙工參考設(shè)計包括BOM及層圖

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

    MOS管的應(yīng)用場景

    mos管的應(yīng)用場景,你了解么?低壓MOS管可稱為金屬氧化物半導(dǎo)體場效應(yīng)管,因為低壓MOS管具有良好的開關(guān)特性,廣泛應(yīng)用在電子開關(guān)的電路中。如開關(guān)電源,電動馬達(dá)、照明調(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)制波形應(yīng)用場景

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

    =>的使用場景有哪些

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

    運放電路有哪些應(yīng)用場景?

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

    藍(lán)牙低功耗常見的應(yīng)用場景及架構(gòu)

    淺談藍(lán)牙低功耗(BLE)的幾種常見的應(yīng)用場景及架構(gòu)
    發(fā)表于 06-15 09:51

    FPGA的應(yīng)用場景

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

    ARM的技術(shù)特征是什么?應(yīng)用場景有哪些?

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

    MS9331的應(yīng)用場景是什么?

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

    RK3308的特點及應(yīng)用場景是什么?

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

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

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

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

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