激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

腳本之家,腳本語言編程技術(shù)及教程分享平臺!
分類導(dǎo)航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務(wù)器之家 - 腳本之家 - Ruby - Ruby 多線程的潛力和弱點(diǎn)分析

Ruby 多線程的潛力和弱點(diǎn)分析

2020-04-24 10:46腳本之家 Ruby

這篇文章主要介紹了Ruby 多線程的潛力和弱點(diǎn)分析,本文講解了Ruby 多線程和 IO Block、Ruby GIL 的影響、JRuby 去除了 GIL、Ruby 多線程總結(jié)等內(nèi)容,需要的朋友可以參考下

Web 應(yīng)用大多是 IO 密集型的,利用 Ruby 多進(jìn)程+多線程模型將能大幅提升系統(tǒng)吞吐量。其原因在于:當(dāng)Ruby 某個(gè)線程處于 IO Block 狀態(tài)時(shí),其它的線程還可以繼續(xù)執(zhí)行。但由于存在 Ruby GIL (Global Interpreter Lock),MRI Ruby 并不能真正利用多線程進(jìn)行并行計(jì)算。JRuby 去除了 GIL,是真正意義的多線程,既能應(yīng)付 IO Block,也能充分利用多核 CPU 加快整體運(yùn)算速度。

上面說得比較抽象,下面就用例子一一加以說明。

Ruby 多線程和 IO Block

先看下面一段代碼(演示目的,沒有實(shí)際用途):

復(fù)制代碼 代碼如下:


# File: block_io1.rb

 

def func1
  puts "sleep 3 seconds in func1\n"
  sleep(3)
end

def func2
  puts "sleep 2 seconds in func2\n"
  sleep(2)
end

def func3
  puts "sleep 5 seconds in func3\n"
  sleep(5)
end

func1
func2
func3

 

代碼很簡單,3 個(gè)方法,用 sleep 模擬耗時(shí)的 IO 操作。 運(yùn)行代碼(環(huán)境 MRI Ruby 1.9.3) 結(jié)果是:

復(fù)制代碼 代碼如下:


$ time ruby block_io1.rb
sleep 3 seconds in func1
sleep 2 seconds in func2
sleep 5 seconds in func3

 

real  0m11.681s
user  0m3.086s
sys 0m0.152s

 

比較慢,時(shí)間都耗在 sleep 上了,總共花了 10 多秒。

采用多線程的方式,改寫如下:

復(fù)制代碼 代碼如下:


# File: block_io2.rb

 

def func1
  puts "sleep 3 seconds in func1\n"
  sleep(3)
end

def func2
  puts "sleep 2 seconds in func2\n"
  sleep(2)
end

def func3
  puts "sleep 5 seconds in func3\n"
  sleep(5)
end

threads = []
threads << Thread.new { func1 }
threads << Thread.new { func2 }
threads << Thread.new { func3 }

threads.each { |t| t.join }

 

運(yùn)行的結(jié)果是:

復(fù)制代碼 代碼如下:


$ time ruby block_io2.rb
sleep 3 seconds in func1
sleep 2 seconds in func2
sleep 5 seconds in func3

 

real  0m6.543s
user  0m3.169s
sys 0m0.147s

 

總共花了 6 秒多,明顯快了許多,只比最長的 sleep 5 秒多了一點(diǎn)。

上面的例子說明,Ruby 的多線程能夠應(yīng)付 IO Block,當(dāng)某個(gè)線程處于 IO Block 狀態(tài)時(shí),其它的線程還可以繼續(xù)執(zhí)行,從而使整體處理時(shí)間大幅縮短。


Ruby GIL 的影響

還是先看一段代碼(演示目的):

 

復(fù)制代碼 代碼如下:


# File: gil1.rb

 

require 'securerandom'
require 'zlib'

data = SecureRandom.hex(4096000)

16.times { Zlib::Deflate.deflate(data) }

 

代碼先隨機(jī)生成一些數(shù)據(jù),然后對其進(jìn)行壓縮,壓縮是非常耗 CPU 的,在我機(jī)器(雙核 CPU, MRI Ruby 1.9.3)運(yùn)行結(jié)果如下:

復(fù)制代碼 代碼如下:


$ time ruby gil1.rb

 

real  0m8.572s
user  0m8.359s
sys 0m0.102s

 

更改為多線程版本,代碼如下:

復(fù)制代碼 代碼如下:


# File: gil2.rb

 

require 'securerandom'
require 'zlib'

data = SecureRandom.hex(4096000)

threads = []
16.times do
  threads << Thread.new { Zlib::Deflate.deflate(data) }
end

threads.each {|t| t.join}

 

多線程的版本運(yùn)行結(jié)果如下:

復(fù)制代碼 代碼如下:


$ time ruby gil2.rb

 

real  0m8.616s
user  0m8.377s
sys 0m0.211s

 

從結(jié)果可以看出,由于 MRI Ruby GIL 的存在,Ruby 多線程并不能重復(fù)利用多核 CPU,使用多線程后整體所花時(shí)間并不縮短,反而由于線程切換的影響,所花時(shí)間還略有增加。

JRuby 去除了 GIL

使用 JRuby (我的機(jī)器上是 JRuby 1.7.0)運(yùn)行 gil1.rb 和 gil2.rb,得到很不一樣的結(jié)果。

 

復(fù)制代碼 代碼如下:


$ time jruby gil1.rb

 

real  0m12.225s
user  0m14.060s
sys 0m0.615s

 

