Multithreading is the ability to execute code on multiple concurrent threads. Each thread exists within a process, and each process can have at least one thread. Multithreading allows you to speed up your program but creates additional problems that do not occur in single-threaded programs. If you use multiple threads in your app, you must ensure your code is thread-safe. Below are examples prepared by experts from iRonin, a top Ruby on Rails development company.
Let's start with this simple program.
# program.rb
class Program
def self.call; new.call; end
def call
number = 1
100.times { number += number }
number
end
end
# program_spec.rb
require './program'
describe Program do
it { expect(described_class.call).to eq(1267650600228229401496703205376) }
end
$ rspec program_spec.rb
.
Finished in 0.02662 seconds (files took 1.12 seconds to load)
1 example, 0 failures
Success! Our program passed the test. Now, let's see what the result will be with two threads. To create a new thread, we will use the Thread class, which is an abstraction for an operating system thread. When you create a Thread, you must pass the code block executed inside the thread. We have to call the join method on each thread to ensure that the main thread will wait for our threads to finish.
# program.rb
class Program
def self.call; new.call; end
def call
number = 1
2.times.map do
Thread.new { 500.times { number += number }}
end.each(&:join)
number
end
end
$ rspec program_spec.rb
.
Finished in 0.02662 seconds (files took 1.12 seconds to load)
1 example, 0 failures
$ rspec program_spec.rb
F
Failures:
1) Program should eq 1267650600228229401496703205376
Failure/Error: it { expect(described_class.call).to eq(1267650600228229401496703205376) }
expected: 1267650600228229401496703205376
got: 14855280471424563298789490688
(compared using ==)
# ./program_spec.rb:4:in `block in (root)'
Finished in 0.04002 seconds (files took 1.12 seconds to load)
1 example, 1 failure
Failed examples:
rspec ./program_spec.rb:4 # Program should eq 1267650600228229401496703205376
Oops, that's not what we expected! What's going on? What you see above happened because, unlike with MRI (Matz's Ruby Interpreter), there is no GIL (Global Interpreter Lock) mechanism in JRuby. The GIL ensures that two threads cannot be executed at the same time. That means threads on MRI won't run in parallel - MRI just switches between threads, giving each some CPU time. To fix our code so it works with JRuby, we can use the Mutex locking system, which will allow synchronization access to specific code sections.
# program.rb
class Program
def self.call; new.call; end
def call
number = 1
mutex = Mutex.new
2.times.map do
Thread.new do
500.times { mutex.synchronize { number += number }}
end
end.each(&:join)
number
end
end
$ rspec program_spec.rb
.
Finished in 0.02662 seconds (files took 1.12 seconds to load)
1 example, 0 failures
# program.rb
class Program
def self.call; new.call; end
def call
number = 1
2.times.map do
Thread.new do
50.times do
number_value = number
sleep(0.001)
number += number_value
end
end
end.each(&:join)
number
end
end
$ rspec program_spec.rb
F
Failures:
1) Program should eq 1267650600228229401496703205376
Failure/Error: it { expect(described_class.call).to eq(1267650600228229401496703205376) }
expected: 1267650600228229401496703205376
got: 1525719740468429782208
(compared using ==)
# ./program_spec.rb:4:in `block (2 levels) in <top (required)>'
Finished in 0.07984 seconds (files took 0.28282 seconds to load)
1 example, 1 failure
Failed examples:
rspec ./program_spec.rb:4 # Program should eq 1267650600228229401496703205376
# program.rb
class Program
def self.call; new.call; end
def call
number = 1
mutex = Mutex.new
2.times.map do |i|
Thread.new do
50.times do
mutex.synchronize do
number_value = number
sleep(0.001)
number += number_value
end
end
end
end.each(&:join)
number
end
end