成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

使用 Spring Boot 實現動態加載 jar 包,動態配置功能太贊了!

開發 前端
本文展示了一個使用 Spring Boot 實現動態加載、卸載 JAR 包和動態修改 YAML 配置信息的完整示例,包括項目配置的更新、相關類的實現以及使用 Thymeleaf 實現的前端頁面,為開發者提供了一個可參考的實現方案。

在當今的軟件開發中,靈活性和可擴展性是至關重要的。Spring Boot 框架為我們提供了強大的工具和機制,使得實現動態加載 jar 包和動態配置變得輕松而高效。這一特性在應對不斷變化的業務需求和復雜的運行環境時具有極大的優勢。

動態加載 jar 包的原理與優勢

動態加載 jar 包的實現基于 Java 的類加載機制。在 Java 中,類加載器負責將類的字節碼加載到 JVM 中,并創建對應的類對象。通常,Java 應用使用默認的類加載器層次結構,包括啟動類加載器、擴展類加載器和應用類加載器。然而,為了實現動態加載 jar 包,我們需要創建自定義的類加載器。

自定義類加載器繼承自 java.lang.ClassLoader 類,并覆蓋其 findClass 或 loadClass 方法來實現自定義的類查找和加載邏輯。當需要動態加載 jar 包時,自定義類加載器首先獲取 jar 包的文件路徑,然后讀取 jar 包中的字節碼數據。

通過解析字節碼數據,找到其中定義的類信息,并將其加載到 JVM 中。在這個過程中,還需要處理類的依賴關系,確保所有相關的類都能正確加載。

動態加載 jar 包帶來了諸多顯著的優勢。

首先,它極大地提高了系統的靈活性。在傳統的應用部署中,如果需要添加新的功能或修復缺陷,往往需要重新編譯、打包和部署整個應用。而通過動態加載 jar 包,可以在應用運行時直接加載新的功能模塊,無需中斷服務,實現了無縫的功能擴展和更新。

其次,它有助于降低系統的維護成本。對于一些頻繁變化的業務需求,不必因為小的功能調整而進行大規模的應用部署,減少了部署過程中的風險和人力投入。

再者,動態加載 jar 包能夠提高開發效率。開發人員可以獨立開發和測試新的功能模塊,然后在需要時將其動態加載到生產環境中,避免了與現有代碼的頻繁集成和沖突。

此外,它還為系統的模塊化設計提供了有力支持。不同的功能模塊可以封裝在獨立的 jar 包中,根據實際需求動態加載,使系統的架構更加清晰和易于管理。

項目依賴配置(pom.xml)

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.10</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.icoderoad</groupId>
    <artifactId>dynamic-loading-demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>dynamic-loading-demo</name>
    <description>Demo project for dynamic loading with Spring Boot</description>

    <properties>
        <java.version>11</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
      	<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-thymeleaf</artifactId>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>
</project>

YAML 屬性文件配置(application.yml)

# 動態配置相關屬性
dynamic:
  enabled: true
  # 其他動態配置項

后端代碼示例

DynamicConfig 類:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.core.io.Resource;
import org.springframework.core.io.UrlResource;
import org.springframework.stereotype.Component;
import org.yaml.snakeyaml.DumperOptions;
import org.yaml.snakeyaml.Yaml;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileWriter;
import java.io.IOException;
import java.net.URL;
import java.util.HashMap;
import java.util.Map;

@Component
@ConfigurationProperties(prefix = "dynamic")
public class DynamicConfig {

    private String configProperty;

    @Autowired
    private String filePath;

    public String getConfigProperty() {
        return configProperty;
    }

    public void setConfigProperty(String configProperty) {
        this.configProperty = configProperty;
        // 同步修改 YAML 文件中的配置信息
        modifyYaml(filePath, "configProperty", configProperty);
    }

