Ruby下的有限状态机:AASM

Ruby世界里有很多非常可爱的plugin供我们来使用,AASM就是其中之一,通过使用这个plugin,我们可以把我们的Entity变成一个有状态的对象。下图就是一个对象的状态变化图:(图片来自 http://coredotnet.blogspot.com/2008/03/creating-state-machine-workflows-using.html)状态机的概念并不陌生,我们在用面向对象的方式编写代码时,有很多情况下,需要对一个对象的状态变化进行规范化的管理。拿蓝点来说,一个文章在刚刚创建时是”新建”的状态,而从”新建”的状态必须要经过“待审核”的状态才可以转换成后续的“审核通过”或是“审核失败”等状态。在上述情况下,最简单的方法就是使用代码逻辑来直接设置文章的状态。所有的状态变化过程的业务逻辑正确性,全部由程序员在代码逻辑中保证。这样的编写方式下,我们不得不依靠业务代码的程序逻辑来保证状态的正确性,如果一旦程序逻辑编写错误,那么业务逻辑也将错误。有没有什么办法,可以让我们不要自己手工地去管理对象的状态变化,而且如果程序员错误地从一个状态转向另一个不允许的状态时,给我们一些警告信息呢?AASM就是来帮助我们完成这项工作的工具。特别是在与Rails框架结合,管理持久层的对象时,AASM非常方便。下面通过一个例子来展示这个模块的使用方法,我们首先创建一个rails项目:

rails invoice

然后进入项目目录,创建一个Model:

script/generate model invoice state

这个Model叫做Invoice,并具有一个状态字段,叫做State。接下来持久化这个类:

rake db:migrate

最后,我们要在Invoice中添加AASM的标记,使它具备业务要求的状态:

require 'AASM'require 'ActiveRecord'class Invoice < ActiveRecord::Baseinclude AASMaasm_column :stateaasm_initial_state :pendingaasm_state :pendingaasm_state :viewed, :enter => :view_invoiceaasm_state :printed, :enter => :print_invoiceaasm_event :view dotransitions :to => :viewed, :from => [:pending]endaasm_event :print dotransitions :to => :printed, :from => [:viewed, :pending]endaasm_event :reset dotransitions :to => :pending, :from => [:viewed, :printed]enddef view_invoiceputs "viewing invoice..."enddef print_invoiceputs "printing invoice..."endend

如上面的代码所示,通过AASM,我们可以控制Invoice的状态定义,和状态变化的流程。我们玩玩看这个代码:

 script/consoleLoading development environment (Rails 2.3.2)>> i = Invoice.new=> #Invoice id: nil, state: nil, created_at: nil, updated_at: nil>> i.viewviewing invoice...=> true>> i=> #Invoice id: nil, state: "viewed", created_at: nil, updated_at: nil>> i.printprinting invoice...=> true>> i.reset=> true>> i=> #Invoice id: nil, state: "pending", created_at: nil, updated_at: nil>> 

从上面的示例中可以看到,状态在随着预设的情况进行变化。AASM的源代码位于github:https://github.com/rubyist/aasm 大小: 36.4 KB 查看图片附件 陪我们走过一段别人无法替代的记忆。

Ruby下的有限状态机:AASM

相关文章:

你感兴趣的文章:

标签云: