До сих пор я никогда не использовал Thread, но думаю, что в этом случае я должен положиться на него. Я хотел бы обрабатывать stdout и stderr командной строки cURL отдельно, потому что я хочу обменять возврат каретки в индикаторе выполнения (который записывается в stderr) на новые строки:
require "open3"
cmd="curl -b cookie.txt #{url} -L -o -"
Open3.popen3(cmd) do |stdin, stdout, stderr, wait_thr|
pid = wait_thr.pid
# I have to process stdout and stderr at the same time but
#asyncronously, because stdout gives much more data then the stderr
#stream. I instantiate a Thread object for reading the stderr, otherwise
#"getc" would block the stdout processing loop.
c=nil
line=""
stdout.each_char do |b|
STDOUT.print b
if c==nil then
c=""
thr = Thread.new {
c=stderr.getc
if c=="\r" || c=="\n" then
STDERR.puts line
line=""
else
line<<c
end
c=nil
}
end
#if stderr still holds some output then I process it:
line=""
stderr.each_char do |c|
if c=="\r" || c=="\n" then
STDERR.puts line
line=""
else
line<<c
end
end
exit_status = wait_thr.value.exitstatus
STDERR.puts exit_status
end #popen3
Мой вопрос: как я могу избежать создания нового экземпляра Thread в каждом цикле цикла при обработке stdout (stdout.each_char)? Я думаю, что это отнимает много времени, я хотел бы создать экземпляр один раз, а затем использовать его методы, такие как остановка и запуск и т. д.
stderr.getc
(вторая используетeach_char
)? - person Arie Xiao   schedule 06.07.2015