    public void modifyYaml(String filePath, String key, String value) {
        try (FileInputStream inputStream = new FileInputStream(new File(filePath))) {
            Yaml yaml = new Yaml();
            Map<String, Object> config = yaml.load(inputStream);

            config.put(key, value);

            DumperOptions options = new DumperOptions();
            options.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);

            try (FileWriter writer = new FileWriter(new File(filePath))) {
                yaml.dump(config, writer, options);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

工具類 JarLoadingUtils:

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class JarLoadingUtils {

    private Map<String, ClassLoader> loadedJars = new HashMap<>();

    public void loadJars(List<String> jarPaths) throws IOException {
        for (String jarPath : jarPaths) {
            URL url = new URL(jarPath);
            CustomClassLoader classLoader = new CustomClassLoader();
            classLoader.loadJar(url.getFile());
            loadedJars.put(jarPath, classLoader);
            System.out.println("正在加載 JAR 包: " + jarPath);
        }
    }

    public void unloadJar(String jarPath) {
        ClassLoader classLoader = loadedJars.remove(jarPath);
        if (classLoader!= null) {
            // 執行卸載相關的邏輯
            System.out.println("正在卸載 JAR 包: " + jarPath);
        }
    }

    class CustomClassLoader extends URLClassLoader {

        public CustomClassLoader() {
            super(new URL[0], getParentClassLoader());
        }

        public void loadJar(String jarPath) {
            try {
                URL url = new File(jarPath).toURI().toURL();
                addURL(url);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

DynamicLoadingController 類:

import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;

import java.io.IOException;
import java.util.List;
import java.util.Map;

public class DynamicLoadingController {

    private JarLoadingUtils jarLoadingUtils;
    private DynamicConfig dynamicConfig;

    public DynamicLoadingController(JarLoadingUtils jarLoadingUtils, DynamicConfig dynamicConfig) {
        this.jarLoadingUtils = jarLoadingUtils;
        this.dynamicConfig = dynamicConfig;
    }

    @PostMapping("/dynamic/load")
    public ResponseEntity<String> loadJars(@RequestBody List<String> jarPaths) {
        try {
            jarLoadingUtils.loadJars(jarPaths);
            return ResponseEntity.status(HttpStatus.OK).body("JAR 包加載成功");
        } catch (IOException e) {
            return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("加載 JAR 包時出錯: " + e.getMessage());
        }
    }

    @PostMapping("/dynamic/unload")
    public ResponseEntity<String> unloadJar(@RequestBody String jarPath) {
        jarLoadingUtils.unloadJar(jarPath);
        return ResponseEntity.status(HttpStatus.OK).body("JAR 包卸載成功");
    }

    @PostMapping("/dynamic/config/update")
    public ResponseEntity<String> updateConfig(@RequestBody Map<String, String> configData) {
        String key = configData.get("key");
        String value = configData.get("value");
        dynamicConfig.setConfigProperty(value);
        return ResponseEntity.status(HttpStatus.OK).body("配置更新成功");
    }
}

核心的后端代碼實現如下:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class DynamicLoadingApplication implements ApplicationRunner {

    @Autowired
    private DynamicConfig dynamicConfig;

    @Autowired
    private JarLoadingUtils jarLoadingUtils;

    public static void main(String[] args) {
        SpringApplication.run(DynamicLoadingApplication.class, args);
    }

    @Override
    public void run(ApplicationArguments args) throws Exception {
        // 模擬動態加載 jar 包的邏輯
        List<String> jarPaths = new ArrayList<>();
        jarPaths.add("path/to/your/jar/file1.jar");
        jarPaths.add("path/to/your/jar/file2.jar");
        jarLoadingUtils.loadJars(jarPaths);
    }
}

使用 Thymeleaf 的前端頁面:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <title>動態加載配置頁面</title>
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
    <script>
        $(document).ready(function() {
            $("#loadButton").click(function() {
                $.ajax({
                    url: "/dynamic/load",
                    type: "POST",
                    success: function(response) {
                        $("#loadResult").text(response);
                    },
                    error: function(xhr, status, error) {
                        $("#loadResult").text("加載出錯: " + error);
                    }
                });
            });

            $("#unloadButton").click(function() {
                var jarPath = $("#unloadPath").val();
                $.ajax({
                    url: "/dynamic/unload",
                    type: "POST",
                    data: JSON.stringify({ "jarPath": jarPath }),
                    contentType: "application/json",
                    success: function(response) {
                        $("#unloadResult").text(response);
                    },
                    error: function(xhr, status, error) {
                        $("#unloadResult").text("卸載出錯: " + error);
                    }
                });
            });

            $("#updateButton").click(function() {
                var key = $("#updateKey").val();
                var value = $("#updateValue").val();
                $.ajax({
                    url: "/dynamic/config/update",
                    type: "POST",
                    data: JSON.stringify({ "key": key, "value": value }),
                    contentType: "application/json",
                    success: function(response) {
                        $("#updateResult").text(response);
                    },
                    error: function(xhr, status, error) {
                        $("#updateResult").text("更新出錯: " + error);
                    }
                });
            });
        });
    </script>
</head>
<body>
    <h2>動態操作</h2>
    <button id="loadButton">觸發動態加載</button>
    <p id="loadResult"></p>
    <form>
        <input type="text" id="unloadPath" placeholder="輸入要卸載的 JAR 路徑" />
        <button id="unloadButton">觸發動態卸載</button>
    </form>
    <p id="unloadResult"></p>
    <form>
        <input type="text" id="updateKey" placeholder="輸入配置鍵" />
        <input type="text" id="updateValue" placeholder="輸入配置值" />
        <button id="updateButton">觸發動態配置更新</button>
    </form>
    <p id="updateResult"></p>
</body>
</html>

總結

本文展示了一個使用 Spring Boot 實現動態加載、卸載 JAR 包和動態修改 YAML 配置信息的完整示例,包括項目配置的更新、相關類的實現以及使用 Thymeleaf 實現的前端頁面,為開發者提供了一個可參考的實現方案。

責任編輯:武曉燕 來源: 路條編程
相關推薦

2021-10-18 12:04:22

Spring BootJava開發

2021-10-18 10:36:31

Spring Boot插件Jar

2025-06-18 07:32:16

SpringJar動態加載

2024-09-05 09:35:58

CGLIBSpring動態代理

2022-08-02 11:24:22

菜鳥Python網站自動簽到

2024-02-04 09:19:00

Nacos動態化線程池

2024-11-08 14:11:09

2022-07-14 10:38:39

動態標簽Spring

2021-12-02 09:31:22

Python 代碼進度條

2023-12-08 14:37:51

接口jar包開發

2009-06-22 15:10:00

java 編程AOP

2022-02-08 17:07:54

Spring BooSpring Aop日志記錄

2025-02-07 09:11:04

JSON對象策略

2024-09-03 10:44:32

2021-09-01 10:07:43

開發零搭建Groovy

2024-12-20 17:29:34

SpringBootAOP開發

2025-01-17 09:11:51

2009-08-28 16:14:26

C#實現加載動態庫

2022-06-16 10:38:24

URL權限源代碼

2024-01-26 07:48:10

SpringKafka提升
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲91| 国产在线播放一区二区三区 | 精品视频在线一区 | 一级久久久久久 | 久久久精品一区 | 欧美激情精品久久久久久变态 | 国产精品揄拍一区二区 | 91精品国产综合久久久久久蜜臀 | 在线观看www高清视频 | 亚洲综合天堂网 | 色网站入口 | 国产精品久久国产愉拍 | 亚洲精品一区在线 | 国产综合久久 | 免费看91| 欧美性久久 | 91精品国产一区二区三区香蕉 | 久久夜视频 | 在线视频中文字幕 | 日韩精品一区二区三区 | av一区二区三区四区 | 亚洲一区中文字幕 | 国产精品视频久久 | 国产亚洲精品区 | 一级黄色av电影 | www精品美女久久久tv | 91精品国产一区二区三区 | 亚洲欧美日韩精品久久亚洲区 | 亚洲黄色av | 男女在线网站 | 亚洲成人一区 | 国产成人综合在线 | 亚洲九九色 | 99精品在线观看 | 99精品欧美一区二区三区综合在线 | 日本成人在线网址 | 亚洲一区二区不卡在线观看 | 一区二区三区在线播放 | 久久99精品久久久久 | 91最新在线视频 | 久久久涩 |