当前位置: 首页 > news >正文

javafx程序启动cmd.exe调用其他软件或者脚本指令

举个例子

ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/c", command);

pb.directory(new File( ConfigContextUtils.pdiConfig.getPdiHome()));

pb.start();

解释下这三句

1.ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/c", command);

作用:创建进程构建器,准备执行外部命令

// 分解参数: "cmd.exe" // 调用 Windows 命令行解释器"/c" // 参数:执行完命令后关闭 cmd 窗口command // 要执行的具体命令字符串 // 示例:如果 command = "dir C:\\" // 实际执行:cmd.exe /c "dir C:\\"

详细说明:

  • ProcessBuilder是 Java 中用于创建操作系统进程的类

  • cmd.exe是 Windows 的命令行程序

  • /c参数告诉 cmd:执行后面的命令,然后退出

  • 如果不加/c,cmd 会保持打开状态等待用户输入

等价于手动操作:

cmd

开始菜单 → 运行 → cmd.exe → 输入命令 → 回车
2.pb.directory(new File(ConfigContextUtils.pdiConfig.getPdiHome()));

作用:设置进程的工作目录

// 分解: ConfigContextUtils.pdiConfig.getPdiHome() // 从配置获取 PDI 安装路径 new File(...) // 创建 File 对象表示目录pb.directory(...) // 设置进程的工作目录

为什么重要:
假设getPdiHome()返回"D:\dev_install\pdi-ce-9.4.0.0-343\data-integration"

// 设置前:默认使用 Java 程序的当前目录 // 设置后:进程的工作目录变为 PDI 的安装目录必要性:
  • carte.batspoon.bat等批处理文件通常需要在其所在目录执行

  • 这些批处理文件内部可能使用了相对路径

  • 设置正确的工作目录可以避免 "找不到文件" 的错误

示例:

// 假设命令是 "carte.bat 127.0.0.1 8819" // 不设置 directory:在 Java 程序目录找 carte.bat(通常找不到) // 设置 directory:在 PDI 安装目录找 carte.bat(能找到)

3.pb.start();

作用:启动进程并异步执行

// start() 方法: // 1. 创建新的操作系统进程 // 2. 执行指定的命令 // 3. 立即返回,不等待进程结束 // 4. 返回 Process 对象用于控制进程

关键特性:

  • 异步执行:不会阻塞当前 Java 线程

  • 返回 Process 对象:可以控制进程(如等待结束、获取输出、终止进程等)

  • 独立进程:在操作系统中独立运行

完整示例与解释

原始代码:

ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/c", command); pb.directory(new File(ConfigContextUtils.pdiConfig.getPdiHome())); pb.start();

假设配置:

ConfigContextUtils.pdiConfig.getPdiHome() = "D:\\pdi\\data-integration" command = "carte.bat 127.0.0.1 8819"

实际执行效果:

// 1. 创建 ProcessBuilder // 相当于告诉系统:"我要执行 cmd.exe,参数是 /c 和 carte.bat ..." // 2. 设置工作目录为 D:\pdi\data-integration // 相当于先执行 "cd /d D:\pdi\data-integration" // 3. 启动进程 // 系统实际执行: // cmd.exe /c "cd /d D:\pdi\data-integration && carte.bat 127.0.0.1 8819"

重要补充:需要处理的几个问题

问题1:需要等待进程执行并获取输出

当前代码的问题:

pb.start(); // 只是启动,不等待完成,也看不到输出

改进版本:

Process process = pb.start(); // 获取输出流(进程的标准输出) try (BufferedReader reader = new BufferedReader( new InputStreamReader(process.getInputStream(), "GBK"))) { String line; while ((line = reader.readLine()) != null) { System.out.println("输出: " + line); // 显示命令输出 } } // 等待进程结束 int exitCode = process.waitFor(); System.out.println("进程结束,退出码: " + exitCode);

问题2:需要处理错误流

防止进程卡住:

Process process = pb.start(); // 必须读取错误流,否则缓冲区满时进程会挂起 new Thread(() -> { try (BufferedReader errorReader = new BufferedReader( new InputStreamReader(process.getErrorStream(), "GBK"))) { String line; while ((line = errorReader.readLine()) != null) { System.err.println("错误: " + line); } } catch (IOException e) { e.printStackTrace(); } }).start();

问题3:处理带空格的路径

如果路径包含空格:

// 错误:路径有空格会被拆分成多个参数 String command = "carte.bat \"C:\\Program Files\\pdi\" args"; // 正确:需要将整个命令作为单个字符串传递 ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/c", command);

