Ruby threads and variable

前端 未结 3 947
抹茶落季
抹茶落季 2021-01-05 15:58

Why the result is not from 1 to 10, but 10s only?

require \'thread\'

def run(i)
  puts i
end

while true
  for i in 0..10
    Thread.new{ run(i)}
  end
  sl         


        
相关标签:
3条回答
  • 2021-01-05 16:39

    The block that is passed to Thread.new may actually begin at some point in the future, and by that time the value of i may have changed. In your case, they all have incremented up to 10 prior to when all the threads actually run.

    To fix this, use the form of Thread.new that accepts a parameter, in addition to the block:

    require 'thread'
    
    def run(i)
      puts i
    end
    
    while true
      for i in 0..10
        Thread.new(i) { |j| run(j) }
      end
      sleep(100)
    end
    

    This sets the block variable j to the value of i at the time new was called.

    0 讨论(0)
  • 2021-01-05 16:39

    @DavidGrayson is right.

    You can see here a side effect in for loop. In your case i variable scope is whole your file. While you are expecting only a block in your for loop as a scope. Actually this is wrong approach in idiomatic Ruby. Ruby gives you iterators for this job.

    (1..10).each do |i|
       Thread.new{ run(i)}
    end
    

    In this case scope of variable i will be isolated in block scope what means for each iteration you will get new local (for this block) variable i.

    0 讨论(0)
  • 2021-01-05 16:41

    The problem is that you have created 11 threads that are all trying to access the same variable i which was defined by the main thread of your program. One trick to avoid that is to call Thread.new inside a method; then the variable i that the thread has access to is just the particular i that was passed to the method, and it is not shared with other threads. This takes advantage of a closure.

    require 'thread'
    
    def run(i)
      puts i
    end
    
    def start_thread(i)
      Thread.new { run i }
    end
    
    for i in 0..10
      start_thread i
      sleep 0.1
    end
    

    Result:

    0
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    

    (I added the sleep just to guarantee that the threads run in numerical order so we can have tidy output, but you could take it out and still have a valid program where each thread gets the correct argument.)

    0 讨论(0)
提交回复
热议问题