我有一个正在进行段错误的脚本,我对解决它的方式感到不舒服,所以我想在这里发布问题,以便更多地了解它的原因,也许是一个更好的解决方案。 / p>
这是我的脚本所做的(删除了一些详细的代码以留下它的“核心”):
# Here's a query I need to do every X seconds to monitor progress of other tasks
# This is apparently the key to my segfault problem
my $stmt = $dbh->prepare($query);
my $all_done = 0;
while(!$all_done) {
$self->debug("Waiting for $n blocker tasks to be finished");
# Execute query to pull the status of the tasks from DB
$stmt->execute();
my $pending = [];
while(my $hr = $stmt->fetchrow_hashref()) {
push @{$pending}, $hr->{'TASK_NAME'} if ($hr->{'STATUS'} ne 'COMPLETE');
}
if(scalar(@{$pending}) > 0) {
$all_done = 0;
sleep($sleep_gap);
}
else { $all_done = 1; }
}
现在,脚本运行良好,大多数情况下。但是,当 3个或更多脚本并行运行(相同的脚本,单独的进程,而不是线程)时,会出现段错误。
我是如何解决的? 我通过在while(!$ all_done)循环期间每次执行DBH :: prepare调用来解决它。
因此,即使多个进程并行运行,此代码也不会出现段错误。我一再重现错误,然后我对新代码做了同样的事情。我很肯定在循环中移动语句会修复问题。
为什么会发生这种情况的任何想法?
我使用的是perl 5.8和perl-DBI版本1.609。
这里也是脚本段错误时strace的输出:
read(5, "\1\7\0\0\6\0\0\0\0\0\20\27\234\312\272\221eG2;\33S\364\230\313\220\221Bxp\4\7"..., 2064) = 263
write(4, "\1\31\0\0\6\0\0\0\0\0\21i \1\1\0\0\0\2\0\0\0\3^!)\4\4\0\0\0\0"..., 281) = 281
read(4, "\0\177\0\0\6\0\0\0\0\0\v\5\2\0\0\0\1\0\0\0\0\0\0\0\0\0\0\0 \10\6"..., 2064) = 127
write(2, "debug:Waiting for 1 blocker task"..., 49debug:Waiting for 1 blocker tasks to be finished
) = 49
write(5, "\0\252\0\0\6\0\0\0\0\0\3^\20p\200\0\0\2\0\0\0\0\0\0\0\0\1\r\0\0\0\0"..., 170) = 170
read(5, "\0\301\0\0\6\0\0\0\0\0\6\"\2\0\0\0@\0\0\0\0\0\0\0\0\0\0\0\7 ru"..., 2064) = 193
write(5, "\1]\0\0\6\0\0\0\0\0\3^\21)\200\0\0\0\0\0\0\1\234\0\0\0\1\r\0\0\0\0"..., 349) = 349
read(5, "\0y\0\0\6\0\0\0\0\0\10\6\0S\254b\f\0\t\0\0\1\0\0\0\1\0\0\0\0\0\0"..., 2064) = 121
write(5, "\0000\0\0\6\0\0\0\0\0\3h\22\1\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0"..., 48) = 48
read(5, "\0\26\0\0\6\0\0\0\0\0\10\2\0\0\0\t\5\0\0\0\21\0", 2064) = 22
time(NULL) = 1333827285
rt_sigprocmask(SIG_BLOCK, [CHLD], [], 8) = 0
rt_sigaction(SIGCHLD, NULL, {SIG_DFL}, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
nanosleep({10, 0}, {10, 0}) = 0
time(NULL) = 1333827295
write(4, "\1\31\0\0\6\0\0\0\0\0\21i\"\1\1\0\0\0\3\0\0\0\3^#)\4\4\0\0\0\0"..., 281) = 281
read(4, "\0\177\0\0\6\0\0\0\0\0\v\5\2\0\0\0\1\0\0\0\0\0\0\0\0\0\0\0 \10\6"..., 2064) = 127
write(2, "debug:Waiting for 1 blocker task"..., 49debug:Waiting for 1 blocker tasks to be finished
) = 49
write(5, "\0)\0\0\6\0\0\0\0\0\21i\23\1\1\0\0\0\1\0\0\0\3N\24\2\0\0\0@\0\0"..., 41) = 41
read(5, "\1>\0\0\6\0\0\0\0\0\20\27\234\312\272\221eG2;\33S\364\230\313\220\221Bxp\4\7"..., 2064) = 318
--- SIGSEGV (Segmentation fault) @ 0 (0) ---
+++ killed by SIGSEGV +++
[ Process PID=22767 runs in 32 bit mode. ]
答案 0 :(得分:1)
我会说你没有太多担心,因为SQLite语句准备往往很快。我发现通过Perl DBI进行的简单SQL准备只需要20微秒(参见下面的结果和代码)。考虑到您的申请,您不应该注意到性能上的任何差异。
由于SQLite的经典锁定一次只允许一个编写者和多个读者,因此您遇到的问题可能与锁定或事务处理有关。然而,SIGSEGV永远不是预期的行为。
结果
Perl 5.014002
DBI 1.618
SQLite 3.7.9
time in s for 100000 prepares: 2.01810789108276
ys per prepare: 20.1810789108276
代码
use DBI;
use Time::HiRes qw ( time );
use strict;
use warnings;
my $dbc = DBI->connect (
'dbi:SQLite:dbname=/tmp/test.db',
undef, undef, { AutoCommit => 0, RaiseError => 1, PrintError => 1 }
) || die $DBI::errstr;
print "Perl $]\n";
print "DBI $DBI::VERSION\n";
print "SQLite $dbc->{sqlite_version}\n";
my $start = time();
my $n = 100_000;
foreach ( 1 .. $n ) {
my $stmt = $dbc->prepare( qq{
select count(*) from sec where sid is not null
} );
}
my $end = time();
print
"time in s for $n prepares: " .
( $end - $start ) .
"\n";
print
"ys per prepare: " .
( ( ( $end - $start ) * 1_000_000 ) / $n ) .
"\n";