復(fù)制代碼 代碼如下:


$ time jruby gil2.rb

 

real  0m7.584s
user  0m22.822s
sys 0m0.819s


可以看到,JRuby 使用多線程時(shí),整體運(yùn)行時(shí)間有明顯縮短(7.58 比 12.22),這是由于 JRuby 去除了 GIL,可以真正并行的執(zhí)行多線程,充分利用了多核 CPU。

 

總結(jié):Ruby 多線程可以在某個(gè)線程 IO Block 時(shí),依然能夠執(zhí)行其它線程,從而降低 IO Block 對整體的影響,但由于 MRI Ruby GIL 的存在,MRI Ruby 并不是真正的并行執(zhí)行,JRuby 去除了 GIL,可以做到真正的多線程并行執(zhí)行。

延伸 · 閱讀

精彩推薦
  • RubyRuby簡潔學(xué)習(xí)筆記(一):字符串、數(shù)字、類和對象

    Ruby簡潔學(xué)習(xí)筆記(一):字符串、數(shù)字、類和對象

    這篇文章主要介紹了Ruby簡潔學(xué)習(xí)筆記(一):字符串、數(shù)字、類和對象,本文是學(xué)習(xí)筆記第一篇,需要的朋友可以參考下 ...

    腳本之家2472020-04-20
  • Ruby簡要說明Ruby中的迭代器

    簡要說明Ruby中的迭代器

    這篇文章主要介紹了Ruby中的迭代器,迭代器的概念在動態(tài)語言的編程中十分重要,文章中介紹了Ruby中的each迭代器和collect迭代器,需要的朋友可以參考下 ...

    goldensun2772020-04-25
  • RubyCentOS中配置Ruby on Rails環(huán)境

    CentOS中配置Ruby on Rails環(huán)境

    經(jīng)過一個(gè)上午的折騰,終于把ROR環(huán)境在CentOS中搞定,繞了很多彎路,把文章寫下來總結(jié)一下 ...

    可樂加糖4762020-04-12
  • Ruby剖析 Ruby 訪問控制

    剖析 Ruby 訪問控制

    前面,我們說 Ruby 沒有函數(shù),只有方法.而且實(shí)際上有不止一種方法.這一節(jié)我們介紹 訪問控制 (accesscontrols). 想想當(dāng)我們在最高層而不是在一個(gè)類的定義里定義...

    ruby教程網(wǎng)3572020-04-08
  • RubyRuby迭代器的7種技巧分享

    Ruby迭代器的7種技巧分享

    這篇文章主要介紹了Ruby迭代器的7種技巧分享,Ruby中的迭代器非常人性化,本文既是講解了7個(gè)技巧也是講解了7種迭代器,需要的朋友可以參考下 ...

    腳本之家4782020-04-20
  • RubyRuby環(huán)境下安裝使用bundler來管理多版本的gem

    Ruby環(huán)境下安裝使用bundler來管理多版本的gem

    這篇文章主要介紹了Ruby環(huán)境下安裝使用bundler來管理多版本的gem的方法,舉了Ruby On Rails中的應(yīng)用實(shí)例來進(jìn)行演示,需要的朋友可以參考下 ...

    日拱一卒4332020-05-10
  • RubyRuby設(shè)計(jì)模式編程中使用Builder建造者模式的實(shí)例

    Ruby設(shè)計(jì)模式編程中使用Builder建造者模式的實(shí)例

    這篇文章主要介紹了Ruby設(shè)計(jì)模式編程中使用Builder建造者模式的實(shí)例,建造者模式將一個(gè)復(fù)雜對象的構(gòu)造與它的表示分離,使同樣的構(gòu)建過程可以創(chuàng)建不同的表...

    范孝鵬2192020-05-07
  • RubyRuby進(jìn)行文件信息輸出實(shí)例代碼

    Ruby進(jìn)行文件信息輸出實(shí)例代碼

    Ruby進(jìn)行文件信息輸出實(shí)例代碼,數(shù)據(jù)是隨機(jī)的,所以每次的記錄都會不同。 ...

    ruby教程網(wǎng)2962020-04-10
主站蜘蛛池模板: 国产免费片 | 色污视频在线观看 | 色中色在线视频 | 日本在线观看视频网站 | 99亚洲国产精品 | 久久综合爱 | av之家在线观看 | 久久精品一区二区三 | 久久草草亚洲蜜桃臀 | www.99re14.com | 午夜一级 | 在线播放亚洲视频 | 香蕉视频1024 | 久久精品1区2区 | 久久经典免费视频 | 黄网站免费观看视频 | 成人免费在线视频 | 久久国产夫妻视频 | 99日韩精品视频 | 欧美视屏一区二区 | arabxxxxvideos| 日本a∨精品中文字幕在线 狠狠干精品视频 | 欧美一级免费高清 | 亚洲看片网 | 黄色网址进入 | 国产无遮挡一级毛片 | 久久看视频| 国产亚洲精品综合一区91 | 日韩欧美精品电影 | 精品国产一区二区三区久久久蜜 | 黄色片视频观看 | bt 自拍 另类 综合 欧美 | 欧美人与牲禽动交精品一区 | 性爱在线免费视频 | 久久色播 | 久久国产精品91 | 久久凹凸 | 欧美性受ⅹ╳╳╳黑人a性爽 | 88xx成人永久免费观看 | 亚洲午夜网站 | 亚洲一区成人在线 |