原です。
In message "[ruby-list:15030] ConditionVariable (cv3.rb)"
on 99/06/17, Shin-ichiro Hara <sinara / blade.nagaokaut.ac.jp> writes:
|test1.rb を改造した test-cv.rb(後掲)で、バッファサイズ3のパ
|イプに、送り手4スレッド受け手3スレッドでオブジェクトを通すも
|のです。停止するのは全ての受け手スレッドが 1000 個のオブジェク
|トを受け取った時とします。
一応 test-cv.rb です。無いよりましという程度で、必ずしもポイント
をついたいいテストプログラムというわけではありません。
幾つかパラメータをコマンドライン・パラメータに取るので、いろいろ
やってみると面白いですが、数字によって結構(数論的な?)動きがあ
ったりして、実は簡単に結論が出せません。cv3.rb はパラメータによ
らず比較的安定しているのは確かですが。もうちょっとちゃんとしたテ
ストプログラムが無いとこれ以上の効率や適正性の追求は難しそうです。
(バッファサイズ10送り手12受け手3で100回す例)
$ ruby test-cv.rb max10 s12 r3 100 cv3.rb
-------------^ test-cv.rb
#!/usr/local/bin/ruby
Thread.abort_on_exception = true
$sender_num = 4
$receiver_num = 3
$rtimes = 1000
$max = 3
$debug = 0
while para = ARGV.shift
case para
when /^n$/
$no_signal_mutex = true
when /^p$/
$pass = true
when /^d(\d)?$/
$debug = ($1||1).to_i
when /^\d+$/
$rtimes = $&.to_i
when /^max(\d+)$/
$max = $1.to_i
when /^r(\d+)$/
$receiver_num = $1.to_i
when /^s(\d+)$/
$sender_num = $1.to_i
else
$o = para
end
end
class Thread
Num = [0]
Obj2num = {Thread.current => 0}
def Thread.start0
Num[0] = Num[0]+1
t = Thread.start {
Thread.stop
yield
}
Obj2num[t] = Num[0]
t
end
def num
Obj2num[self]
end
def Thread.num
Obj2num[Thread.current]
end
alias to_s0 to_s
def inspect; "#{Obj2num[self]}"; end
def to_s; "#{Obj2num[self]}"; end
end
case $o
when /cv(\d+)/
require $o
$ver = $1.to_i
$full = ConditionVariable.new
$mutex = Mutex.new
if $ver > 2
$signal_param = $mutex unless $no_signal_mutex
end
$wait_param = $mutex
$empty = ConditionVariable.new
when /^monitor/
require $o
$mutex = Monitor.new
$full = $mutex.new_cond
$empty = $mutex.new_cond
when /^mutex/
require $o
$mutex = Mutex.new
$full = ConditionVariable.new($mutex)
$empty = ConditionVariable.new($mutex)
else
raise "can't get option: #$o"
end
print "TEST CONDITION VARIABLE <<< #{$o.upcase} >>>\n"
printf("Sender:%d, Receiver:%d (each receives %d times), Buffer:#$max\n",
$sender_num, $receiver_num, $rtimes)
$queue = []
$rwaste = 0
$rwaste1 = 0
$swaste = 0
$swaste1 = 0
$snds = []
$rcvs = []
$snds_times = {}
$rcvs_times = {}
$sender_num.times do
$snds.push Thread.start0 {
n = 0
$snds_times[Thread.current] = 0
loop do
$mutex.synchronize {
k = 0
while $queue.size >= $max
k += 1
if $wait_param
$full.wait($wait_param)
else
$full.wait
end
$swaste += 1 if k > 1
end
$swaste1 += 1 if k > 1
$queue.push [n, Thread.num]
if $queue.size >= 1 # == true
if $signal_param
$empty.signal($signal_param)
else
$empty.signal
end
end
}
n += 1
$snds_times[Thread.current] = n
Thread.pass if $pass
end
}
end
$receiver_num.times do
$rcvs.push Thread.start0 {
n = 0
$rcvs_times[Thread.current] = 0
$rtimes.times do
$mutex.synchronize {
k = 0
while $queue.size <= 0
k += 1
if $wait_param
$empty.wait($wait_param)
else
$empty.wait
end
$rwaste += 1 if k > 1
end
$rwaste1 += 1 if k > 1
v = $queue.shift
if $queue.size < $max
if $signal_param
$full.signal($signal_param)
else
$full.signal
end
end
printf("get %s (#%s->#%s)\n", v[0], v[1], Thread.num) if $debug > 0
}
n += 1
$rcvs_times[Thread.current] = n
Thread.pass if $pass
end
}
end
# START
Thread.critical = true
Thread::Obj2num.each do |t, n|
t.wakeup if n > 0
end
Thread.critical = false
$rcvs.each do |t|
VERSION.to_f >= 1.3 ? t.join : Thread.join(t)
end
$snds.each do |t|
Thread.kill(t)
end
printf("Idling1: send =%2d, receive =%2d\n", $swaste1, $rwaste1)
printf("Idling2: send =%2d, receive =%2d\n", $swaste, $rwaste)
$snds.each do |t|
printf("#%s sends %d times\n", t, $snds_times[t])
end
-------------$ test-cv.rb
以外と長かった。(^^;