黑马程序员技术交流社区

标题: 【上海校区】Java 设计模式(6) —— 命令模式 [打印本页]

作者: wuqiong    时间: 2018-7-16 10:00
标题: 【上海校区】Java 设计模式(6) —— 命令模式
一、命令模式
将请求、命令、动作等封装成对象,这样可以让项目使用这些对象来 参数化其他对象。使得命令的请求者和执行者解耦
二、示例
家电自动化遥控器API项目:智能遥控器上有多个开关按钮,可是实现不同家电的控制,也可对新家电进行接入拓展
如果用传统面向对象的模式,则是定义一个遥控的对象,在对象内进行每个按钮的监听,分别判断进行相应的处理
1.电灯对象,控制开关
/**
* 电灯对象
*/
public class Light {
    String loc = "";
    public Light(String loc) {
        this.loc = loc;
    }
    public void On() {
        System.out.println(loc + " On");
    }
    public void Off() {
        System.out.println(loc + " Off");
    }
}
2.收音机对象,控制开关和音量

/**
* 收音机对象
*/
public class Stereo {
    static int volume = 0;
    public void On() {
        System.out.println("Stereo On");
    }
    public void Off() {
        System.out.println("Stereo Off");
    }
    public void SetCd() {
        System.out.println("Stereo SetCd");
    }
    public void SetVol(int vol) {
        volume = vol;
        System.out.println("Stereo volume=" + volume);
    }
    public int GetVol() {
        return volume;
    }
    public void Start() {
        System.out.println("Stereo Start");
    }
}
3.遥控器对象,进行判断处理

/**
* 面向对象模式,遥控对象根据按钮的序号进行相应的处理
*/
public class TraditionControl implements Control {
    Light light;
    Stereo stereo;
    public TraditionControl(Light light, Stereo stereo) {
        this.light = light;
        this.stereo = stereo;
    }
    @Override
    public void onButton(int slot) {
        // TODO Auto-generated method stub
        switch (slot) {
        case 0:
            light.On();
            break;
        case 1:
            stereo.On();
            break;
        case 2:
            int vol = stereo.GetVol();
            if (vol < 11) {
                stereo.SetVol(++vol);
            }
            break;
        }
    }
    @Override
    public void offButton(int slot) {
        // TODO Auto-generated method stub
        switch (slot) {
        case 0:
            light.Off();
            break;
        case 1:
            stereo.Off();
            break;
        case 2:
            int vol = stereo.GetVol();
            if (vol > 0) {
                stereo.SetVol(--vol);
            }
            break;
        }
    }
    @Override
    public void undoButton() {
        // TODO Auto-generated method stub
    }
}
缺点:所有操作均在遥控对象内控制,耦合度太高,且如果新增或删减电器则会对整个控制对象进行改动,不利于维护

三、命令模式的实现

1.定义统一的命令方法接口

/**
* 命令模式,命令方法接口
*/
public interface Command {
    public void execute();
    public void undo();
}
2.创建各自不同的命令对象

/**
* 命令模式,电灯打开的命令操作
*/
public class LightOnCommand implements Command {
    private Light light;
    public LightOnCommand(Light light) {
        this.light = light;
    }
    @Override
    public void execute() {
        light.On();
    }
    @Override
    public void undo() {
        light.Off();
    }
}
/**
* 命令模式,电灯关闭的命令操作
*/
public class LightOffCommand implements Command {
    private Light light;
    public LightOffCommand(Light light)
    {
        this.light=light;
    }
    @Override
    public void execute() {
        light.Off();
    }
    @Override
    public void undo() {
        light.On();
    }
}
3.遥控器对象对各命令进行抽象处理

/**
* 命令模式,所有响应均可抽象成命令,调用时直接调用命令执行方法即可
*/
public class CommandModeControl implements Control {
    private Command[] onCommands;
    private Command[] offCommands;
    public CommandModeControl() {
        onCommands = new Command[5];
        offCommands = new Command[5];
        Command noCommand = new NoCommand();
        for (int i = 0, len = onCommands.length; i < len; i++) {
            onCommands = noCommand;
            offCommands = noCommand;
        }
    }
    public void setCommand(int slot, Command onCommand, Command offCommand) {
        onCommands[slot] = onCommand;
        offCommands[slot] = offCommand;
    }
    // 开
    @Override
    public void onButton(int slot) {
        onCommands[slot].execute();
    }
    // 关
    @Override
    public void offButton(int slot) {
        offCommands[slot].execute();
    }
    // 返回
    @Override
    public void undoButton() {
    }
}
4.新增和删减电器时传入相应的实体类进行命令的初始化

/**
* 命令模式,便于拓展
*/
public class ControlTest {
    public static void main(String[] args) {
        CommandModeControl control = new CommandModeControl();
        Light bedroomlight = new Light("BedRoom");
        Light kitchlight = new Light("Kitch");
        Stereo stereo = new Stereo();
        LightOnCommand bedroomlighton = new LightOnCommand(bedroomlight);
        LightOffCommand bedroomlightoff = new LightOffCommand(bedroomlight);
        LightOnCommand kitchlighton = new LightOnCommand(kitchlight);
        LightOffCommand kitchlightoff = new LightOffCommand(kitchlight);
        StereoOnCommand stereoOn = new StereoOnCommand(stereo);
        StereoOffCommand stereoOff = new StereoOffCommand(stereo);
        StereoAddVolCommand stereoaddvol = new StereoAddVolCommand(stereo);
        StereoSubVolCommand stereosubvol = new StereoSubVolCommand(stereo);
        control.setCommand(0, bedroomlighton, bedroomlightoff);
        control.setCommand(1, kitchlighton, kitchlightoff);
        control.setCommand(2, stereoOn, stereoOff);
        control.setCommand(3, stereoaddvol, stereosubvol);
        control.onButton(0);
        control.offButton(0);
        control.onButton(1);
        control.offButton(1);
        control.onButton(2);
        control.onButton(3);
        control.offButton(3);
        control.offButton(2);
    }
}优点:

Java设计模式所有示例代码,持续更新中


作者: 不二晨    时间: 2018-7-16 11:43
奈斯
作者: wuqiong    时间: 2018-7-17 15:15

作者: 小影姐姐    时间: 2018-7-18 10:36
牛牛牛!
作者: 吴琼老师    时间: 2018-7-18 14:23





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2