状态模式

August 06, 2019

状态模式「State Pattern」:允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类。状态模式的关键是区分事物内部的状态,事物内部状态的改变往往会带来事物的行为改变

模式动机

在很多情况下,一个对象的行为取决于一个或多个动态变化的属性,这样的属性叫做状态,这样的对象叫做有状态的「stateful」对象,这样的对象状态是从事先定义好的一系列值中取出的。当一个这样的对象与外部事件产生互动时,其内部状态就会改变,从而使得系统的行为也随之发生变化。

实现

上传文件时,文件会有「未上传」,「正在上传」,「暂停」,「上传成功」,「上传失败」五种状态,对于这五种状态,我们点击上传按钮时对应的表现形式是不同的

// 定义好五种状态点击时的表现形式
const FMS = {
  unUpload: {
    onClick: function() {
      console.info('开始上传')
      this.state = FMS.uploading
    },
  },
  uploading: {
    onClick: function() {
      console.info('暂停上传')
      this.state = FMS.pause
    },
  },
  pause: {
    onClick: function() {
      console.info('继续上传')
      this.state = FMS.uploading
    },
  },
  done: {
    onClick: function() {
      console.info('已上传成功,点击无效')
    },
  },
  failed: {
    onClick: function() {
      console.info('重新上传')
      this.state.uploading
    },
  },
}

function File() {
  this.state = FMS.pause // 初始状态为未上传
}

const file = new File()
// 将 file 的 this 传进去,改状态时需要
button.onclick = function() {
  file.state.onClick.call(file)
}

此时如果你说文件还需要进行删除操作,那只需要在 FMS 中添加 del 的行为,并再用一个删除按钮点击时调用即可

//...
done: {
  onClick:...
  del:function() {
    console.info('删除成功')
  }
}
delButton.onclick = function() {
  file.state.del.call(file)
}

小结

优点

状态模式定义了状态与行为之间的关系,并将它们封装在一个类里。通过增加新的状态类,很容易增加新的状态和转换。状态切换的逻辑也都被分到了状态类中,省去了很多的条件分支,请求动作与状态类中的封装行为非常清晰而且互不影响

缺点

缺点就是会在系统中定义许多状态类,而且由于逻辑分散在状态类中,我们无法一个地方就看出整个状态机的逻辑。