Java 程序只运行一个进程--思路

这篇博客探讨了如何确保Java程序仅运行一个进程,提供了4种解决方案:文件锁、Socket绑定端口、JNI调用C程序以及利用Java6以上的`jps`命令。作者选择了方案3和4进行实现,通过`jps`命令验证程序是否已经运行,以避免程序意外退出导致的错误启动。代码示例展示了如何包装`jps`命令以检查指定程序是否运行,从而决定是否启动新进程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

写了一个SWT程序 Windows 下运行,由于启动时需要检测 改程序是否已经启动,如已经启动给予提示不再另打开程序。

经搜索:

综合网上各种方案 最终总结4方案:

1.lock File 

2.Socket 绑定端口

3.通过JNI 调用 C 程序进行判断。

4.Java6 以上 可以支持 命令行“jps” 命令 可以显示出各个java运行程序的进程ID 和主程序名称或者程序打包的jar报名。


方案特点:

1.方案1简单。

每当程序启动锁定文件,对文件进行标记。再次启动程序读取该文件 发现已被标记,进而不在启动新程序。当程序退出时进行标记清除。

 存在致命缺陷:    当程序运行中发生意外退出。没有对文件进行清除标记。将导致程序判断错误。

2.方案2 比方案1 :  可靠,但是这种方案显得 有些 奇怪变通绕行太远,导致其他的问题 局限,

    比如 使用的端口被其他程序占用时 或 当有防火墙组织使用端口时 你讲无法绑定端口,程序无法启动。


3.由于 C \C++ 程序可以很好的进行此 判断,因此考虑用C实现 判断方法,让java进行调用。

该方法优点:针对目标直接针对问题解决。 没有绕行,避免绕行带来的其他问题。

该方法缺点:复杂。需要C、C++ 编程经验、

4.方案4 简单 直接。


准备实现: 方案 3 和方案4 

实现后我再追加详细代码

【方案4】实现

1.首先找到你要监控成程序名称 :通过 jps 命令

2.验证程序是否已经存在: 通过 包装  Runtime.getRuntime().exec(command); 进行验证

3.程序启动时进行 验证:


事例:中我的程序名称是:AppsHolder.jar

1:打包并运行程序 使程序处于运行状态 使得 jps 命令能捕捉到该程序: 

2: 在CMD 中运行 jps命令:


  现在:我们从命令行中得到 java 程序名称和进程ID 我们取得 java程序名称AppsHolder.jar

3: 接下来要做的就是通过 java 方法包装 jps 命令 --运行 -- 返回结果 -- 验证是否存在 指定的程序名称 返回 true 或 false:

java Commander 类代码:


/**
     * 包装 jps命令
     * @return
     * @throws IOException
     */
    public static String jpsCommand() throws IOException {
        return executeCommand("jps").trim().toLowerCase();
    }

/**
     * 包装 Runtime.getRuntime().exec 命令
     * @param command 要执行的本地命令
     * @return res 返回执行结果
     * @throws IOException 运行出现异常
     */
    public static String executeCommand(String command) throws IOException {
        Process process = Runtime.getRuntime().exec(command);
        InputStream inputStream = process.getInputStream();

        BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
        String line = null;

        StringBuilder tasklist = new StringBuilder();
        while ((line = reader.readLine()) != null) {
            tasklist.append(line + "\n");
        }
        reader.close();
        inputStream.close();
        String rs = tasklist.toString();
        return rs;
    }


 /**
     * 分析 jps 中是符合 传入名称的程序
     * @param jar 验证过滤的 java程序名称
     * @return List<String> 返回符合的程序名称列表
     */
    public static List<String> javaTaskApps(String jar) {
        Set set = new HashSet();
        String javaTaskApps = null;
        try {
            javaTaskApps = jpsCommand();
        } catch (IOException e) {
            return null;
        }
        String reg = ".*"+jar.trim().toLowerCase();
        Pattern pattern = Pattern.compile(reg);
        Matcher matcher = pattern.matcher(javaTaskApps);
        while (matcher.find()) {
            set.add(matcher.group());
        }
        List<String> exeList = new Vector<String>(set);
        return exeList;
    }

 /**
     * 通过本地jps命令验证是否已经有指定的java 程序已经云溪
     * @param appName 需要验证的 java 程序名称:该名称是 jps 中能显示的java程序名称
     * @return 是否 已经运行
     */

    public static boolean isApplicationRunning(String appName) {
        List<String> list = javaTaskApps(appName);
        if (list != null && list.size() > 1) {
            return true;
        } else {
            return false;
        }
    }

至此:我们应经拥有了 根据 传入java程序名称 验证是否 存在应运行有该名称的程序 方法 isApplicationRunning (String appName)

注意: 1. javaTaskApps 方法传入的程序名称和执行执行的命令结果返回,再进行验证前都进行了trim() 和 toLowerCase(); 是为了 忽略大小写和空格

  2.isApplicationRunning 方法中进行验证 符合的 程序列表 list.size > 1 而不是大于 0:是因为 当你进行验证的时候 该程序就应经存在在 jps 结果集中,

当大于 1 时才能证明,已经有一个给相同名称的进程已经运行。


最后把该方法添加到程序启动初始位置:

如果验证: 已经有程序启动 将给与提示,并不在打开新 进程。

如果验证: 没有程序已启动 讲启动你的程序:

java 代码:

 public static void main(String[] args) {
        boolean running = Commander.isApplicationRunning("AppsHolder.jar");
	// 验证已经启动给予提示 并不在启动新进程
        if (running) {
            MessageBox box = new MessageBox(new Shell(),SWT.ICON_WARNING);
            box.setText("警告消息");
            box.setMessage("有一个程序已经运行!\n请查看状态栏右下角是否有已经打开的程序.");
            box.open();
        } else {
	//验证失败 说明没有启动过程序 进行程序启动吧 兄弟...
            ApplicationRunnerUI app = new ApplicationRunnerUI();
            app.open();//.... 启动程序
        }
    }

至此 方案4 已完成:

优点: 1.技术简单

   2. 不存在方案1 和方案2 迂回变通带来的严重负面 

缺点: 1.需要java6以上 才能支持 jps命令

  2.想要跨平台 需要实现 另外平台版本的验证方法。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值