所谓(有限)状态机?就是一个有着不同状态的黑盒子。我们可以看到它的状态,也可以改变它的状态,无论是从内部还是从外部。当然,我们希望能根据它不同的状态来做一些设置或操作。根据时机的需要,我们可能也希望能在它转换状态之前或之后做一些操作。
作者:@nixzhu
=================================
下图是一个火箭(你可能会觉得不像),上面有两个按钮,Fire 就是“点火”,Abort 就是“中止”。在点火之后,会倒计时 5 秒,如果在这 5 秒之内没有被中止,那么火箭就会发射。
这个火箭是一个 RocketView,除了背景图,上面只有两个 Button 和 一个 Label,非常简单。这个 RocketView 被放置在一个 VC 的 view 里,除了它的高宽之外,一并用 AutoLayout 约束其横向居中,以及 bottom 的位置。
初始时,只有 Fire 可以点击,当它被点击后,它自己就不能再被点击了,同时 Abort 变为可以点击。当倒计时结束,火箭真正发射时,要设置这两个按钮都不能点击,因为它们已没有作用了。这一点很容易理解:因为状态不一样了,外观(UI)自然该不一样。
假如我们现在要编写一些逻辑代码,为两个按钮增加 target-action 以便完成一些操作。我们可以在 VC 里访问到这个 RocketView,再拿到它的 Button,然后 addTarget… 就可以了。
大概类似如下:
1 2 3 4 5 6 7 8 9 10 11
| let rocketView = RocketView() rocketView.fireButton.addTarget(self, action: "fire", forControlEvents: .TouchUpInside) func fire() { }
|
但这样代码就会很散乱。一种更好的办法是在 RocketView 里就绑定好 target-action,然后用 delegate 或“闭包”来触发外部代码。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| var fireAction: (() -> Void)? fireButton.addTarget(self, action: "fire", forControlEvents: .TouchUpInside) func fire() { if let action = fireAction { action() } } let rocketView = RocketView() rocketView.fireAction = { }
|
这样写的好处是类似“改变 rocketView 的状态”这样的操作就不需要暴露在 VC 中。而且,如果“外部”的 VC 认为不需要在 fire 时“做其它事情”,那它完全可以不去设置 fireAction,非常自然。对于另外一个 abortButton,我们照样写一个内部的 target-action,以及一个 abortAction 给外部就行了。
但这样做仍然不够完美。现在有两个按钮,那就有两个“改变 rocketView 的状态”这样的操作。还因为有倒计时,它也会触发一些操作,并“改变 rocketView 的状态”,代码就更分散了。如果有更多状态,状态转换时的逻辑更加不好理清。
例如对于我们 RocketView 的状态:
当按下 fireButton 时,火箭从 Standby 状态进入 CountDown 状态,也就是发射倒计时。当倒计时结束时,火箭就会进入 Launch 状态,也就发射了。若在倒计时结束前按下 abortButton,那么火箭就会停止倒计时,回到 Standby 状态。当然了,火箭上了太空完成任务后,我们可以遥控它着陆,于是它会从 Launch 状态返回 Standby 状态,由此可以重复利用。
我们希望在代码层面做到什么样呢?制造一个集中管理状态的地方,这样分散的几个“改变内部状态”就可以集中起来管理了。同时,我们也应该为 RocketView 暴露出一个闭包,它提供“之前的状态”和“现在的状态”给外部操作,这样外面的 VC 就好利用状态信息做一些操作。比如当火箭进入 Launch 状态时,我们希望 RocketView 向上飞行,那只需用动画改变其 AutoLayout 的 bottom 约束即可。
合理利用 Swift 的 enum、属性的 willSet 和 didSet 以及闭包,我们可以写出这样的代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
| enum State: Printable { case Standby case CountDown case Launch var description: String { switch self { case .Standby: return "Standby" case .CountDown: return "CountDown" case .Launch: return "Launch" } } } var previousState: State = .Standby var currentState: State = .Standby { willSet { previousState = currentState } didSet { if let stateTransitionAction = stateTransitionAction { stateTransitionAction(previousState: previousState, currentState: currentState) } switch currentState { case .Standby: fireTimer?.invalidate() countDown = 5 countDownLabel.text = "\(currentState)" fireButton.enabled = true abortButton.enabled = false case .CountDown: countDown = 5 fireTimer = makeNewFireTimer() fireButton.enabled = false abortButton.enabled = true case .Launch: fireTimer?.invalidate() countDownLabel.text = "\(currentState)" fireButton.enabled = false abortButton.enabled = false default: break } } } var stateTransitionAction: ((previousState: State, currentState: State) -> Void)?
|
我们成功的将不同状态时“改变内部状态”的代码都集中在 currentState 的 didSet 里。
而按钮以及定时器的 Action 只需要简单地改变状态即可,看起来格外清爽:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| func fire() { currentState = .CountDown } func abort() { currentState = .Standby } func countDownToFire(timer: NSTimer) { countDown-- if (countDown == 0) { currentState = .Launch } }
|
而 VC 里只需要关心 RocketView 的飞行和着陆:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
| rocketView.stateTransitionAction = { (previousState, currentState) in println("state from \(previousState) to \(currentState)") switch (previousState, currentState) { case (.CountDown, .Launch): UIView.animateWithDuration(3.0, delay: 0, options: .CurveEaseIn, animations: { () -> Void in self.rocketViewBottomConstraint.constant = CGRectGetHeight(self.view.bounds) self.view.layoutIfNeeded() }, completion: { (finished) -> Void in self.landingButton.enabled = true }) default: if currentState == .Standby { self.landingButton.enabled = false UIView.animateWithDuration(1.0, delay: 0, options: .CurveEaseOut, animations: { () -> Void in self.rocketViewBottomConstraint.constant = 20 self.view.layoutIfNeeded() }, completion: { (finished) -> Void in }) } } }
|
由此,在 VC 里设置一个按钮做返回遥控器也就一句话的事情:
1 2 3
| @IBAction func landing(sender: UIBarButtonItem) { rocketView.currentState = .Standby }
|
最后的效果请查看并运行 Demo 代码:https://github.com/nixzhu/StateMachineDemo
小结
这是一个放在 View 内部的很简单的状态机(实际上状态机的原理就是这么简单),通过它我们能将状态转换时的各种操作集中管理,并且 VC 会变得更轻量。不同的状态机可能有不同的要求,比如有的可能需要在状态转换之前做一些操作,本例中的状态机也可以扩展。
状态机是一种对可变模型的抽象,实际上几乎没有不变的模型。从状态机的视角,我们可以站在更高层面观察问题。而且很可能你在无意中就已经在使用状态机的视角,只不过没有太明确而已。
虽然很多时候你不会面临很复杂的情况,但懂一些状态机的知识可以让你写出更易读的代码,维护起来更加轻松。而且很酷,对吧?
===============
欢迎转载,但请一定注明出处! https://github.com/nixzhu/dev-blog
欢迎转发此条 Tweet https://twitter.com/nixzhu/status/591072888003776512 或微博 http://weibo.com/2076580237/CezRcFOGY 以分享此文!
如果你认为这篇文章不错,也有闲钱,那你可以用支付宝扫描下方二维码随便捐助一点,以慰劳作者的辛苦: