第八章、状态模式

1.定义

状态模式中的行为是由状态来决定,不同的状态下有不同的行为。当一个对象的内在状态改变时同意改变其行为,这个对象看起来像是改变了其类。

2.使用场景

1.一个对象的行为取决于它的状态,而且它必须在执行时依据状态改变它的行为。

2.代码中包括大量与对象状态有关的条件语句,比如,一个操作中含有大量的多分支语句。且这些分支依赖于该对象的状态。

3.简单实现

实现效果:首先将电视的状态分为开机与关机状态,开机时能够通过遥控器实现频道切换和调节音量,可是关机时,这些操作都会失效。

首先是普通的实现方法:

public class TVController {
//开机状态
private final static int POWER_ON = 1;
//关机状态
private final static int POWER_OFF = 2;
//默认状态
private int mState = POWER_OFF; public void powerOn(){
if(mState ==POWER_OFF){
System.out.println("电视开机了");
}
mState = POWER_ON;
} public void powerOff(){
if(mState ==POWER_ON){
System.out.println("电视关机了");
}
mState = POWER_OFF;
} public void nextChannel(){
if(mState ==POWER_ON){
System.out.println("下一频道");
}else{
System.out.println("没有开机");
}
} public void prevChannel(){
if(mState ==POWER_ON){
System.out.println("上一频道");
}else{
System.out.println("没有开机");
}
} public void turnUp(){
if(mState ==POWER_ON){
System.out.println("调高音量");
}else{
System.out.println("没有开机");
}
} public void turnDown(){
if(mState ==POWER_ON){
System.out.println("调低音量");
}else{
System.out.println("没有开机");
}
}
}

能够看到,每次执行通过推断当前状态来进行操作。部分的代码反复。如果状态和功能添加,就会越来越难以维护。这时能够使用状态模式。例如以下:

电视的操作

/**
* 电视状态接口,定义了电视的操作函数
*
* */
public interface TVState { public void nextChannel();
public void prevChannel();
public void turnUp();
public void turnDown(); }

关机状态

/**
*
* 关机状态,操作无结果
*
* */
public class PowerOffState implements TVState{ @Override
public void nextChannel() { } @Override
public void prevChannel() { } @Override
public void turnUp() { } @Override
public void turnDown() { }
}

开机状态

/**
*
* 开机状态。操作有效
*
* */
public class PowerOnState implements TVState{ @Override
public void nextChannel() {
System.out.println("下一频道");
} @Override
public void prevChannel() {
System.out.println("上一频道");
} @Override
public void turnUp() {
System.out.println("调高音量");
} @Override
public void turnDown() {
System.out.println("调低音量");
}
}

电源操作接口

/**
* 电源操作接口
*
* */
public interface PowerController {
public void powerOn(); public void powerOff();
}

电视遥控器

/**
* 电视遥控器
*
* */
public class TVController implements PowerController{ TVState mTVState; public void setTVState(TVState mTVState){
this.mTVState = mTVState;
} @Override
public void powerOn() {
setTVState(new PowerOnState());
System.out.println("开机了");
} @Override
public void powerOff() {
setTVState(new PowerOffState());
System.out.println("关机了");
} public void nextChannel(){
mTVState.nextChannel();
} public void prevChannel(){
mTVState.prevChannel();
} public void turnUp(){
mTVState.turnUp();
} public void turnDown(){
mTVState.turnDown();
}
}

调用:

public class Client {
public static void main(String[] args) {
TVController tvController = new TVController();
//设置开机状态
tvController.powerOn();
//下一频道
tvController.nextChannel();
//调高音量
tvController.turnUp();
//关机
tvController.powerOff();
//调低音量,此时不会生效
tvController.turnDown();
}
}

结果

开机了
下一频道
调高音量
关机了

能够看出。状态模式将这些行为封装到状态类中。在进行操作时将这些功能转发给状态对象,不同的状态有不同的实现,去除了反复了if-else语句,这正是状态模式的精髓所在。

4.与策略模式的差别

状态模式与策略模式的结构差点儿是一样的,就像是孪生兄弟。可是他们的目地、本质不一样。

状态模式的行为是平行的、不可替换的。策略模式的行为是彼此独立的、可相互替换的。状态模式,一般是自我控制状态的改变。而策略模式,是由外部指定使用什么样的策略。

5.Android实战中的使用

1.登录系统,依据用户是否登录。推断事件的处理方式。

2.Wi-Fi管理,在不同的状态下,WiFi的扫描请求处理不一。

6.总结

1.长处

将全部与一个特定的状态相关的行为都放入一个状态对象中,它提供了一个更好的方法来组织与特定状态相关的代码。将繁琐的状态推断转换成结构清晰的状态类族。在避免代码膨胀的同一时候也保证了可扩展性与可维护性。

2.缺点

状态模式的使用必定会添加系统类和对象的个数。

