Java小型操作系统模拟(采用策略模式结合反射进行搭建,支持一些简单的命令)

news2025/1/23 6:12:50

Java小型操作系统模拟

    • 项目说明
    • 第一阶段:反射结合策略模式搭建基本的命令结构
    • 第二阶段:注解结合反射与策略模式,将结构进一步规范
    • 第三阶段:开启新的窗口,将控制台输入切换到新窗口中,同时创建右键菜单,使效果贴近命令行
    • 第四阶段,发现IDEA的编译与命令行编译不一致且导致类装载出错的情况进行处理--重新编写初始化的装载方法
    • 第五阶段,解决打包为jar后的装载问题及采用脚本自动编译
    • 项目下载

项目说明

主要是为了学习Java反射的知识,以及对操作系统的一些概念进行回顾,搭建了一个小型的操作系统,包括基本的一些命令,如:clear、help、cd、mkdir、ls等;同时支持用户及角色创建,密码登录等,相关文件采用加密存储在本地;同时采用资源分配的形式设计了磁盘资源和内存资源,采用首次适应、循环首次适应、最佳适应、最坏适应等策略;采用Java的Socket通信设计了简单的socket通信模式

第一阶段:反射结合策略模式搭建基本的命令结构

采用配置文件的形式指定命令的名称、描述与对应类的全路径,采用反射的形式创建对象并装载到容器中;策略模式结合HashMap的形式,可以较为方便的获取到命令对象,并执行对应的策略方法。

策略模式的体现,采用Map进行命令策略的管理

package os.strategy;

import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;

/**
 * @author bbyh
 * @date 2023-07-27 21:19
 * @description
 */
public abstract class BaseStrategy {
    private static final Map<String, BaseStrategy> STRATEGY_FACTORY = new HashMap<>(4);
    private static final Map<String, String> STRATEGY_DOC = new HashMap<>(4);
    private final String name;
    private final String doc;

    public BaseStrategy(String name, String doc) {
        this.name = name;
        this.doc = doc;
    }

    protected final void register() {
        STRATEGY_FACTORY.put(name, this);
        STRATEGY_DOC.put(name, doc);
    }

    public abstract void execute();

    public static BaseStrategy getStrategy(String name) {
        if (!STRATEGY_FACTORY.containsKey(name)) {
            throw new UnsupportedOperationException("该策略还不支持");
        }
        return STRATEGY_FACTORY.get(name);
    }

    public static Set<String> getCommandNameSet() {
        return new HashSet<>(STRATEGY_FACTORY.keySet());
    }

    public static Map<String, String> getCommandDoc() {
        return new HashMap<>(STRATEGY_DOC);
    }
}

采用配置文件结合反射装载对象;分为只装载指定目录,或装载指定目录及其子目录,要求都是基础策略类的子类

package os.util;

import os.strategy.BaseStrategy;

import java.io.*;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashSet;
import java.util.Properties;

/**
 * @author bbyh
 * @date 2023/7/28 12:19
 */
public class OsApplication {
    private static final String PACKAGE_PREFIX = "src/";
    private static final String DEFAULT_STRATEGY_PACKAGE = "strategy.impl";
    private static final String OS_COMMAND_PROPERTIES = "os-command-config.properties";
    private static final HashSet<Class<?>> CLASS_SET = new HashSet<>(4);

