Ruby下的有限状态机:AASM

Ruby世界里有很多非常可爱的plugin供我们来使用,AASM就是其中之一,通过使用这个plugin,我们可以把我们的Entity变成一个有状态的对象。下图就是一个对象的状态变化图: 


 
(图片来自 http://coredotnet.blogspot.com/2008/03/creating-state-machine-workflows-using.html) 

状态机的概念并不陌生,我们在用面向对象的方式编写代码时,有很多情况下,需要对一个对象的状态变化进行规范化的管理。拿蓝点来说,一个文章在刚刚创建时是"新建"的状态,而从"新建"的状态必须要经过“待审核”的状态才可以转换成后续的“审核通过”或是“审核失败”等状态。 

在上述情况下,最简单的方法就是使用代码逻辑来直接设置文章的状态。所有的状态变化过程的业务逻辑正确性,全部由程序员在代码逻辑中保证。这样的编写方式下,我们不得不依靠业务代码的程序逻辑来保证状态的正确性,如果一旦程序逻辑编写错误,那么业务逻辑也将错误。 

有没有什么办法,可以让我们不要自己手工地去管理对象的状态变化,而且如果程序员错误地从一个状态转向另一个不允许的状态时,给我们一些警告信息呢?AASM就是来帮助我们完成这项工作的工具。特别是在与Rails框架结合,管理持久层的对象时,AASM非常方便。 

下面通过一个例子来展示这个模块的使用方法,我们首先创建一个rails项目: 

Bash代码   收藏代码
  1. rails invoice  


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

Bash代码   收藏代码
  1. script/generate model invoice state  


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

Bash代码   收藏代码
  1. rake db:migrate  


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

Ruby代码   收藏代码
  1. require 'AASM'  
  2. require 'ActiveRecord'  
  3. class Invoice < ActiveRecord::Base  
  4. include AASM  
  5.   
  6. aasm_column :state  
  7. aasm_initial_state :pending  
  8. aasm_state :pending  
  9. aasm_state :viewed:enter => :view_invoice  
  10. aasm_state :printed:enter => :print_invoice  
  11.   
  12. aasm_event :view do  
  13. transitions :to => :viewed:from => [:pending]  
  14. end  
  15.   
  16. aasm_event :print do  
  17. transitions :to => :printed:from => [:viewed:pending]  
  18. end  
  19.   
  20. aasm_event :reset do  
  21. transitions :to => :pending:from => [:viewed:printed]  
  22. end  
  23.   
  24. def view_invoice  
  25. puts "viewing invoice..."  
  26. end  
  27.   
  28. def print_invoice  
  29. puts "printing invoice..."  
  30. end  
  31.   
  32. end  


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

Bash代码   收藏代码
  1.  script/console  
  2. Loading development environment (Rails 2.3.2)  
  3. >> i = Invoice.new  
  4. => #Invoice id: nil, state: nil, created_at: nil, updated_at: nil  
  5. >> i.view  
  6. viewing invoice...  
  7. => true  
  8. >> i  
  9. => #Invoice id: nil, state: "viewed", created_at: nil, updated_at: nil  
  10. >> i.print  
  11. printing invoice...  
  12. => true  
  13. >> i.reset  
  14. => true  
  15. >> i  
  16. => #Invoice id: nil, state: "pending", created_at: nil, updated_at: nil  
  17. >>   


从上面的示例中可以看到,状态在随着预设的情况进行变化。 

AASM的源代码位于github: 

https://github.com/rubyist/aasm
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值