使用swing+调用外部命令的方法写一个网口测试的程序,想要将命令窗口里输出的内容实时显示在UI界面上,而不是执行完了再一次行输出。
在网上查了查可以通过ZT Process Executor包实现该功能。
这是作者给出的demo
Handling output line-by-line while process is running
new ProcessExecutor().command("java", "-version")
.redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
...
}
})
.execute();
我自己通过上述方法实现了调用ping命令,并把输出实时打印到控制台的功能,代码如下:
ProcessExecutor pe = new ProcessExecutor();
String[] Command = {};
Command = new String[]{"ping", “192.168.1.23”, "-c", "10"};
pe.command(Command)
.redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
System.out.println(line);
}
}
})
.execute();
输出结果如下所示:
可见,在调用ping命令时,是可以达到预期结果的。但是现在要调用iperf3来完成吞吐量测试,代码如下:
ProcessExecutor pe = new ProcessExecutor();
String[] Command = {};
Command = new String[]{"iperf3", "-c", “192.168.1.23”};
pe.command(Command)
.redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
System.out.println(line);
}
}
})
.execute();
但是使用相同的方法又不能把输出实时打印到控制台了,需要等进程执行完了再一次性输出。现在就想在调用iperf3时和调用ping指令一样实时输出命令窗口的信息,请问有没有什么好的解决方法?
iperf3命令和ping命令不一样,ping命令在探测过程中一直会有日志输出,所以才符合预期,但iperf3在测试过程中没有任何输出,只有测试完成才会输出报告。
可以通过指定iperf3得参数来实现要求,参数含义如下:
-p, --port #,Server 端监听、Client 端连接的端口号;
-f, --format [kmgKMG],报告中所用的数据单位,Kbits, Mbits, KBytes, Mbytes;
-i, --interval #,每次报告的间隔,单位为秒;
-F, --file name,测试所用文件的文件名。如果使用在 Client 端,发送该文件用作测试;如果使用在 Server 端,则是将数据写入该文件,而不是丢弃;
-A, --affinity n/n,m,设置 CPU 亲和力;
-B, --bind ,绑定指定的网卡接口;
-V, --verbose,运行时输出更多细节;
-J, --json,运行时以 JSON 格式输出结果;
--logfile f,输出到文件;
-d, --debug,以 debug 模式输出结果;
-v, --version,显示版本信息并退出;
-h, --help,显示帮助信息并退出。
Server 端参数:
-s, --server,以 Server 模式运行;
-D, --daemon,在后台以守护进程运行;
-I, --pidfile file,指定 pid 文件;
-1, --one-off,只接受 1 次来自 Client 端的测试,然后退出。
Client 端参数
-c, --client ,以 Client 模式运行,并指定 Server 端的地址;
-u, --udp,以 UDP 协议进行测试;
-b, --bandwidth #[KMG][/#],限制测试带宽。UDP 默认为 1Mbit/秒,TCP 默认无限制;
-t, --time #,以时间为测试结束条件进行测试,默认为 10 秒;
-n, --bytes #[KMG],以数据传输大小为测试结束条件进行测试;
-k, --blockcount #[KMG],以传输数据包数量为测试结束条件进行测试;
-l, --len #[KMG],读写缓冲区的长度,TCP 默认为 128K,UDP 默认为 8K;
--cport ,指定 Client 端运行所使用的 TCP 或 UDP 端口,默认为临时端口;
-P, --parallel #,测试数据流并发数量;
-R, --reverse,反向模式运行(Server 端发送,Client 端接收);
-w, --window #[KMG],设置套接字缓冲区大小,TCP 模式下为窗口大小;
-C, --congestion ,设置 TCP 拥塞控制算法(仅支持 Linux 和 FreeBSD );
-M, --set-mss #,设置 TCP/SCTP 最大分段长度(MSS,MTU 减 40 字节);
-N, --no-delay,设置 TCP/SCTP no delay,屏蔽 Nagle 算法;
-4, --version4,仅使用 IPv4;
-6, --version6,仅使用 IPv6;
-S, --tos N,设置 IP 服务类型(TOS,Type Of Service);
-L, --flowlabel N,设置 IPv6 流标签(仅支持 Linux);
-Z, --zerocopy,使用 “zero copy”(零拷贝)方法发送数据;
-O, --omit N,忽略前 n 秒的测试;
-T, --title str,设置每行测试结果的前缀;
--get-server-output,从 Server 端获取测试结果;
--udp-counters-64bit,在 UDP 测试包中使用 64 位计数器(防止计数器溢出)。
修改后代码如下:
ProcessExecutor pe = new ProcessExecutor();
String[] Command = {};
Command = new String[]{"iperf3", "-c", “192.168.1.23", "-V", "-d", "-i", "1"};
pe.command(Command)
.redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
System.out.println(line);
}
}
})
具体可以根据需求增加或减少参数
一些建议:
建议看看ping和iperf3的源码,从源码角度看是什么问题导致的。
linux下ping命令的实现源码:https://blog.csdn.net/xujuan0815/article/details/38397473
Iperf 源代码分析:https://blog.csdn.net/water_cow/article/details/6907935
确实存在这个问题,下面是复现代码
import org.zeroturnaround.exec.ProcessExecutor;
import org.zeroturnaround.exec.stream.LogOutputStream;
// maven依赖
//<!-- https://mvnrepository.com/artifact/org.zeroturnaround/zt-exec -->
//<dependency>
//<groupId>org.zeroturnaround</groupId>
//<artifactId>zt-exec</artifactId>
//<version>1.12</version>
//</dependency>
public class Test {
public static void exec () throws Exception {
ProcessExecutor pe = new ProcessExecutor();
String[] Command = {};
Command = new String[]{"ping", "192.168.101.203", "-c", "10"};
pe.command(Command).redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
System.out.println(line);
}
}).execute();
}
public static void exec1 () throws Exception {
ProcessExecutor pe = new ProcessExecutor();
String[] Command = {};
// 服务断续开启 iperf3 -s -i 1 -p 1314
Command = new String[]{"iperf3", "-u", "-c", "192.168.61.125", "-i", "1", "-t", "5", "-p", "1314"};
pe.command(Command).redirectOutput(new LogOutputStream() {
@Override
protected void processLine(String line) {
System.out.println(line);
}
}).execute();
}
public static void main(String[] args) throws Exception {
// exec();
exec1();
}
}