我想为 perl 中的子例程编写单元测试。该子例程使用多个线程来完成其任务。因此,它首先创建一些线程,然后等待它们加入。
问题是我们的单元测试在无法运行多线程测试的服务器上运行,所以我需要以某种方式模拟线程行为。基本上我想覆盖线程创建和连接函数,使其不再线程化。任何指针我该如何做到这一点并测试代码?
编辑:服务器无法运行线程代码,原因如下:
Devel::Cover 还不支持线程
我想为 perl 中的子例程编写单元测试。该子例程使用多个线程来完成其任务。因此,它首先创建一些线程,然后等待它们加入。
问题是我们的单元测试在无法运行多线程测试的服务器上运行,所以我需要以某种方式模拟线程行为。基本上我想覆盖线程创建和连接函数,使其不再线程化。任何指针我该如何做到这一点并测试代码?
编辑:服务器无法运行线程代码,原因如下:
Devel::Cover 还不支持线程
更新:此答案不能解决已编辑问题中描述的 OP 问题,但它可能对某人有用。
Perl 线程是解释器模拟,而不是操作系统功能。所以,他们应该在任何平台上工作。如果您的测试服务器不支持线程,则可能是由于以下原因之一:
前两个可以通过更新您的环境轻松纠正。但是,我怀疑您的问题是第三个问题。
我认为你不应该通过模拟线程行为来解决这个问题。这对原始代码进行了太多更改,无法进行有效测试。无论如何,这将是大量的工作,那么为什么不将精力用于使线程测试正常工作呢?
确切的问题取决于您的代码,但问题可能是您的子例程启动一个线程然后返回,而该线程仍在运行。然后你的测试框架一遍又一遍地运行 sub,积累一大堆并发线程。
在这种情况下,您所需要的只是一个包装子程序,它调用您正在测试的子程序,然后阻塞直到线程完成。这应该相当简单。看看threads->list()
如何检测正在运行的线程。在退出包装子之前,只需有一个循环等待相关线程不再运行。
这是一个简单的完整示例,演示了一个包装子:
#!usr/bin/perl
use strict;
use warnings;
use threads;
sub sub_to_test {
threads->create(sub { sleep 5; print("Thread done\n"); threads->detach() });
return "Sub done\n";
}
sub wrapper {
#Get a count of the running threads.
my $original_running_threads = threads->list(threads::running);
my @results = sub_to_test(@_);
#block until the number of running threads is the same as when we started.
sleep 1 while (threads->list(threads::running) > $original_running_threads);
return @results;
}
print wrapper;