Java中将标准输入和输出重定向到字节流

本文介绍了如何在Java单元测试中重定向标准输入和输出,以避免文件读取开销。通过创建getOutput工具函数,实现了以字符串形式传递输入和获取输出,但在多线程环境下可能引发线程安全问题。为了解决这个问题,建议将输入输出依赖改为使用InputStream和OutputStream,修改Executable接口和getOutput方法,确保测试的线程安全性。
摘要由CSDN通过智能技术生成

在Java程序中,可以通过给System.inSystem.out重新赋值来重定向标准输入和输出。

在单元测试的时候,为了对涉及到标准输入和输出的程序进行测试,常用的做法是将标准输入和输出重定向到文件,这样就可以用程序实现自动化的比对。但是这样做会有文件读取的开销,可能会降低单元测试的运行效率。

所以,更优的做法是将标准输入和输出重定向到字节流,这样就能避免文件读取的开销。

为了实现以上的效果,可以封装一个叫做getOutput的工具函数,它以字符串的形式传递标准输入内容,并以字符串的形式返回标准输出的内容。在getOutput内部通过对System.inSystem.out重新赋值来实现重定向的效果,并在测试完成之后恢复原来的值。

public interface Executable {
    void execute();
}

public String getOutput(String input, Executable executable) {
    try (
            // 字节输入流
            ByteArrayInputStream is = new ByteArrayInputStream(input.getBytes(StandardCharsets.UTF_8));
            // 字节输出流
            ByteArrayOutputStream os = new ByteArrayOutputStream()
    ) {
        // 保存原始的标准输入和输出
        InputStream in = System.in;
        PrintStream out = System.out;

        // 将标准输入和输出重定向到字节流
        System.setIn(is);
        System.setOut(new PrintStream(os));

        // 执行待验证的程序
        executable.execute();

        // 恢复标准输入和输出
        System.setOut(out);
        System.setIn(in);

        // 返回标准输出的内容
        return os.toString();
    } catch (Throwable e) {
        throw new RuntimeException(e);
    }
}

可以写以下的测试程序:

public void funcToTest() {
    Scanner scanner = new Scanner(System.in);
    while (scanner.hasNext()) {
        int a = scanner.nextInt();
        int b = scanner.nextInt();
        System.out.println(a + b);
    }
}

String output = getOutput("2 3 10 20 55 100", () -> funcToTest());
System.out.println(output);

控制台输出如下:

5
30
155

然而,由于System.inSystem.out本质上是全局变量,所以以上程序会有线程安全问题,当有多个线程同时调用getOutput方法时,输出内容会错乱。

更优的做法是让被测函数不直接依赖于System.inSystem.out,而是依赖于更抽象的InputStreamOutputStream

public static void funcToTest(InputStream is, OutputStream os) {
    Scanner scanner = new Scanner(is);
    PrintStream printStream = new PrintStream(os);
    while (scanner.hasNext()) {
        int a = scanner.nextInt();
        int b = scanner.nextInt();
        printStream.println(a + b);
    }
}

然后改造Executable接口和getOutput方法:

public interface Executable {
    void execute(InputStream is, OutputStream os);
}

public String getOutput(String input, Executable executable) {
    try (
            // 字节输入流
            ByteArrayInputStream is = new ByteArrayInputStream(input.getBytes(StandardCharsets.UTF_8));
            // 字节输出流
            ByteArrayOutputStream os = new ByteArrayOutputStream()
    ) {
        // 执行待验证的程序
        executable.execute(is, os);

        // 返回标准输出的内容
        return os.toString();
    } catch (Throwable e) {
        throw new RuntimeException(e);
    }
}

测试代码:

String output = getOutput("2 3 10 20 55 100", (is, os) -> funcToTest(is, os));
System.out.println(output);

运行结果与之前相同:

5
30
155
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

byx2000

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值