完整的安全版本

public static void startPDIService(String serviceType) { String pdiHome = ConfigContextUtils.pdiConfig.getPdiHome(); String host = ConfigContextUtils.pdiConfig.getCarteHost(); String port = ConfigContextUtils.pdiConfig.getCartePort(); // 构建命令 String command; if ("carte".equals(serviceType)) { command = String.format("chcp 65001 && set JAVA_TOOL_OPTIONS=-Dfile.encoding=UTF-8 && carte.bat %s %s", host, port); } else { command = "chcp 65001 && set JAVA_TOOL_OPTIONS=-Dfile.encoding=UTF-8 && spoon.bat"; } try { ProcessBuilder pb = new ProcessBuilder("cmd.exe", "/c", command); pb.directory(new File(pdiHome)); // 设置工作目录 // 启动进程 Process process = pb.start(); // 启动线程读取输出(防止阻塞) startOutputReader(process); // 启动线程读取错误(防止阻塞) startErrorReader(process); // 可以记录进程ID供后续管理 System.out.println(serviceType + " 进程已启动,PID可通过其他方式获取"); } catch (IOException e) { e.printStackTrace(); } } private static void startOutputReader(Process process) { new Thread(() -> { try (BufferedReader reader = new BufferedReader( new InputStreamReader(process.getInputStream(), "GBK"))) { String line; while ((line = reader.readLine()) != null) { System.out.println("输出: " + line); } } catch (IOException e) { // 进程可能已结束,正常关闭流 } }).start(); } private static void startErrorReader(Process process) { new Thread(() -> { try (BufferedReader reader = new BufferedReader( new InputStreamReader(process.getErrorStream(), "GBK"))) { String line; while ((line = reader.readLine()) != null) { System.err.println("错误: " + line); } } catch (IOException e) { // 进程可能已结束,正常关闭流 } }).start(); }

总结

你的三行代码:

  1. 创建命令:告诉系统执行什么

  2. 设置目录:告诉系统在哪里执行

  3. 启动进程:开始执行命令

这是一个标准的、正确的启动外部进程的方式,特别适合启动 PDI 这种需要特定工作目录的程序。

http://www.jsqmd.com/news/290257/

相关文章:

  • 如何解决 Spoon 与 Carte 互联的 UTF-8 编码问题
  • spoon如何连接carte如何将.ktr任务或者.kjb任务提交个远程carte服务让,carte的服务端来执行 etl脚本的任务呢?
  • Pentaho Data Integration:Spoon与Carte的对比
  • mast3r-smf测试 - MKT
  • 备份 还原 系统 数据保护
  • 中小厂 AI 招聘隐藏要求:除了技术,这 2 点更关键
  • nodejs基于vue的教学质量评价管理系统的设计与实现 评教系统39j06
  • 二叉树递归实现
  • nodejs基于vue的教师科研项目申报信息管理系统的设计与实现_c7z6m
  • nodejs基于vue二手商品物品商城网站_s926p
  • nodejs基于vue基于MVC模式的考研论坛互动交流系统的私信设计与实现
  • nodejs基于vue技术人人美食菜谱分享点餐配送平台的设计与实现
  • 税筹园区助力企业合规减负与税务优化
  • 气体涡轮流量计 本土精造 精准守护气体管控
  • 企业级邮件服务优化实战:从550错误到高可用架构
  • 格恩朗金属管浮子流量计 本土精造 稳控流体计量
  • Excel动态生成SQL更新语句:批量处理数据的高效技巧
  • 救命神器9个AI论文平台,自考学生轻松搞定毕业论文!
  • vLLM 推理 GPU 选型指南:显存、KV Cache 与性能瓶颈全解析
  • 详解redis(7):数据结构List
  • 详解redis(8):数据结构Hash
  • 详解redis(9):数据结构set
  • 一文学习 了解 OSI模型、TCP/IP模型、网络封包
  • 深入解析:Linux动态存储管理的逻辑卷使用示例
  • 北京附近上门回收酒
  • YOLOv8目标检测:从理论到实战的飞跃之旅
  • 用AI制作表格实战:20个高频ChatExcel指令词,告别低效Excel操作
  • 打破 NotebookLM 最后的限制:我写了个开源工具,把 PDF 瞬间变回可编辑 PPT!
  • 力扣122 买卖股票的最佳时机II java实现
  • STM32项目分享:图书馆环境监测系统