Ruby - มัลติเธรด
โปรแกรมแบบดั้งเดิมมีเธรดเดียวของการดำเนินการคำสั่งหรือคำสั่งที่ประกอบด้วยโปรแกรมจะดำเนินการตามลำดับจนกว่าโปรแกรมจะสิ้นสุด
โปรแกรมมัลติเธรดมีเธรดการดำเนินการมากกว่าหนึ่งเธรด ภายในแต่ละเธรดคำสั่งจะถูกดำเนินการตามลำดับ แต่เธรดเองอาจถูกเรียกใช้แบบขนานบนซีพียูแบบมัลติคอร์ บ่อยครั้งในเครื่องซีพียูเครื่องเดียวเธรดหลายเธรดจะไม่ถูกดำเนินการแบบขนานจริง ๆ แต่การขนานถูกจำลองขึ้นโดยการสลับการทำงานของเธรด
Ruby ทำให้ง่ายต่อการเขียนโปรแกรมมัลติเธรดด้วยคลาสเธรด เธรด Ruby เป็นวิธีที่มีน้ำหนักเบาและมีประสิทธิภาพในการบรรลุการทำงานพร้อมกันในโค้ดของคุณ
การสร้าง Ruby Threads
ในการเริ่มต้นหัวข้อใหม่เพียงแค่เชื่อมโยงบล็อกที่มีการเรียกไปยังThread.new เธรดใหม่จะถูกสร้างขึ้นเพื่อรันโค้ดในบล็อกและเธรดเดิมจะกลับมาจากเธรดใหม่ทันทีและดำเนินการต่อด้วยคำสั่งถัดไป -
# Thread #1 is running here
Thread.new {
# Thread #2 runs this code
}
# Thread #1 runs this code
ตัวอย่าง
นี่คือตัวอย่างซึ่งแสดงให้เห็นว่าเราสามารถใช้โปรแกรม Ruby แบบมัลติเธรดได้อย่างไร
#!/usr/bin/ruby
def func1
i = 0
while i<=2
puts "func1 at: #{Time.now}"
sleep(2)
i = i+1
end
end
def func2
j = 0
while j<=2
puts "func2 at: #{Time.now}"
sleep(1)
j = j+1
end
end
puts "Started At #{Time.now}"
t1 = Thread.new{func1()}
t2 = Thread.new{func2()}
t1.join
t2.join
puts "End at #{Time.now}"
สิ่งนี้จะให้ผลลัพธ์ดังต่อไปนี้ -
Started At Wed May 14 08:21:54 -0700 2008
func1 at: Wed May 14 08:21:54 -0700 2008
func2 at: Wed May 14 08:21:54 -0700 2008
func2 at: Wed May 14 08:21:55 -0700 2008
func1 at: Wed May 14 08:21:56 -0700 2008
func2 at: Wed May 14 08:21:56 -0700 2008
func1 at: Wed May 14 08:21:58 -0700 2008
End at Wed May 14 08:22:00 -0700 2008
วงจรชีวิตของเธรด
กระทู้ใหม่ได้ถูกสร้างขึ้นด้วยThread.new นอกจากนี้คุณยังสามารถใช้คำพ้องthread.startและThread.fork
ไม่จำเป็นต้องเริ่มเธรดหลังจากสร้างเธรดแล้วจะเริ่มทำงานโดยอัตโนมัติเมื่อทรัพยากร CPU พร้อมใช้งาน
คลาสเธรดกำหนดวิธีการหลายวิธีในการสอบถามและจัดการเธรดในขณะที่รัน เธรดรันโค้ดในบล็อกที่เกี่ยวข้องกับการเรียกไปยังThread.newจากนั้นจะหยุดทำงาน
ค่าของนิพจน์สุดท้ายในบล็อกนั้นคือค่าของเธรดและสามารถหาได้โดยการเรียกใช้วิธีค่าของอ็อบเจ็กต์เธรด หากเธรดทำงานจนเสร็จสิ้นค่าจะส่งกลับค่าของเธรดทันที มิฉะนั้นเมธอดค่าจะบล็อกและไม่ส่งคืนจนกว่าเธรดจะเสร็จสมบูรณ์
เมธอดคลาสThread.currentส่งคืนอ็อบเจ็กต์ Thread ที่แสดงถึงเธรดปัจจุบัน สิ่งนี้ทำให้เธรดสามารถจัดการตัวเองได้ เมธอดคลาสThread.mainส่งคืนอ็อบเจ็กต์ Thread ที่แสดงถึงเธรดหลัก นี่คือเธรดเริ่มต้นของการดำเนินการที่เริ่มต้นเมื่อโปรแกรม Ruby เริ่มทำงาน
คุณสามารถรอให้เธรดหนึ่ง ๆ เสร็จสิ้นโดยเรียกใช้เมธอดThread.joinของเธรดนั้น เธรดการเรียกจะบล็อกจนกว่าเธรดที่กำหนดจะเสร็จสิ้น
เธรดและข้อยกเว้น
หากมีข้อยกเว้นเกิดขึ้นในเธรดหลักและไม่ได้รับการจัดการที่ใด ๆ ตัวแปล Ruby จะพิมพ์ข้อความและออก ในเธรดนอกเหนือจากเธรดหลักข้อยกเว้นที่ไม่สามารถจัดการได้ทำให้เธรดหยุดทำงาน
ถ้าด้าย t ออกเนื่องจากข้อยกเว้นที่ไม่สามารถจัดการได้และเธรดอื่น sเรียกt.join หรือ t.valueตามด้วยข้อยกเว้นที่เกิดขึ้นในt ถูกยกขึ้นในเธรด s.
หากThread.abort_on_exceptionเป็นเท็จเงื่อนไขดีฟอลต์ข้อยกเว้นที่ไม่สามารถจัดการได้เพียงแค่ฆ่าเธรดปัจจุบันและส่วนที่เหลือทั้งหมดจะทำงานต่อไป
หากคุณต้องการจัดการข้อยกเว้นใด ๆ ในหัวข้อใด ๆ ที่จะก่อให้เกิดล่ามเพื่อออกจากการตั้งค่าวิธีการเรียนThread.abort_on_exceptionไปจริง
t = Thread.new { ... }
t.abort_on_exception = true
ตัวแปรของเธรด
โดยปกติเธรดสามารถเข้าถึงตัวแปรใด ๆ ที่อยู่ในขอบเขตเมื่อเธรดถูกสร้างขึ้น ตัวแปรภายในสำหรับบล็อกของเธรดเป็นแบบโลคัลของเธรดและจะไม่ถูกแชร์
คลาสเธรดมีสิ่งอำนวยความสะดวกพิเศษที่อนุญาตให้สร้างและเข้าถึงตัวแปรเธรดโลคัลโดยใช้ชื่อ คุณเพียงแค่ปฏิบัติต่อออบเจ็กต์เธรดราวกับว่าเป็นแฮชเขียนไปยังองค์ประกอบโดยใช้ [] = และอ่านกลับโดยใช้ []
ในตัวอย่างนี้แต่ละหัวข้อบันทึกมูลค่าปัจจุบันของจำนวนตัวแปรในตัวแปร ThreadLocal กับคีย์mycount
#!/usr/bin/ruby
count = 0
arr = []
10.times do |i|
arr[i] = Thread.new {
sleep(rand(0)/10.0)
Thread.current["mycount"] = count
count += 1
}
end
arr.each {|t| t.join; print t["mycount"], ", " }
puts "count = #{count}"
สิ่งนี้ก่อให้เกิดผลลัพธ์ดังต่อไปนี้ -
8, 0, 3, 7, 2, 1, 6, 5, 4, 9, count = 10
เธรดหลักรอให้เธรดย่อยเสร็จสิ้นจากนั้นพิมพ์ค่าของจำนวนที่จับโดยแต่ละเธรด
ลำดับความสำคัญของเธรด
ปัจจัยแรกที่มีผลต่อการจัดกำหนดการเธรดคือลำดับความสำคัญของเธรด: เธรดที่มีลำดับความสำคัญสูงจะถูกจัดกำหนดการก่อนเธรดที่มีลำดับความสำคัญต่ำ อย่างแม่นยำยิ่งขึ้นเธรดจะได้รับเวลาของ CPU ก็ต่อเมื่อไม่มีเธรดที่มีลำดับความสำคัญสูงกว่ารอรัน
คุณสามารถตั้งค่าและแบบสอบถามความสำคัญของวัตถุกระทู้ทับทิมกับpriority =และจัดลำดับความสำคัญ เธรดที่สร้างขึ้นใหม่จะเริ่มต้นด้วยลำดับความสำคัญเดียวกับเธรดที่สร้างขึ้น เธรดหลักเริ่มต้นที่ลำดับความสำคัญ 0
ไม่มีวิธีกำหนดลำดับความสำคัญของเธรดก่อนที่จะเริ่มทำงาน อย่างไรก็ตามเธรดสามารถเพิ่มหรือลดลำดับความสำคัญของตัวเองได้ตามการดำเนินการครั้งแรก
การยกเว้นเธรด
หากเธรดสองเธรดแชร์การเข้าถึงข้อมูลเดียวกันและเธรดอย่างน้อยหนึ่งเธรดแก้ไขข้อมูลนั้นคุณต้องใช้ความระมัดระวังเป็นพิเศษเพื่อให้แน่ใจว่าเธรดจะไม่เห็นข้อมูลในสถานะที่ไม่สอดคล้องกัน นี้เรียกว่าการยกเว้นด้าย
Mutexเป็นคลาสที่ใช้การล็อกสัญญาณอย่างง่ายสำหรับการเข้าถึงทรัพยากรที่ใช้ร่วมกันโดยเฉพาะ นั่นคือเธรดเดียวเท่านั้นที่สามารถล็อคได้ในเวลาที่กำหนด เธรดอื่น ๆ อาจเลือกที่จะรอเพื่อให้ล็อคพร้อมใช้งานหรืออาจเลือกที่จะได้รับข้อผิดพลาดทันทีที่ระบุว่าไม่มีการล็อก
ด้วยการกำหนดให้การเข้าถึงข้อมูลที่แชร์ทั้งหมดอยู่ภายใต้การควบคุมของmutexเราจึงมั่นใจได้ว่าการทำงานของอะตอมมีความสอดคล้องกัน มาลองดูตัวอย่างอันแรกที่ไม่มี mutax และอันที่สองด้วย mutax -
ตัวอย่างที่ไม่มี Mutax
#!/usr/bin/ruby
require 'thread'
count1 = count2 = 0
difference = 0
counter = Thread.new do
loop do
count1 += 1
count2 += 1
end
end
spy = Thread.new do
loop do
difference += (count1 - count2).abs
end
end
sleep 1
puts "count1 : #{count1}"
puts "count2 : #{count2}"
puts "difference : #{difference}"
สิ่งนี้จะให้ผลลัพธ์ดังต่อไปนี้ -
count1 : 1583766
count2 : 1583766
difference : 0
#!/usr/bin/ruby
require 'thread'
mutex = Mutex.new
count1 = count2 = 0
difference = 0
counter = Thread.new do
loop do
mutex.synchronize do
count1 += 1
count2 += 1
end
end
end
spy = Thread.new do
loop do
mutex.synchronize do
difference += (count1 - count2).abs
end
end
end
sleep 1
mutex.lock
puts "count1 : #{count1}"
puts "count2 : #{count2}"
puts "difference : #{difference}"
สิ่งนี้จะให้ผลลัพธ์ดังต่อไปนี้ -
count1 : 696591
count2 : 696591
difference : 0
การจัดการ Deadlock
เมื่อเราเริ่มใช้Mutexวัตถุสำหรับการยกเว้นด้ายเราจะต้องระมัดระวังเพื่อหลีกเลี่ยงการหยุดชะงัก Deadlock เป็นเงื่อนไขที่เกิดขึ้นเมื่อเธรดทั้งหมดกำลังรอรับทรัพยากรที่มีเธรดอื่นอยู่ เนื่องจากเธรดทั้งหมดถูกบล็อกจึงไม่สามารถคลายล็อกที่ยึดไว้ได้ และเนื่องจากไม่สามารถปลดล็อกได้จึงไม่มีเธรดอื่นที่สามารถรับล็อกเหล่านั้นได้
นี่คือที่ที่ตัวแปรเงื่อนไขเข้ามาในภาพ ตัวแปรสภาพเป็นเพียงสัญญาณที่เกี่ยวข้องกับทรัพยากรและมีการใช้ในการป้องกันโดยเฉพาะอย่างยิ่งที่mutex เมื่อคุณต้องการทรัพยากรที่ไม่พร้อมใช้งานคุณจะต้องรอตัวแปรเงื่อนไข การดำเนินการดังกล่าวจะคลายล็อกบนmutex ที่เกี่ยวข้อง เมื่อเธรดอื่น ๆ บางตัวส่งสัญญาณว่าทรัพยากรพร้อมใช้งานเธรดดั้งเดิมจะหยุดการรอและล็อกในพื้นที่วิกฤตในเวลาเดียวกัน
ตัวอย่าง
#!/usr/bin/ruby
require 'thread'
mutex = Mutex.new
cv = ConditionVariable.new
a = Thread.new {
mutex.synchronize {
puts "A: I have critical section, but will wait for cv"
cv.wait(mutex)
puts "A: I have critical section again! I rule!"
}
}
puts "(Later, back at the ranch...)"
b = Thread.new {
mutex.synchronize {
puts "B: Now I am critical, but am done with cv"
cv.signal
puts "B: I am still critical, finishing up"
}
}
a.join
b.join
สิ่งนี้จะให้ผลลัพธ์ดังต่อไปนี้ -
A: I have critical section, but will wait for cv
(Later, back at the ranch...)
B: Now I am critical, but am done with cv
B: I am still critical, finishing up
A: I have critical section again! I rule!
สถานะของเธรด
มีค่าส่งคืนที่เป็นไปได้ห้าค่าที่สอดคล้องกับสถานะที่เป็นไปได้ห้าสถานะดังแสดงในตารางต่อไปนี้ สถานะวิธีการส่งกลับรัฐของด้าย
สถานะเธรด | ส่งคืนค่า |
---|---|
รันได้ | วิ่ง |
นอน | นอน |
การยกเลิก | แท้ง |
สิ้นสุดตามปกติ | เท็จ |
สิ้นสุดโดยมีข้อยกเว้น | ศูนย์ |
วิธีการคลาสของเธรด
วิธีการต่อไปนี้จัดเตรียมโดยคลาสของเธรดและใช้ได้กับเธรดทั้งหมดที่มีอยู่ในโปรแกรม วิธีการเหล่านี้จะถูกเรียกโดยใช้ชื่อคลาสของเธรดดังนี้ -
Thread.abort_on_exception = true
ซีเนียร์ | วิธีการและคำอธิบาย |
---|---|
1 | Thread.abort_on_exception ส่งคืนสถานะของการยกเลิกส่วนกลางในเงื่อนไขข้อยกเว้น เริ่มต้นเป็นเท็จ เมื่อตั้งค่าเป็นจริงจะทำให้เธรดทั้งหมดยกเลิก (กระบวนการจะออกจาก (0)) หากมีการเพิ่มข้อยกเว้นในเธรดใด ๆ |
2 | Thread.abort_on_exception= เมื่อตั้งค่าเป็นจริงเธรดทั้งหมดจะยกเลิกหากมีการเพิ่มข้อยกเว้น ส่งคืนสถานะใหม่ |
3 | Thread.critical ส่งคืนสถานะของสภาวะวิกฤตเธรดส่วนกลาง |
4 | Thread.critical= ตั้งค่าสถานะของสภาวะวิกฤตเธรดส่วนกลางและส่งคืน เมื่อตั้งค่าเป็นจริงห้ามมิให้กำหนดเวลาของเธรดที่มีอยู่ ไม่บล็อกเธรดใหม่จากการสร้างและเรียกใช้ การดำเนินการของเธรดบางอย่าง (เช่นการหยุดหรือการฆ่าเธรดการนอนหลับในเธรดปัจจุบันและการเพิ่มข้อยกเว้น) อาจทำให้เธรดถูกกำหนดเวลาได้แม้ว่าจะอยู่ในส่วนวิกฤตก็ตาม |
5 | Thread.current ส่งคืนเธรดที่กำลังดำเนินการอยู่ |
6 | Thread.exit ยุติเธรดที่กำลังรันอยู่และกำหนดเวลาเธรดอื่นที่จะรัน หากเธรดนี้ถูกทำเครื่องหมายว่าถูกฆ่าแล้วexitจะส่งกลับเธรด ถ้านี่คือเธรดหลักหรือเธรดสุดท้ายให้ออกจากกระบวนการ |
7 | Thread.fork { block } ไวพจน์สำหรับ Thread.new. |
8 | Thread.kill( aThread ) ทำให้เธรดที่กำหนดออก |
9 | Thread.list ส่งคืนอาร์เรย์ของวัตถุเธรดสำหรับเธรดทั้งหมดที่รันได้หรือหยุดทำงาน เกลียว. |
10 | Thread.main ส่งคืนเธรดหลักสำหรับกระบวนการ |
11 | Thread.new( [ arg ]* ) {| args | block } สร้างเธรดใหม่เพื่อดำเนินการตามคำสั่งที่ให้ไว้ในบล็อกและเริ่มรัน อาร์กิวเมนต์ใด ๆ ที่ส่งไปยังThread.newจะถูกส่งผ่านไปยังบล็อก |
12 | Thread.pass เรียกใช้ตัวกำหนดตารางเวลาเธรดเพื่อส่งผ่านการดำเนินการไปยังเธรดอื่น |
13 | Thread.start( [ args ]* ) {| args | block } โดยทั่วไปเช่นเดียวกับThread.new อย่างไรก็ตามหากคลาสเธรดเป็นคลาสย่อยการเรียกstartในคลาสย่อยนั้นจะไม่เรียกใช้เมธอดinitializeของคลาสย่อย |
14 | Thread.stop หยุดการดำเนินการของเธรดปัจจุบันวางลงในการนอนหลับของรัฐและตารางเวลาการดำเนินการของหัวข้ออื่น รีเซ็ตเงื่อนไขวิกฤตเป็นเท็จ |
วิธีการอินสแตนซ์ของเธรด
วิธีการเหล่านี้ใช้ได้กับอินสแตนซ์ของเธรด วิธีการเหล่านี้จะถูกเรียกโดยใช้ตัวอย่างของเธรดดังนี้ -
#!/usr/bin/ruby
thr = Thread.new do # Calling a class method new
puts "In second thread"
raise "Raise exception"
end
thr.join # Calling an instance method join
ซีเนียร์ | วิธีการและคำอธิบาย |
---|---|
1 | thr[ aSymbol ] Attribute Reference - ส่งกลับค่าของตัวแปร thread-local โดยใช้สัญลักษณ์หรือชื่อaSymbol ถ้าตัวแปรที่ระบุไม่มีผลตอบแทนNil |
2 | thr[ aSymbol ] = การกำหนดแอตทริบิวต์ - ตั้งค่าหรือสร้างค่าของตัวแปรเธรดโลคัลโดยใช้สัญลักษณ์หรือสตริง |
3 | thr.abort_on_exception ส่งกลับสถานะของการยกเลิกในการยกเว้นเงื่อนไขในการTHR เริ่มต้นเป็นเท็จ |
4 | thr.abort_on_exception= เมื่อตั้งค่าเป็นจริงทำให้เกิดกระทู้ทั้งหมด (รวมทั้งโปรแกรมหลัก) จะยกเลิกถ้ายกเว้นจะเติบโตในTHR กระบวนการอย่างมีประสิทธิภาพจะออก (0) |
5 | thr.alive? คืนค่าจริงถ้าThrกำลังทำงานหรือนอนหลับ |
6 | thr.exit ยุติการกระตุกและกำหนดเวลาให้เธรดอื่นทำงาน ถ้ากระทู้นี้ถูกทำเครื่องหมายแล้วที่จะถูกฆ่าตายทางออกกลับกระทู้ หากนี่คือเธรดหลักหรือเธรดสุดท้ายให้ออกจากกระบวนการ |
7 | thr.join ด้ายโทรจะระงับการดำเนินการและการทำงานที่นั่น ไม่กลับมาจนกว่าTHRออก เธรดใด ๆ ที่ไม่ได้เข้าร่วมจะถูกฆ่าเมื่อโปรแกรมหลักออก |
8 | thr.key? ส่งคืนจริงหากสตริง (หรือสัญลักษณ์) ที่กำหนดมีอยู่เป็นตัวแปรเธรดโลคัล |
9 | thr.kill ไวพจน์Thread.exit |
10 | thr.priority ผลตอบแทนที่ได้จัดลำดับความสำคัญของTHR ค่าเริ่มต้นคือศูนย์ เธรดที่มีลำดับความสำคัญสูงกว่าจะทำงานก่อนเธรดที่มีลำดับความสำคัญต่ำกว่า |
11 | thr.priority= ตั้งค่าลำดับความสำคัญของthrเป็นจำนวนเต็ม เธรดที่มีลำดับความสำคัญสูงกว่าจะทำงานก่อนเธรดที่มีลำดับความสำคัญต่ำกว่า |
12 | thr.raise( anException ) ยกข้อยกเว้นจากที่นั่น โทรไม่จำเป็นต้องเป็นนั่น |
13 | thr.run ตื่นขึ้นมานั่นทำให้มันมีสิทธิ์สำหรับการจัดตาราง หากไม่ได้อยู่ในส่วนที่สำคัญให้เรียกใช้ตัวกำหนดตารางเวลา |
14 | thr.safe_level ผลตอบแทนในระดับที่ปลอดภัยมีผลสำหรับที่นั่น |
15 | thr.status ส่งคืนสถานะของthr : sleepหากthrอยู่ในโหมดสลีปหรือรอบน I / O รันหากthrกำลังดำเนินการเป็นเท็จหากthrสิ้นสุดตามปกติและไม่มีถ้าthrสิ้นสุดโดยมีข้อยกเว้น |
16 | thr.stop? คืนค่าจริงหากthrตายหรือหลับ |
17 | thr.value รอให้ thr ดำเนินการผ่านThread.joinและส่งคืนค่า |
18 | thr.wakeup ทำเครื่องหมายว่าthrมีสิทธิ์สำหรับการตั้งเวลาอย่างไรก็ตามอาจยังคงถูกบล็อกบน I / O |