色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術(shù)文章
文章詳情頁

Java基于自定義類加載器實(shí)現(xiàn)熱部署過程解析

瀏覽:3日期:2022-09-04 15:12:59

熱部署:

熱部署就是在不重啟應(yīng)用的情況下,當(dāng)類的定義即字節(jié)碼文件修改后,能夠替換該Class創(chuàng)建的對(duì)象。一般情況下,類的加載都是由系統(tǒng)自帶的類加載器完成,且對(duì)于同一個(gè)全限定名的java類,只能被加載一次,而且無法被卸載。可以使用自定義的 ClassLoader 替換系統(tǒng)的加載器,創(chuàng)建一個(gè)新的 ClassLoader,再用它加載 Class,得到的 Class 對(duì)象就是新的(因?yàn)椴皇峭粋€(gè)類加載器),再用該 Class 對(duì)象創(chuàng)建一個(gè)實(shí)例,從而實(shí)現(xiàn)動(dòng)態(tài)更新。如:修改 JSP 文件即生效,就是利用自定義的 ClassLoader 實(shí)現(xiàn)的。

還需要?jiǎng)?chuàng)建一個(gè)守護(hù)線程,不斷地檢查class文件是否被修改過,通過判斷文件的上次修改時(shí)間實(shí)現(xiàn)。

演示:

原來的程序:

Java基于自定義類加載器實(shí)現(xiàn)熱部署過程解析

修改后重新編譯:

Java基于自定義類加載器實(shí)現(xiàn)熱部署過程解析

代碼:

package Dynamic; import java.io.*;import java.nio.file.Files;import java.nio.file.Path;import java.util.concurrent.TimeUnit; public class ClassLoadStudy { public static void main(String[] args) throws Exception { HotDeploy hot = new HotDeploy('Dynamic.Task'); hot.monitor(); while (true) { TimeUnit.SECONDS.sleep(2); hot.getTask().run(); } }} // 熱部署 class HotDeploy { private static volatile Runnable instance; private final String FILE_NAME; private final String CLASS_NAME; public HotDeploy(String name) { CLASS_NAME = name; // 類的完全限定名 name = name.replaceAll('.', '/') + '.class'; FILE_NAME = (getClass().getResource('/') + name).substring(6); // 判斷class文件修改時(shí)間使用,substring(6)去掉開頭的file:/ } // 獲取一個(gè)任務(wù) public Runnable getTask() { if (instance == null) { // 雙重檢查鎖,單例,線程安全 synchronized (HotDeploy.class) {if (instance == null) { try { instance = createTask(); } catch (Exception e) { e.printStackTrace(); }} } } return instance; } // 創(chuàng)建一個(gè)任務(wù),重新加載 class 文件 private Runnable createTask() { try { Class clazz = MyClassLoader.getLoader().loadClass(CLASS_NAME); if (clazz != null)return (Runnable)clazz.newInstance(); } catch (Exception e) { e.printStackTrace(); } return null; } // 監(jiān)視器,監(jiān)視class文件是否被修改過,如果是的話,則重新加載 public void monitor() throws IOException { Thread t = new Thread(()->{ try {long lastModified = Files.getLastModifiedTime(Path.of(FILE_NAME)).toMillis();while(true) { Thread.sleep(500); long now = Files.getLastModifiedTime(Path.of(FILE_NAME)).toMillis(); if(now != lastModified) { // 如果class文件被修改過了 lastModified = now; instance = createTask(); // 重新加載 }} } catch (InterruptedException | IOException e) {e.printStackTrace(); } }); t.setDaemon(true); // 守護(hù)線程 t.start(); }} // 自定義的類加載器class MyClassLoader extends ClassLoader { @Override public Class<?> findClass(String name) throws ClassNotFoundException { try { String fileName = '/' + name.replaceAll('.', '/') + '.class'; InputStream is = getClass().getResourceAsStream(fileName); byte[] b = is.readAllBytes(); return defineClass(name, b, 0, b.length); } catch (IOException e) { throw new ClassNotFoundException(name); } } public static MyClassLoader getLoader() { return new MyClassLoader(); }}

遇到的坑:

剛開始自定義類加載器時(shí),重寫的是 loadClass(String name) 方法,但不斷地報(bào)錯(cuò),后來明白了,因?yàn)?Task 類實(shí)現(xiàn)了 Java.lang.Runnable 接口,且重寫 loadClass 方法破壞了雙親委派機(jī)制,導(dǎo)致了自定義的類加載器去加載 java.lang.Runnable,但被Java安全機(jī)制禁止了所以會(huì)報(bào)錯(cuò)。defineClass調(diào)用preDefineClass,preDefineClass 會(huì)檢查包名,如果以java開頭,就會(huì)拋出異常,因?yàn)樽層脩糇远x的類加載器來加載Java自帶的類庫會(huì)引起混亂。

于是又重寫findClass 方法,但還是不行,findClass方法總是得不到執(zhí)行,因?yàn)榫幾g好的類是在 classpath 下的,而自定義的 ClassLoader 的父加載器是 AppClassLoader,由于雙親委派機(jī)制,類就會(huì)被 Application ClassLoader來加載了。因此自定義的 findClass 方法就不會(huì)被執(zhí)行。解決方法是,向構(gòu)造器 ClassLoader(ClassLoader parent) 傳入null,或傳入 getSystemClassLoader().getParent()。

還有就是路徑問題:

path不以 / 開頭時(shí),默認(rèn)是從此類所在的包下取資源;path 以 / 開頭時(shí),則是從ClassPath根下獲取; URL getClass.getResource(String path) InputStream getClass().getResourceAsStream(String path) getResource('') 返回當(dāng)前類所在的包的路徑 getResource('/') 返回當(dāng)前的 classpath 根據(jù)路徑 path 不能以 / 開始,path 是從 classpath 根開始算的, 因?yàn)閏lassloader 不是用戶自定義的類,所以沒有相對(duì)路徑的配置文件可以獲取,所以默認(rèn)都是從哪個(gè)classpath 路徑下讀取,自然就沒有必要以 / 開頭了 。 URL Class.getClassLoader().getResource(String path) InputStream Class.getClassLoader().getResourceAsStream(String path)

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 亚洲操综合| 欧美性色生活片天天看99 | 日本乱人伦在线观看免费 | 欧美老熟妇bbbb毛片 | 特级片视频| 蜜桃88av | 2020国产成人免费视频 | 国产免费高清 | 国产午夜人做人视频羞羞 | 欧美视频在线观看 | 手机在线色| 国产免费一区二区三区在线观看 | 一区二区三区免费精品视频 | 一及黄色 | 国产亚洲精品免费 | 白白在线观看永久免费视频 | 色一伦一情一区二区三区 | 成 人 黄 色 激 情视频网站 | 嫩草影院在线观看网站成人 | 三级毛片免费观看 | 国产成视频 | 久国产精品视频 | 成年视频国产免费观看 | 成人禁在线观看午夜亚洲 | 成人在线不卡视频 | 色视频在线免费 | 成人在线观看国产 | 中文字幕播放 | 国产区二区 | 久久巨乳 | 毛片a级三毛片免费播放 | 精品久久久久久综合网 | 久操免费在线视频 | 中文字幕在线一区二区在线 | 美国毛片基地a级e片 | 亚洲成a人v大片在线观看 | 亚洲男人天堂久久 | 日本一级特黄aa毛片免费观看 | 伊人黄色片 | 国产精品九九免费视频 | 欧美综合成人 |