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

第八章、状态模式

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.缺点

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

时间: 2024-10-24 01:04:00

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

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

第二十二章.享元模式 享元模式是结构型设计模式之中的一个.是对对象池的一种实现.就像它的名字一样,共享对象.避免反复的创建. 我们经常使用的String 就是使用了共享模式.所以String类型的对象创建后就不可改变,假设当两个String对象所包括的内容同样时,JVM仅仅创建一个String对象相应这两个不同的对象引用. 1.定义 採用一个共享来避免大量拥有同样内容对象的开销.使用享元模式可有效支持大量的细粒度对象. 2.使用场景 (1)系统中存在大量的类似对象. (2)细粒度的对象都具备较接

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

第十章.解释器模式 解释器模式是一种用的比較少的行为型模式.其提供了一种解释语言的语法或表达式的方式. 可是它的使用场景确实非常广泛,仅仅是由于我们自己非常少回去构造一个语言的文法,所以使用较少. 1.定义 给定一个语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子. (当中语言就是我们须要解释的对象,文法就是这个语言的规律,解释器就是翻译机.通过文法来翻译语言.) 2.使用场景 1.假设某个简单的语言须要解释运行并且能够将该语言中的语句表示为一个抽象的语法树时

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

第十八章.代理模式 代理模式也称托付模式,是结构型设计模式之中的一个.是应用广泛的模式之中的一个. 1.定义 为其它对象提供一种代理以控制对这个对象的訪问. 2.使用场景 当无法或不想直接訪问某个对象或訪问某个对象存在困难时能够通过一个代理对象来间接訪问,为了保证client使用的透明性.托付对象与代理对象须要实现相同的接口. 3.UML类图 (1)Subject:抽象主题类.声明真实主题与共同接口方法,该类能够是抽象类或接口. (2)RealSubject:真实主题类(被托付类).尤其运行详细

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

1.定义 将对象组合成树形结构以表示"部分-总体"的层次结构,使得用户对单个对象和组合对象的使用具有一致性. 2.使用场景 (1)表示对象的部分-总体层次结构时. (2)从一个总体中可以独立出部分模块或功能的场景. 3.UML类图 (1)Component:抽象根节点,为组合中的对象声明接口.在适当的情况下.实现全部类共同拥有接口的缺省行为. 声明一个接口用于訪问和管理Component的子节点.可在递归结构中定义一个接口,用于訪问一个父节点,并在合适的情况下实现它. (2)Compo

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

第八章.状态模式 1.定义 状态模式中的行为是由状态来决定,不同的状态下有不同的行为.当一个对象的内在状态改变时同意改变其行为,这个对象看起来像是改变了其类. 2.使用场景 1.一个对象的行为取决于它的状态,而且它必须在执行时依据状态改变它的行为. 2.代码中包括大量与对象状态有关的条件语句,比如,一个操作中含有大量的多分支语句.且这些分支依赖于该对象的状态. 3.简单实现 实现效果:首先将电视的状态分为开机与关机状态,开机时能够通过遥控器实现频道切换和调节音量,可是关机时,这些操作都会失效.

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

第十三章.备忘录模式 备忘录模式是一种行为模式,该模式用于保存对象当前的状态,并且在之后可以再次恢复到此状态,有点像是我们平常说的"后悔药". 1.定义 在不破坏封闭的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,这样,以后就可将该对象恢复到原先保存的状态. 2.使用场景 (1)需要保存一个对象在某一个时刻的状态或部分状态. (2)如果用一个接口来让其他对象得到这些状态,将会暴露对象的实现细节并破坏对象的封装性,一个对象不希望外界直接访问其内部状态,通过中间对象可以间接访

Android深度探索(卷1)HAL与驱动开发 第四章 源代码的下载和编译 读书笔记

Android深度探索(卷1)HAL与驱动开发 第四章 源代码的下载和编译 读书笔记     本章学习了使用git下载两套源代码并搭建两个开发环境.分别为Android源代码和Linux内核源代码.Android源代码中包含了HAL(即硬件抽象层) 的代码,并学习了如何搭建这两种开发环境. Android 的移植的本质就是Linux内核的移植,Linux内核的移植主要是Linux驱动的移植.而开发和测试Linux驱动就需要安装以上两个开发环境. 搭建两套环境的具体步骤如下: 一.下载编译和测试A

R实战读书笔记四

第三章 图形入门 本章概要 1 创建和保存图形 2 定义符号.线.颜色和坐标轴 3 文本标注 4 掌控图形维数 5 多幅图合在一起 本章所介绍内容概括如下. 一图胜千字,人们从视觉层更易获取和理解信息. 图形工作 R具有非常强大的绘图功能,看下面代码. > attach(mtcars) > plot(wt, mpg) > abline(lm(mpg~wt)) > title("Regression of MPG on Weight") > detach(m

JavaScript 设计模式与开发实践读书笔记 http://www.open-open.com/lib/view/open1469154727495.html

JavaScript 设计模式与开发实践读书笔记 最近利用碎片时间在 Kindle 上面阅读<JavaScript 设计模式与开发实践读书>这本书,刚开始阅读前两章内容,和大家分享下我觉得可以在项目中用的上的一些笔记. 我的 github 项目会不定时更新,有需要的同学可以移步到我的 github 中去查看源码: https://github.com/lichenbuliren/design-mode-notes 1.currying 函数柯里化 currying 又称 部分求值 .一个 cu

JAVA并发编程实战 读书笔记(二)对象的共享

<java并发编程实战>读书摘要 birdhack 2015年1月2日 对象的共享 JAVA并发编程实战读书笔记 我们已经知道了同步代码块和同步方法可以确保以原子的方式执行操作,但一种常见的误解是,认为关键之synchronized只能用于实现原子性或者确定临界区.同步还有另一个重要的方面:内存可见性. 1.可见性 为了确保多个线程之间对内存写入操作的可见性,必须使用同步机制. 在没有同步的情况下,编译器.处理器以及运行时等都可能对操作的执行顺序进行一些意想不到的调整.在缺乏足够同步的多线程程