- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
use Parallel::ForkManager;
my $number_running = 0;
my $pm = new Parallel::ForkManager(30);
$pm->run_on_start( sub { ++$number_running; } );
$pm->run_on_finish( sub { --$number_running; } );
for (my $i=0; $i<=100; $i++)
{
if ($number_running == 5) { while ($number_running > 0) {} } # waits forever
$pm->start and next;
print $i;
$pm->finish;
}
$number_running
相应地变化。一旦 5 个子进程正在运行,我希望它等到 0 个子进程正在运行后再继续。
wait_all_children
但我不想使用它。
最佳答案
短 回调 run_on_finish
通常不会触发每个 child 的退出,所以 $number_running
不会减少,因此它无法控制循环。解决此问题的方法:
reap_finished_children
为了在个别 child 退出时进行交流,以便 run_on_finish
当每个 child 退出时,确实可以运行wait_for_available_procs
等待整个批处理完成后再开始一个新批处理run_on_start
与每个新进程一起运行并且计数器递增。但是回调
run_on_finish
永远不会触发,因此计数器永远不会递减。因此一旦达到
5
代码位于
while
环形。请注意, parent 和 child 是独立的进程,因此不知道彼此的变量并且无法更改它们。
run_on_finish
通常由
wait_all_children
触发在所有进程被 fork 之后。它的工作也完成了
start
中完成的调用
wait_one_child
(调用
on_finish
,见下文)。
This is a non-blocking call to reap children and execute callbacks independent of calls to
start
orwait_all_children
. Use this in scenarios wherestart
is called infrequently but you would like the callbacks executed quickly.
wait_all_children
.
use warnings;
use strict;
use feature 'say';
use Parallel::ForkManager;
$| = 1;
my $total_to_process = 3; # only a few for this test
my $number_running = 0;
my @ds;
my $pm = Parallel::ForkManager->new(30);
$pm->run_on_start( sub {
++$number_running;
say "Started $_[0], total: $number_running";
});
$pm->run_on_finish( sub {
--$number_running;
my ($pid, $code, $iden, $sig, $dump, $rdata) = @_;
push @ds, "gone-$pid";
say "Cleared $pid, ", ($rdata->[0] // ''), ($code ? " exit $code" : '');
});
foreach my $i (1 .. $total_to_process)
{
$pm->start and next;
run_job($i);
$pm->finish(10*$i, [ "kid #$i" ]);
}
say "Running: ", map { "$_ " } $pm->running_procs; # pid's of children
# Reap right as each process exits, retrieve and print info
my $curr = $pm->running_procs;
while ($pm->running_procs)
{
$pm->reap_finished_children; # may be fewer now
if ($pm->running_procs < $curr) {
$curr = $pm->running_procs;
say "Remains: $number_running. Data: @ds";
}
sleep 1; # or use Time::HiRes::sleep 0.1;
}
sub run_job {
my ($num) = @_;
my $sleep_time = ($num == 1) ? 1 : ($num == 2 ? 10 : 20);
sleep $sleep_time;
say "\tKid #$num slept for $sleep_time, exiting";
}
使用该方法相当于调用
waitpid -1, POSIX::WNOHANG
在
fork
之后的循环中.这比最大 (
30
) 进程 fork 更少,以便更容易地查看输出并证明回调在子退出时正确运行。更改这些数字以查看其完整操作。
10*$i
退出,以便能够跟踪输出中的子进程。匿名数组中返回的数据
[...]
是标识子进程的字符串。尽快
reap_finished_children
通话完成
$number_running
在回调中减少。这就是拥有
$curr
的原因。变量,再次用于诊断。
start: Started 4656, running: 1start: Started 4657, running: 2start: Started 4658, running: 3Running: 4656 4658 4657 Kid #1 slept for 1, exitingCleared 4656, kid #1 exit 10Remains: 2. Data: gone-4656 Kid #2 slept for 10, exitingCleared 4657, kid #2 exit 20Remains: 1. Data: gone-4656 gone-4657 Kid #3 slept for 20, exitingCleared 4658, kid #3 exit 30Remains: 0. Data: gone-4656 gone-4657 gone-4658
The direct question is of how to wait for the whole batch to finish before starting a new one. This can be done directly by wait_for_available_procs($n)
Wait until
$n
available process slots are available. If$n
is not given, defaults to 1.
If $MAX
is used for $n
, that many slots will become available only once the whole batch completed. What to use for $n
can also be decided at runtime.
Some details of module's operation
When a child exits the SIGCHLD
signal is sent to the parent, which it must catch in order to know that the child is gone (and to avoid zombies, in the first place). This is done by using wait
or waitpid
, in code or in the SIGCHLD
handler (but only at one place). See fork, Signals in perlipc, waitpid and wait.
We see from P::FM's source that this is done in wait_one_child
(via _waitpid
sub)
sub wait_one_child { my ($s,$par)=@_;
my $kid;
while (1) {
$kid = $s->_waitpid(-1,$par||=0);
last if $kid == 0 || $kid == -1; # AS 5.6/Win32 returns negative PIDs
redo if !exists $s->{processes}->{$kid};
my $id = delete $s->{processes}->{$kid};
$s->on_finish( $kid, $? >> 8 , $id, $? & 0x7f, $? & 0x80 ? 1 : 0);
last;
}
$kid;
};
用于
wait_all_children
sub wait_all_children { my ($s)=@_;
while (keys %{ $s->{processes} }) {
$s->on_wait;
$s->wait_one_child(defined $s->{on_wait_period} ? &WNOHANG : undef);
};
}
方法
reap_finished_children
上面使用的是此方法的同义词。
wait_one_child
start
使用获取信号的当最大进程数被填满并且一个退出时,获取子进程。这就是模块知道何时可以启动另一个进程并尊重其最大值的方式。 (它也被其他一些等待进程的例程使用。
run_on_finish
被
$s->on_finish( $kid, ... )
触发
sub on_finish {
my ($s,$pid,@par)=@_;
my $code=$s->{on_finish}->{$pid} || $s->{on_finish}->{0} or return 0;
$code->($pid,@par);
};
回调在 coderef
$code
中, 从对象的
on_finish
中检索 key ,它本身在子
run_on_finish
中设置.一旦该子运行,这就是设置回调的方式。
wait_all_children
和
reap_finished_children
.
$number_running
没有更新所以
while
是一个无限循环。回想一下变量
$number_running
在父进程中不能被子进程直接更改。
关于perl - 如何等待子进程在父进程中设置变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39071281/
我有这个 html 代码: HELLO WORLD! X V HELLO WORLD! X V 我想按 X(类关闭)将父 div 的高度更改为 20px 并显示 V(类打开),但在每个 d
在会计应用程序的许多不同实现中,有两种主要的数据库设计方法来保存日志和分类帐数据。 只保留 Journal 信息,然后 Ledger 只是 Journal 的一个 View (因为 journal 总
我想在另一个子里面有一个子, sub a { sub b { } } 我想为每次调用 sub b 创建一个新的 sub a 实例。有没有办法在 Perl 中做到这一点? 当我运行上面的
我有一些代码正在查找重复项并突出显示单元格: Private Sub cmdDups_Click() Dim Rng As Range Dim cel As Range Set Rng = ThisW
可能有一个简单的解决方案,但我很难过。 我有一个包含一个 ID 字段的主表。在两个可能的字段中有一个具有该 ID 的子表。想象一个由选手 A 和选手 B 组成的 double 队。Master 表将有
假设我有一个包含对象的数组: [ { "id": "5a97e047f826a0111b754beb", "name": "Hogwarts", "parentId": "
我正在尝试对 MySQL 数据库表执行一对父/子模型的批量插入,但似乎无法使用标准的 ActiveRecord 功能来完成。所以,我尝试了 activerecord-import gem,但它也不支持
我有一个带有多个子类的父抽象类。最终,我希望通过 GUI 中的进度条显示子类中完成的进度。 我目前所做的,我意识到这是行不通的,是在父类中声明为每个子类将覆盖的虚拟方法的事件方法定义。所以像: pub
是否可以通过键数组在对象中设置变量?例如我有这个对象: var obj = {'outer': {'inner': 'value'} }; 并希望设置由键数组选择的值: var keys = ['ou
我有一个名为 companies 的 MySQL 表,如下所示: +---------+-----------+-----------+ | id_comp | comp_name | id_pare
我正在尝试使用 sublime text 在 sublime text 上的 ionic 上打开我的第一个应用程序。它给了我一个“找不到命令”的错误。如何修复? 我试过这些命令: sudo rm -r
不好意思问,但我正在使用 webapp2,我正在设计一个解决方案,以便更容易定义路由 based on this google webapp2 route function .但这完全取决于能够在子级
我有代表树的数字字符串(我不知道是否有官方名称): 012323301212 上面的例子代表了 2 棵树。根用 0 表示。根的直接子代为“1”,“1”的直接子代为“2”,依此类推。我需要将它们分组到由
是否可以在当前 Activity 之上添加 Activity 。例如,假设我单击一个按钮,然后它将第二个 Activity 添加到当前 Activity 。而第二个 Activity 只覆盖了我当前
我很难思考如何为子资源建模。 以作者的书籍为例。你可以有 N 本书,每本书只有一位作者。 /books GET /books POST /books/id PUT /books/id DELETE 到
有人可以向我解释以下内容(python 2.7) 来自已解析文件的两个字符串数字: '410.9''410.9 '(注意尾随空格) A_LIST = ['410.9 '] '410.9' in '41
背景 在 PowerShell 中构建 hash table 是很常见的通过特定属性快速访问对象,例如以 LastName 为基础建立索引: $List = ConvertFrom-Csv @' I
我真的很难弄清楚如何调用嵌套 Polymer Web 组件的函数。 这是标记: rise-distribution组件有 canPlay我想从 rise-playlist
我写了一个小工具转储(以 dot 格式)一个项目的依赖关系图,其中所有位于同一目录中的文件都聚集在一个集群中。当我尝试生成包含相应图形的 pdf 时,dot开始哭: 命令 dot -Tpdf trim
给定一个 CODE ref,是否可以: 访问该 CODE ref 的解析树 通过指定 CODE ref 的解析树来创建一个新的 CODE ref,该解析树可以包含在 1 中返回的解析树的元素 通常我们
我是一名优秀的程序员,十分优秀!