模块 MonitorMixin

在并发编程中,监视器是一个对象或模块,旨在被多个线程安全地使用。监视器的定义特征是其方法以互斥方式执行。也就是说,在每个时间点,最多只能有一个线程执行其任何方法。与推理更新数据结构的并行代码相比,这种互斥性大大简化了对监视器实现的推理。

您可以在 Wikipedia 页面上阅读有关 监视器 的一般原则的更多信息。

示例

简单的 object.extend

require 'monitor.rb'

buf = []
buf.extend(MonitorMixin)
empty_cond = buf.new_cond

# consumer
Thread.start do
  loop do
    buf.synchronize do
      empty_cond.wait_while { buf.empty? }
      print buf.shift
    end
  end
end

# producer
while line = ARGF.gets
  buf.synchronize do
    buf.push(line)
    empty_cond.signal
  end
end

buf.empty? 时,消费者线程等待生产者线程将一行推送到 buf。生产者线程(主线程)从 ARGF 读取一行并将其推送到 buf,然后调用 empty_cond.signal 以通知消费者线程有新数据。

简单的 Class include

require 'monitor'

class SynchronizedArray < Array

  include MonitorMixin

  def initialize(*args)
    super(*args)
  end

  alias :old_shift :shift
  alias :old_unshift :unshift

  def shift(n=1)
    self.synchronize do
      self.old_shift(n)
    end
  end

  def unshift(item)
    self.synchronize do
      self.old_unshift(item)
    end
  end

  # other methods ...
end

SynchronizedArray 实现了一个对项目进行同步访问的 Array。这个类实现为 Array 的子类,其中包含了 MonitorMixin 模块。

公共类方法

new(...) 点击以切换源代码

使用 extend MonitorMixininclude MonitorMixin 而不是此构造函数。请查看上面的示例以了解如何使用此模块。

调用超类方法
# File monitor/lib/monitor.rb, line 222
def initialize(...)
  super
  mon_initialize
end

公共实例方法

mon_enter() 点击以切换源代码

进入互斥区。

# File monitor/lib/monitor.rb, line 169
def mon_enter
  @mon_data.enter
end
mon_exit() 点击以切换源代码

离开互斥区。

# File monitor/lib/monitor.rb, line 176
def mon_exit
  mon_check_owner
  @mon_data.exit
end
mon_locked?() 点击以切换源代码

如果此监视器被任何线程锁定,则返回 true。

# File monitor/lib/monitor.rb, line 184
def mon_locked?
  @mon_data.mon_locked?
end
mon_owned?() 点击以切换源代码

如果此监视器被当前线程锁定,则返回 true。

# File monitor/lib/monitor.rb, line 191
def mon_owned?
  @mon_data.mon_owned?
end
mon_synchronize(&b) 点击以切换源代码

进入互斥区并执行代码块。当代码块退出时,自动离开互斥区。请参阅 MonitorMixin 下的示例。

# File monitor/lib/monitor.rb, line 200
def mon_synchronize(&b)
  @mon_data.synchronize(&b)
end
也别名为:synchronize
mon_try_enter() 点击以切换源代码

尝试进入互斥区。如果锁定失败,则返回 false

# File monitor/lib/monitor.rb, line 160
def mon_try_enter
  @mon_data.try_enter
end
也别名为:try_mon_enter
new_cond() 点击以切换源代码

创建一个与 Monitor 对象关联的新 MonitorMixin::ConditionVariable

# File monitor/lib/monitor.rb, line 209
def new_cond
  unless defined?(@mon_data)
    mon_initialize
    @mon_initialized_by_new_cond = true
  end
  return ConditionVariable.new(@mon_data)
end
synchronize(&b)
别名为:mon_synchronize
try_mon_enter()

为了向后兼容

别名为:mon_try_enter

私有实例方法

mon_check_owner() 点击以切换源代码

确保 MonitorMixin 由当前线程拥有,否则引发异常。

# File monitor/lib/monitor.rb, line 243
def mon_check_owner
  @mon_data.mon_check_owner
end
mon_initialize() 点击以切换源代码

在类中包含 MonitorMixin 或使用 MonitorMixin 扩展对象后初始化 MonitorMixin

# File monitor/lib/monitor.rb, line 229
def mon_initialize
  if defined?(@mon_data)
    if defined?(@mon_initialized_by_new_cond)
      return # already initialized.
    elsif @mon_data_owner_object_id == self.object_id
      raise ThreadError, "already initialized"
    end
  end
  @mon_data = ::Monitor.new
  @mon_data_owner_object_id = self.object_id
end