使用watir-webdriver打开多个线程会导致“连接被拒绝”错误

我有这个简单的例子:

require 'watir-webdriver' arr = [] sites = [ "www.google.com", "www.bbc.com", "www.cnn.com", "www.gmail.com" ] sites.each do |site| arr << Thread.new { b = Watir::Browser.new :chrome b.goto site puts b.url b.close } end arr.each {|t| t.join} 

每次我运行这个脚本,我都会

 ruby/2.1.0/net/http.rb:879:in `initialize': Connection refused - connect(2) for "127.0.0.1" port 9517 (Errno::ECONNREFUSED) 

或者其中一个浏览器在至少其中一个线程上意外关闭。

另一方面,如果我在每个循环周期结束时设置sleep 2 ,一切运行顺利! 知道为什么会这样吗?

必须与理解线程如何工作有关…

您基本上是在浏览器实例之间创建竞争条件以连接到开放端口watir-webdriver正在查找。 在这种情况下,您的第一个浏览器实例看到端口9517已打开并连接到它。 因为您并行启动这些实例,所以您的第二个实例也认为端口9517已打开并尝试连接。 但是oops,第一个浏览器实例已经在使用该端口。 这就是你得到这个特殊错误的原因。

这也解释了为什么sleep 2解决了这个问题。 第一个浏览器实例连接到端口9517,睡眠使第二个浏览器实例看到9517被占用。 然后它在端口9518上连接。

编辑

您可以看到如何使用Selenium::WebDriver::Chrome::Service#initialize ( 此处 )实现它,它调用Selenium::WebDriver::PortProber ( 此处 )。 PortProber是webdriver如何确定打开哪个端口。