Rails应用程序中的自定义多线程服务

时间:2013-10-14 12:49:51

标签: ruby-on-rails ruby multithreading

我正在开发RoR应用程序,我需要创建一些应该与我的基础和控制器一起使用的后台服务。我有一个带线程的类,必须在队列中做一些工作。所以我们有:

class ServerThread

   def initialize(name)
       @name = name
       @queue = Queue.new
       @thread = Thread.new do
          while true
              if @queue.empty?
                  Thread.stop
              end
              item = @queue.pop
              log = `run #{item}` 
              save_log(log)    #save into DB
          end    
       end
   end

   def add_to_queue(item)
       @queue << item
       @thread.run
   end

end

现在它的工作方式如下。我只是在某个文件中创建全局变量:

$threads = SomeService.new
Server.all.each do |server|                    #servers from DB
    $threads << ServerThread.new(server.name)
end

有时浏览器用户会通过控制器将项目添加到队列中:

class ServerController < ApplicationController

   def add_to_queue       
       $threads.get_thread_by_name(params['server']).add_to_queue(params['item'])       
       render :nothing => true
   end    

end

我有一些用户将项目添加到队列中,我需要监视我的应用程序上的所有线程。 当我启动我的rails应用程序时,我需要创建这个$ threads,这个线程对于所有浏览器用户都应该是通用的。现在我尝试使用apache2和passenger来部署我的应用程序,所以这个全局变量不起作用!

如果没有全局变量怎么做?

RoR 3.2,ruby 1.9.2

1 个答案:

答案 0 :(得分:0)

像@Sergio已经说过了。不要从头开始写这样的东西是个好主意。可靠地编写并发程序并不是最简单的任务之一,也很难调试。红宝石中有一些非常好的替代品。看看ruby-toolbox:https://www.ruby-toolbox.com/categories/Background_Jobs

如果你想要自己做,那么你必须用可以由同一甚至不同机器上的多个进程共享的东西替换你的全局变量。我认为这个问题的一个非常简单的解决方案是使用redis的排队功能并在此基础上实现后台工作程序。