    private static void load() {
        for (Class<?> instance : CLASS_SET) {
            if (instance.getSuperclass() == BaseStrategy.class) {
                try (InputStream inputStream = Files.newInputStream(Paths.get(OS_COMMAND_PROPERTIES))) {
                    Properties properties = new Properties();
                    properties.load(inputStream);
                    if (properties.containsKey(instance.getName())) {
                        String[] split = properties.getProperty(instance.getName()).split(",");
                        try {
                            Constructor<?> constructor = instance.getConstructor(String.class, String.class);
                            Object newInstance = constructor.newInstance(split[0], split[1]);
                            Method register = newInstance.getClass().getSuperclass().getDeclaredMethod("register");
                            register.setAccessible(true);
                            register.invoke(newInstance);
                        } catch (NoSuchMethodException | InvocationTargetException |
                                 IllegalAccessException | InstantiationException e) {
                            throw new RuntimeException(e);
                        }
                    }
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    public static void runWithSpecifiedDirectory(Class<?> application) {
        initSpecifiedDirectory(application.getPackage().getName() + "." + DEFAULT_STRATEGY_PACKAGE);
        load();
    }

    private static void initSpecifiedDirectory(String scanPackage) {
        File basePackage = new File(PACKAGE_PREFIX + scanPackage.replace(".", "/"));
        File[] files = basePackage.listFiles(file -> file.getName().endsWith(".java"));
        loadClasses(scanPackage, files);
    }

    private static void loadClasses(String scanPackage, File[] files) {
        if (files != null) {
            for (File file : files) {
                try {
                    Class<?> instance = Class.forName(scanPackage + "." + file.getName().replace(".java", ""));
                    CLASS_SET.add(instance);
                } catch (ClassNotFoundException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    public static void runWithRecursiveDirectory(Class<?> application) {
        initRecursiveDirectory(application.getPackage().getName());
        load();
    }

    private static void initRecursiveDirectory(String scanPackage) {
        File basePackage = new File(PACKAGE_PREFIX + scanPackage.replace(".", "/"));
        File[] files = basePackage.listFiles(file -> {
            if (file.isDirectory()) {
                initRecursiveDirectory(scanPackage + "." + file.getName());
            }
            return file.getName().endsWith(".java");
        });
        loadClasses(scanPackage, files);
    }
}

配置文件

os.strategy.impl.ClearStrategy=clear,清屏命令,清空屏幕
os.strategy.impl.HelpStrategy=help,帮助命令,查看当前系统支持的命令

项目结构
在这里插入图片描述

第二阶段:注解结合反射与策略模式,将结构进一步规范

采用类似SpringBoot的启动方式,自动装载指定目录下的命令类,采用注解的形式设置命令的名称与描述

采用注解设置启动装载包目录名,以及各个策略类的目录名称与相关说明

package os.annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * @author bbyh
 * @date 2023/7/2 15:29
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface StrategyScan {
    String value() default "strategy.impl";
}

help命令,简单输出系统支持的命令和说明

package os.strategy.impl;

import os.annotation.StrategyAnnotation;
import os.annotation.StrategyDocAnnotation;
import os.strategy.BaseStrategy;

import java.util.Map;
import java.util.Set;

/**
 * @author bbyh
 * @date 2023-07-27 21:26
 * @description
 */
@StrategyAnnotation(commandName = "help")
@StrategyDocAnnotation(commandDoc = "帮助命令,查看当前系统支持的命令")
public class HelpStrategy extends BaseStrategy {
    public HelpStrategy(String name, String doc) {
        super(name, doc);
    }

    @Override
    public void execute() {
        System.out.println("系统支持的命令");
        Set<String> commandNameSet = BaseStrategy.getCommandNameSet();
        Map<String, String> commandDoc = BaseStrategy.getCommandDoc();
        for (String commandName : commandNameSet) {
            System.out.println(commandName + ":" + commandDoc.get(commandName));
        }
    }
}

同样加载指定目录下携带注解的策略子类,或者加载目录及其子目录的所有符合条件的类

package os.util;

import os.annotation.StrategyAnnotation;
import os.annotation.StrategyDocAnnotation;
import os.annotation.StrategyScan;
import os.strategy.BaseStrategy;

import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashSet;

/**
 * @author bbyh
 * @date 2023/7/28 12:19
 */
public class OsApplication {
    private static final String PACKAGE_PREFIX = "src/";
    private static final HashSet<Class<?>> CLASS_SET = new HashSet<>(4);

    private static void load() {
        for (Class<?> instance : CLASS_SET) {
            if (instance.getSuperclass() == BaseStrategy.class) {
                try {
                    StrategyAnnotation nameAnnotation = instance.getAnnotation(StrategyAnnotation.class);
                    StrategyDocAnnotation docAnnotation = instance.getAnnotation(StrategyDocAnnotation.class);
                    if (nameAnnotation != null && docAnnotation != null) {
                        Constructor<?> constructor = instance.getConstructor(String.class, String.class);
                        Object newInstance = constructor.newInstance(nameAnnotation.commandName(), docAnnotation.commandDoc());
                        Method register = newInstance.getClass().getSuperclass().getDeclaredMethod("register");
                        register.setAccessible(true);
                        register.invoke(newInstance);
                    }
                } catch (NoSuchMethodException | InvocationTargetException |
                         IllegalAccessException | InstantiationException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    /**
     * 加载配置目录下的添加了StrategyAnnotation注解的类
     *
     * @param application 启动类
     */
    public static void runWithSpecifiedDirectory(Class<?> application) {
        StrategyScan strategyScan = application.getAnnotation(StrategyScan.class);
        initSpecifiedDirectory(application.getPackage().getName() + "." + strategyScan.value());
        load();
    }

    private static void initSpecifiedDirectory(String scanPackage) {
        File basePackage = new File(PACKAGE_PREFIX + scanPackage.replace(".", "/"));
        File[] files = basePackage.listFiles(file -> file.getName().endsWith(".java"));
        loadClasses(scanPackage, files);
    }

    private static void loadClasses(String scanPackage, File[] files) {
        if (files != null) {
            for (File file : files) {
                try {
                    Class<?> instance = Class.forName(scanPackage + "." + file.getName().replace(".java", ""));
                    CLASS_SET.add(instance);
                } catch (ClassNotFoundException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    /**
     * 加载配置目录及其子目录下的添加了StrategyAnnotation注解的类
     *
     * @param application 启动类
     */
    public static void runWithRecursiveDirectory(Class<?> application) {
        initRecursiveDirectory(application.getPackage().getName());
        load();
    }

    private static void initRecursiveDirectory(String scanPackage) {
        File basePackage = new File(PACKAGE_PREFIX + scanPackage.replace(".", "/"));
        File[] files = basePackage.listFiles(file -> {
            if (file.isDirectory()) {
                initRecursiveDirectory(scanPackage + "." + file.getName());
            }
            return file.getName().endsWith(".java");
        });
        loadClasses(scanPackage, files);
    }
}

项目结构
在这里插入图片描述

第三阶段:开启新的窗口,将控制台输入切换到新窗口中,同时创建右键菜单,使效果贴近命令行

经过测试发现,上面两个系统在初始化加载类时有一些小问题,因为在实际使用时都是执行class文件,而在IDEA里面的执行是以Java文件为路径的,需要进行一些简单的小修改,当前这个阶段主要以IDEA里面的运行为主,后面再对路径和初始化逻辑进行一些调整,来保证兼容性

主窗体

package os.util;

import os.strategy.BaseStrategy;

import javax.swing.*;
import java.awt.*;
import java.awt.datatransfer.Clipboard;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.event.*;
import java.io.IOException;

/**
 * @author bbyh
 * @date 2023-07-29 15:51
 * @description
 */
public class MainFrame extends JFrame {
    public static final JLabel DISPLAY = new JLabel();
    private static final String DEFAULT_WORD_DIR = "C:\\root>";

    public MainFrame() {
        setTitle("BBYH OS System");
        setSize(1000, 600);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setLocationRelativeTo(null);

        JScrollPane scrollPane = new JScrollPane(DISPLAY);
        DISPLAY.setOpaque(true);
        DISPLAY.setBackground(Color.WHITE);

        setLayout(new GridLayout(1, 1));
        add(scrollPane);
        setVisible(true);

        Font font = new Font("Consolos", Font.PLAIN, 24);
        DISPLAY.setFont(font);
        DISPLAY.setVerticalAlignment(SwingConstants.TOP);
        DISPLAY.setFocusable(true);
        init();

        // 右键菜单
        JPopupMenu rightMenu = new JPopupMenu();
        JMenuItem backgroundItem = new JMenuItem();
        JMenuItem fontItem = new JMenuItem();
        JMenuItem copyItem = new JMenuItem();

        backgroundItem.setFont(font);
        fontItem.setFont(font);
        copyItem.setFont(font);
        backgroundItem.setText("设置背景颜色");
        fontItem.setText("设置字体颜色");
        copyItem.setText("复制剪切版内容");

        rightMenu.add(backgroundItem);
        rightMenu.add(fontItem);
        rightMenu.add(copyItem);

        backgroundItem.addActionListener(evt -> {
            Color color = JColorChooser.showDialog(null, "设置背景颜色", Color.WHITE);
            if (color != null) {
                DISPLAY.setBackground(color);
            }
        });
        fontItem.addActionListener(evt -> {
            Color color = JColorChooser.showDialog(null, "设置字体颜色", Color.BLACK);
            if (color != null) {
                DISPLAY.setForeground(color);
            }
        });
        copyItem.addActionListener(evt -> {
            Clipboard clipboard = Toolkit.getDefaultToolkit().getSystemClipboard();
            Transferable transferable = clipboard.getContents(null);
            if (transferable != null) {
                if (transferable.isDataFlavorSupported(DataFlavor.stringFlavor)) {
                    try {
                        String data = (String) clipboard.getData(DataFlavor.stringFlavor);
                        appendDisplay(data);
                    } catch (UnsupportedFlavorException | IOException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        });

        DISPLAY.addKeyListener(new KeyAdapter() {
            @Override
            public void keyPressed(KeyEvent e) {
                char keyChar = e.getKeyChar();
                if (keyChar == KeyEvent.VK_BACK_SPACE && !"".equals(getContent())) {
                    DISPLAY.setText(getContent().substring(0, getContent().length() - 1));
                    appendDisplay("");
                } else if (keyChar == KeyEvent.VK_ENTER) {
                    appendDisplay("<br />");
                    try {
                        BaseStrategy.getStrategy(Context.getNextCommand()).execute();
                    } catch (Exception ex) {
                        appendDisplay(ex.getMessage() + "<br />");
                    }
                    appendDisplay(Context.getWorkDir());
                } else if (Character.isLetterOrDigit(keyChar)) {
                    appendDisplay(String.valueOf(keyChar));
                }
            }
        });

        DISPLAY.addMouseListener(new MouseAdapter() {
            @Override
            public void mouseClicked(MouseEvent e) {
                if (e.getButton() == MouseEvent.BUTTON3) {
                    rightMenu.show(DISPLAY, e.getX(), e.getY() - 10);
                }
            }
        });
    }

    private static void init() {
        Context.setWorkDir(DEFAULT_WORD_DIR);
        appendDisplay(DEFAULT_WORD_DIR);
    }

    public static void appendDisplay(String content) {
        DISPLAY.setText("<html><body style='padding: 20px 10px'>" + getContent() + content + "</body></html>");
    }

    private static String getContent() {
        return DISPLAY.getText().replace("<html><body style='padding: 20px 10px'>", "").replace("</body></html>", "");
    }
}

Context上下文操作类

package os.util;

/**
 * @author bbyh
 * @date 2023-07-27 22:23
 * @description
 */
public class Context {
    private static String workDir;

    public static String getWorkDir() {
        return workDir;
    }

    public static void setWorkDir(String workDir) {
        Context.workDir = workDir;
    }

    public static String getNextCommand() {
        String text = MainFrame.DISPLAY.getText();
        return text.substring(text.lastIndexOf(getWorkDir()) + getWorkDir().length(), text.lastIndexOf("<br /></body></html>"));
    }
}

主程序

package os;

import os.annotation.StrategyScan;
import os.util.MainFrame;
import os.util.OsApplication;

/**
 * @author bbyh
 * @date 2023-07-27 21:18
 * @description
 */
@StrategyScan
public class Main {
    public static void main(String[] args) {
        OsApplication.runWithRecursiveDirectory(Main.class);
        new MainFrame();
    }
}

测试运行效果展示
在这里插入图片描述

第四阶段,发现IDEA的编译与命令行编译不一致且导致类装载出错的情况进行处理–重新编写初始化的装载方法

采用getResource方法获取类路径,来装在class文件,同时采用脚本来解决javac命令行执行时无法编译不被别的文件依赖的策略实现类

重新编写的装载方法,采用递归装载,装载启动类目录下及其子目录中带有注解的策略实现类

package os.util;

import os.annotation.StrategyAnnotation;
import os.annotation.StrategyDocAnnotation;
import os.strategy.BaseStrategy;

import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashSet;
import java.util.Objects;

/**
 * @author bbyh
 * @date 2023/7/28 12:19
 */
public class OsApplication {
    private static final HashSet<Class<?>> CLASS_SET = new HashSet<>(4);
    private static String BASE_PATH;

    /**
     * 加载配置目录及其子目录下的添加了StrategyAnnotation注解的类
     *
     * @param application 启动类
     */
    public static void runWithRecursiveDirectory(Class<?> application) {
        BASE_PATH = Objects.requireNonNull(application.getResource("/")).getPath();
        initRecursiveDirectory(application.getPackage().getName());
        load();
    }

    private static void initRecursiveDirectory(String packageName) {
        File[] files = new File(BASE_PATH + packageName.replace(".", System.getProperty("file.separator"))).listFiles(file -> {
            if (file.isDirectory()) {
                initRecursiveDirectory(packageName + "." + file.getName());
            }
            return file.getName().endsWith(".class");
        });
        loadClasses(packageName, files);
    }

    private static void load() {
        for (Class<?> instance : CLASS_SET) {
            if (instance.getSuperclass() == BaseStrategy.class) {
                try {
                    StrategyAnnotation nameAnnotation = instance.getAnnotation(StrategyAnnotation.class);
                    StrategyDocAnnotation docAnnotation = instance.getAnnotation(StrategyDocAnnotation.class);
                    if (nameAnnotation != null && docAnnotation != null) {
                        Constructor<?> constructor = instance.getConstructor(String.class, String.class);
                        Object newInstance = constructor.newInstance(nameAnnotation.commandName(), docAnnotation.commandDoc());
                        Method register = newInstance.getClass().getSuperclass().getDeclaredMethod("register");
                        register.setAccessible(true);
                        register.invoke(newInstance);
                    }
                } catch (NoSuchMethodException | InvocationTargetException |
                         IllegalAccessException | InstantiationException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    private static void loadClasses(String packageName, File[] files) {
        if (files != null) {
            for (File file : files) {
                try {
                    Class<?> instance = Class.forName(packageName + "." + file.getName().replace(".class", ""));
                    CLASS_SET.add(instance);
                } catch (ClassNotFoundException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }
}

需要注意的一点是,目前采用getResource方法对于打包为jar包时则会报空指针错误,后续将考虑采用新方法装载来解决该问题

第五阶段,解决打包为jar后的装载问题及采用脚本自动编译

自动编译的脚本,Windows

mkdir out
cd src
javac -d ../out os/Main.java os/strategy/impl/*.java -encoding UTF-8

cd ../out

java os.Main

自动编译的脚本,Linux

#!/bin/bash
mkdir out
cd src
javac -d ../out os/Main.java os/strategy/impl/*.java -encoding UTF-8
cd ../out
java os.Main

需要注意几点小问题,脚本.sh需要在Linux下创建使用,同时需要给它赋予x执行权限(chmod +x javac_os_java.sh);然后在执行时还需要对XShell安装一下图形服务x11,参考文章:xshell-linux 启动 jmeter 报 No X11 DISPLAY variable was set, but this program performed …;不过这个Xming对中文的显示有问题,有点难办

主函数(根据启动不一样采用不一样的装载策略)

package os;

import os.annotation.StrategyScan;
import os.util.MainFrame;
import os.util.OsApplication;

import java.util.Objects;

/**
 * @author bbyh
 * @date 2023-07-27 21:18
 * @description
 */
@StrategyScan
public class Main {
    public static boolean applicationRunning = true;
    private static final String FILE_PROTOCOL = "file";
    private static final String JAR_PROTOCOL = "jar";

    public static void main(String[] args) {
        String protocol = Objects.requireNonNull(OsApplication.class.getResource("")).getProtocol();
        if (Objects.equals(protocol, FILE_PROTOCOL)) {
            OsApplication.runWithRecursiveDirectory(Main.class);
        } else if (Objects.equals(protocol, JAR_PROTOCOL)) {
            OsApplication.runWithRecursiveJar(Main.class);
        }
        new MainFrame();
    }
}

装载工具类

package os.util;

import os.annotation.StrategyAnnotation;
import os.annotation.StrategyDocAnnotation;
import os.strategy.BaseStrategy;

import java.io.File;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.Objects;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * @author bbyh
 * @date 2023/7/28 12:19
 */
public class OsApplication {
    private static final HashSet<Class<?>> CLASS_SET = new HashSet<>(4);
    private static String BASE_PATH;
    public static final String FILE_SEPARATOR = "/";
    private static final String PREFIX_FILE = "file:";
    private static final String PREFIX_JAR = "jar:file:";
    private static final String SUFFIX_CLASS = ".class";

    /**
     * 加载配置目录及其子目录下的添加了StrategyAnnotation注解的类
     *
     * @param application 启动类
     */
    public static void runWithRecursiveDirectory(Class<?> application) {
        BASE_PATH = Objects.requireNonNull(application.getResource("")).toString().replace(PREFIX_FILE, "");
        String packageName = application.getPackage().getName();
        BASE_PATH = BASE_PATH.substring(0, BASE_PATH.lastIndexOf("/" + packageName + "/")) + FILE_SEPARATOR;
        initRecursiveDirectory(packageName);
        load();
    }

    private static void initRecursiveDirectory(String packageName) {
        File[] files = new File(BASE_PATH + packageName.replace(".", FILE_SEPARATOR)).listFiles(file -> {
            if (file.isDirectory()) {
                initRecursiveDirectory(packageName + "." + file.getName());
            }
            return file.getName().endsWith(SUFFIX_CLASS);
        });
        loadFileClasses(packageName, files);
    }

    private static void loadFileClasses(String packageName, File[] files) {
        if (files != null) {
            for (File file : files) {
                try {
                    Class<?> instance = Class.forName(packageName + "." + file.getName().replace(SUFFIX_CLASS, ""));
                    CLASS_SET.add(instance);
                } catch (ClassNotFoundException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    private static void load() {
        for (Class<?> instance : CLASS_SET) {
            if (instance.getSuperclass() == BaseStrategy.class) {
                try {
                    StrategyAnnotation nameAnnotation = instance.getAnnotation(StrategyAnnotation.class);
                    StrategyDocAnnotation docAnnotation = instance.getAnnotation(StrategyDocAnnotation.class);
                    if (nameAnnotation != null && docAnnotation != null) {
                        Constructor<?> constructor = instance.getConstructor(String.class, String.class);
                        Object newInstance = constructor.newInstance(nameAnnotation.commandName().split(" ")[0], docAnnotation.commandDoc());
                        Method register = newInstance.getClass().getSuperclass().getDeclaredMethod("register");
                        register.setAccessible(true);
                        register.invoke(newInstance);
                    }
                } catch (NoSuchMethodException | InvocationTargetException |
                         IllegalAccessException | InstantiationException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }

    public static void runWithRecursiveJar(Class<?> application) {
        BASE_PATH = Objects.requireNonNull(application.getResource("")).toString().replace(PREFIX_JAR, "");
        BASE_PATH = BASE_PATH.substring(0, BASE_PATH.lastIndexOf("!/" + application.getPackage().getName() + "/")) + FILE_SEPARATOR;
        initRecursiveJar();
        load();
    }

    private static void initRecursiveJar() {
        try (JarFile jarFile = new JarFile(BASE_PATH)) {
            Enumeration<JarEntry> jarEntryEnumeration = jarFile.entries();
            while (jarEntryEnumeration.hasMoreElements()) {
                JarEntry jarEntry = jarEntryEnumeration.nextElement();
                if (jarEntry.getName().endsWith(SUFFIX_CLASS)) {
                    Class<?> instance = Class.forName(jarEntry.getName().replace(SUFFIX_CLASS, "").replace("/", "."));
                    CLASS_SET.add(instance);
                }
            }
        } catch (IOException | ClassNotFoundException e) {
            throw new RuntimeException(e);
        }
    }

}

运行效果
在这里插入图片描述

项目下载

Gitee链接–Java模拟操作系统

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/812535.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何理解Diffusion

Diffusion算法可以有多个角度进行理解&#xff0c;不同的理解方式只是对目标函数进行了不同的解释。其主体思想是不变的&#xff0c;可以归纳为&#xff1a; 训练时通过图片逐步添加噪声&#xff0c;变为一个纯噪声。然后学习每一步的噪声。推理时给定一个随机噪声图片&#x…

Leetcode | Binary search | 22. 74. 162. 33. 34. 153.

22. Generate Parentheses 要意识到只要还有左括号&#xff0c;就可以放到path里。只要右括号数量小于左括号&#xff0c;也可以放进去。就是valid的组合。recurse两次 74. Search a 2D Matrix 看成sorted list就好。直接用m*n表示最后一位的index&#xff0c;并且每次只需要 …

Android启动速度优化

本节主要内容&#xff1a;了解APP启动流程、启动状态、查看启动时间、CPU Profile定位启动耗时代码、StrictMode严苛模式检测不合理写法、解决启动黑白屏问题。 一、APP启动流程 ①用户点击桌面App图标&#xff0c;Launcher进程采用Binder IPC向system_server进程发起startAc…

Spring Boot集成Swagger3.0,Knife4j导出文档

文章目录 Spring Boot集成Swagger3.0,Knife4j导出文档效果展示如何使用简要说明添加依赖添加配置类测试接口token配置位置 官网 说明情况 demo Spring Boot集成Swagger3.0,Knife4j导出文档 效果展示 如何使用 简要说明 Knife4j的前身是swagger-bootstrap-ui,前身swagger-boo…

day48-Random Image Feed(随机图片显示)

50 天学习 50 个项目 - HTMLCSS and JavaScript day48-Random Image Feed&#xff08;随机图片显示&#xff09; 效果 index.html <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport&…

msvcp120.dll丢失的解决方法?有什么解决方法比较推荐?

第一&#xff1a;介绍几种可能导致msvcp120.dll文件丢失或损坏的原因 损坏的程序安装&#xff1a;在安装某个程序时&#xff0c;可能会出现意外中断或其他错误&#xff0c;导致msvcp120.dll文件未能正确地被安装或被破坏。这可能是由于软件安装过程中的错误、病毒感染或硬件问题…

OSPF协议RIP协议+OSPF实验(eNSP)

本篇博客主要讲解单区域的ospf&#xff0c;多区域的仅作了解。 目录 一、OSPF路由协议概述 1.内部网关协议和外部网关协议 二、OSPF的应用环境 1.从以下几方面考虑OSPF的使用 2.OSPF的特点 三、OSPF重要基本概念 3.1&#xff0c;辨析邻居和邻接关系以及七种邻居状态 3…

BPMNJS插件使用及汉化(Activiti绘制流程图插件)

BPMNJS插件运行最重要的就是需要安装nodejs插件,这不一定要安装和测试好。 主要是使用npm命令 1、配置BPMNJS插件绘制activiti7工作流 1.1、安装和配置nodejs 插件 1.1.1、下载nodejs 下载地址:https://nodejs.org/en 1.1.2、安装nodejs,傻瓜式安装 安装之后在安装…

【etcd】docker 启动单点 etcd

etcd: v3.5.9 etcd-browser: rustyx/etcdv3-browser:latest 本文档主要描述用 docker 部署单点的 etcd&#xff0c; 用 etcd-browser 来查看注册到 etcd 的 key 默认配置启动 docker run -d --name ai-etcd --networkhost --restart always \-v $PWD/etcd.conf.yml:/opt/bitn…

一文搞懂mysql(安装、基础命令、存储文件)

1、安装 除此之外&#xff0c;windows在安装前需要额外补加两个东西 dxwebsetup.exe、 vcredist_x64.exe 这俩随便一搜就能找到 在安装前者时要注意取消勾选bing工具栏 mysql下载链接 2、初始化 管理员身份打开cmd >> "path_to_mysql/bin/msqld.exe" --initi…

Windows11+Opencv+Clion编译源码

Windows11OpencvClion编译源码 参考&#xff1a;https://www.robotsfan.com/posts/69395e08.html 注意事项 编译过程中使用的软件&#xff0c;开源码等所有工具的安装路径一定不要有中文和空格。cmake过程会下载一些文件&#xff0c;如果是局域网的话可能下载不下来&#xf…

力扣 70. 爬楼梯

题目来源&#xff1a;https://leetcode.cn/problems/climbing-stairs/description/ C题解&#xff08;来源代码随想录&#xff09;&#xff1a; 本质上是一道斐波那契数题。 动规五部曲&#xff1a;定义一个一维数组来记录不同楼层的状态 确定dp数组以及下标的含义。dp[i]&am…

【Java】JDBC快速入门及详解(MySQL版)

文章目录 1. JDBC概述2. JDBC快速入门2.1 下载驱动jar包2.2 数据准备2.3 创建工程2.4 编写代码 3. JDBC API详解3.1 DriverManager3.2 Connection3.2.1 获取执行SQL对象3.2.1 管理事务 3.3 Statement3.3.1 执行DML语句3.3.2 执行DDL语句 3.4 ResultSet3.4.1 ResultSet对象方法3…

统信UOS V20桌面专业版(1060)

统信UOS V20桌面专业版&#xff08;1060&#xff09;也是发布许久我来发一下体验感受吧。 1.如果你是首次接触统信UOS&#xff0c;1060版本提供了快速上手指南&#xff0c;可以帮助你快速了解统信UOS的基础功能和办公实用小技巧&#xff0c;让你快速上手&#xff0c;降低使用门…

Dockerfile简介

1、什么是Dockerfile 要获得镜像&#xff0c;可以从Docker仓库中进行下载。那如果我们想自己开发一个镜像&#xff0c;那该如何做呢&#xff1f;答案是&#xff1a;Dockerfile Dockerfile其实就是一个文本文件&#xff0c;由一系列命令和参数构成&#xff0c;Docker可以读取Do…

Pytest自动化测试 - 必知必会的一些插件

这可能是B站最详细的pytest自动化测试框架教程&#xff0c;整整100小时&#xff0c;全程实战&#xff01;&#xff01;&#xff01; ​ Pytest拥有丰富的插件架构&#xff0c;超过800个以上的外部插件和活跃的社区&#xff0c;在PyPI项目中以“ pytest- *”为标识。 本篇将列举…

python——案例二 求两个数的和

#案例二 求两个数的和 num1input(请输入第一个数字&#xff1a;) num2input(请输入第二个数字&#xff1a;) sumfloat(num1)float(num2) #计算公式 print(sum) #显示结果 输入num11、num22得到结果sum3

从Vue2到Vue3【七】——Vue2中响应式原理的实现及其缺陷

系列文章目录 内容链接从Vue2到Vue3【零】Vue3简介从Vue2到Vue3【一】Composition API&#xff08;第一章&#xff09;从Vue2到Vue3【二】Composition API&#xff08;第二章&#xff09;从Vue2到Vue3【三】Composition API&#xff08;第三章&#xff09;从Vue2到Vue3【四】C…

JVM入门篇-JVM的概念与学习路线

JVM入门篇-JVM的概念与学习路线 什么是 JVM 定义 Java Virtual Machine - java 程序的运行环境&#xff08;java 二进制字节码的运行环境&#xff09; 好处 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收功能数组下标越界检查多态 比较 jvm jre jdk 常…

[golang gin框架] 44.Gin商城项目-微服务实战之后台Rbac微服务之权限的增删改查微服务

上一节讲解了[golang gin框架] 43.Gin商城项目-微服务实战之后台Rbac微服务之管理员的增删改查以及管理员和角色关联,这里讲解权限管理Rbac微服务权限的增删改查微服务 一.实现后台权限管理Rbac之权限增删改查微服务服务端功能 1.创建Access模型 要实现权限的增删改查,就需要…