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