《Android源代码设计模式解析与实战》读书笔记(八)的更多相关文章

  1. 《Android源代码设计模式解析与实战》读书笔记(十七)

    第十七章.中介者模式 中介者模式也称为调解者模式或调停者模式,是一种行为型模式. 1.定义 中介者模式包装了一系列对象相互作用的方式.使得这些对象不必相互明显作用.从而使它们能够松散耦合.当某些对象之 ...

  2. 《Android源代码设计模式解析与实战》读书笔记(十四)

    第十四章.迭代器模式 迭代器模式,又叫做游标模式.是行为型设计模式之中的一个.我们知道对容器对象的訪问必定会涉及遍历算法.我们能够将遍历的方法封装在容器中,或者不提供遍历方法,让使用容器的人自己去实现 ...

  3. 《Android源代码设计模式解析与实战》读书笔记(十)

    第十章.解释器模式 解释器模式是一种用的比較少的行为型模式.其提供了一种解释语言的语法或表达式的方式. 可是它的使用场景确实非常广泛,仅仅是由于我们自己非常少回去构造一个语言的文法,所以使用较少. 1 ...

  4. 《Android源代码设计模式解析与实战》读书笔记(二十)

    第二十章.适配器模式 适配器模式是结构型设计模式之中的一个,它在我们的开发中使用率极高,比方ListView.GridView以及RecyclerView都须要使用Adapter. 1.定义 适配器模 ...

  5. 《Android源代码设计模式解析与实战》读书笔记(十八)

    第十八章.代理模式 代理模式也称托付模式,是结构型设计模式之中的一个.是应用广泛的模式之中的一个. 1.定义 为其它对象提供一种代理以控制对这个对象的訪问. 2.使用场景 当无法或不想直接訪问某个对象 ...

  6. 《Android源代码设计模式解析与实战》读书笔记

    1.定义 将对象组合成树形结构以表示"部分-总体"的层次结构,使得用户对单个对象和组合对象的使用具有一致性. 2.使用场景 (1)表示对象的部分-总体层次结构时. (2)从一个总体 ...

  7. 《Android源代码设计模式解析》读书笔记——Android中你应该知道的设计模式

    断断续续的,<Android源代码设计模式解析>也看了一遍.书中提到了非常多的设计模式.可是有部分在开发中见到的几率非常小,所以掌握不了也没有太大影响. 我认为这本书的最大价值有两点,一个 ...

  8. R语言实战读书笔记(八)回归

    简单线性:用一个量化验的解释变量预测一个量化的响应变量 多项式:用一个量化的解决变量预测一个量化的响应变量,模型的关系是n阶多项式 多元线性:用两个或多个量化的解释变量预测一个量化的响应变量 多变量: ...

  9. iPhone与iPad开发实战读书笔记

    iPhone开发一些读书笔记 手机应用分类1.教育工具2.生活工具3.社交应用4.定位工具5.游戏6.报纸和杂志的阅读器7.移动办公应用8.财经工具9.手机购物应用10.风景区相关应用11.旅游相关的 ...

随机推荐

  1. var的变量提升的底层原理是什么?

    原理:JS引擎的工作方式是①先解析代码,获取所有被声明的变量:②然后在运行.也就是专业来说是分为预处理和执行两个阶段. 变量提升的定义:所有变量的声明语句都会被提升到代码头部,这就是变量提升. 例如: ...

  2. 取消VS2017窗口置顶

    今天打开VS2017,莫名其妙窗口置顶了,百度了一下如何取消窗口置顶,就是Ctrl+Alt+Esc组合键,就可以取消窗口置顶了,至于到底怎么会突然置顶的我也不知道emmm... /********** ...

  3. 5.7 Liquibase:与具体数据库独立的追踪、管理和应用数据库Scheme变化的工具。-mybatis-generator将数据库表反向生成对应的实体类及基于mybatis的mapper接口和xml映射文件(类似代码生成器)

    一. liquibase 使用说明 功能概述:通过xml文件规范化维护数据库表结构及初始化数据. 1.配置不同环境下的数据库信息 (1)创建不同环境的数据库. (2)在resource/liquiba ...

  4. SQL SERVER 2000 如何提高大数据筛选GROUP BY 的效率

    数据库有83W条记录,本想计算20180101之后的每天赔付情况,故写了以下SQL语句: SELECT 起保时间,sum(赔付金额) as 日赔付 FROM maindata WHERE 起保时间&g ...

  5. Android App 开机启动画面和开机自动启动APP程序设置

    1.当前比较成熟一点的应用基本上都会在进入应用之显示一个启动界面 如腾讯微博 2.准备元素  需要开机启动的图片一张 3.新建Activity AlphaAnimation动画:控制对象alpha水平 ...

  6. mysql Workbench新建数据库连接时提示SSl not enabled的问题

    在使用workbench时会弹出以下的对话框,不过这不影响使用,点击ok之后对话框关闭,接着直接点击close就行了,之后会显示已经连接上了数据库.具体ssl的方面还没研究,等研究了再贴

  7. [转]VIM字符替换

    语法为 :[addr]s/源字符串/目的字符串/[option] 全局替换命令为::%s/源字符串/目的字符串/g [addr] 表示检索范围,省略时表示当前行. 如:"1,20" ...

  8. Pjax无刷新跳转页面实现,支持超链接与表单提交

    什么是pjax? 当你点击一个站内的链接的时候,不是做页面跳转,而是只是站内页面刷新.这样的用户体验,比起整个页面都闪一下来说, 好很多. 其中有一个很重要的组成部分, 这些网站的ajax刷新是支持浏 ...

  9. react功能实现-数组遍历渲染

    在react中如何将一个数组遍历,并且逐个渲染在页面上? 1.在jsx渲染中,如果这个变量是一个数组,则会展开这个数组的所有成员. var arr = [ <h1>Hello world! ...

  10. 【剑指Offer】11、二进制中1的个数

      题目描述:   输入一个整数,输出该数二进制表示中1的个数.其中负数用补码表示.   解题思路:   本题有以下两个解决方案:   (1)依次判断每一位.判断的方法是先与1相与,为1则说明该位为1 ...