命令模式--(1)基本认识

什么是命令模式:

将请求(命令)封装为对象,实现将“发出请求的对象”与“接收和执行这些请求的对象”分隔开来

命令模式中有几个关键的概念:

客户:发起请求的对象,开关,或者你(因为开关是由你来控制的)

命令:具体的一个命令,比如开灯命令,关灯命令

执行者:接收和执行请求的对象,比如电灯,是开灯关灯命令的接收者和执行者

调用者:通过这个对象来实现客户(开关或你)和执行者(电灯)之间的耦合

扫描二维码关注公众号,回复: 756469 查看本文章

命令模式支持撤销。对于这个demo来说,开灯这个命令对应的撤销就是关灯。同理:关灯命令对应的撤销就是开灯。

LightOnCommand.java文件可以看到这一点,我通过一个List维护着用户的开灯和关灯的命令,当用户需要撤销的时候,从取出这个List的最后一个Command,执行它的undo()就实现了撤销。当然实现后你还要记得remove它,才能实现撤销到更早的Command。

这里我用一个ImageView当做灯泡。通过切换不同的图片,来实现on和off.

假如现在我要使用命令模式做一个类似按钮打开和关闭电灯的引用。实现效果如下图:

下面是工程结构:

我将开灯和关灯这两个命令封装成对象,它们都实现了Command接口,RemoteInvoker就是调用者。

下面上代码:

Command.java:

/**
 * 这是第一步,让所有的命令对象实现相同的包含一个方法的接口
 * 
 * */
public interface Command {

	public void execute();
	public void undo();
}
 

实现这个接口:

LightOnCommand.java:(LightOffCommand.java是类似的,只不过图片是相反的,这里就不贴代码了)

public class LightOnCommand implements Command{

	private ImageView light;
	
	public LightOnCommand(ImageView light){
		
		this.light = light;
	}
	
	@Override
	public void execute() {
		
		light.setBackgroundResource(R.drawable.light_on);
	}
	
	@Override
	public void undo() {
		
		light.setBackgroundResource(R.drawable.light_off);
	}
	
	@Override
	public String toString() {

		return "On Command";
	}
	
}
 

RemoteInvoker.java:

/**
 * 这个就是遥控器类,这个类在这里还负责撤销操作
 * 
 * */
public class RemoteInvoker {

	Command slot;
	Command onCommand;
	Command offCommand;
	LinkedList<Command> commandList;
	Context context;
	
	public RemoteInvoker(Context context) {
		
		this.context = context;
		commandList = new LinkedList<Command>();
	}
	
	public void setCommand(Command onCommand, Command offCommand){
		
		this.onCommand = onCommand;
		this.offCommand = offCommand;
	}
	
	public void onButtonWasPressed(){
		
		onCommand.execute();
		commandList.add(onCommand);
	}
	
	public void offButtonWasPressed(){
		
		offCommand.execute();
		commandList.add(offCommand);
	}
	
	public void undoButtonWasPressed(){
		
		for(Command c: commandList){
			Log.e("**********:", ":"+c.toString());
		}
		
		int sizeOfCommands = commandList.size();
		if(sizeOfCommands <= 0){
			
			Toast.makeText(context, "当前已经没有要撤销的对象了", Toast.LENGTH_SHORT).show();
		}else{
			
			Command lastCommand = commandList.get(sizeOfCommands - 1);//获取最后一个命令对象,执行它的undo方法
			lastCommand.undo();
			commandList.removeLast();//删除最后一个命令
		}
		
		
	}
}
 

最后是调用者类

MainActivity.java:

public class MainActivity extends Activity {

	private RemoteInvoker remoteInvoker;
	private LightOnCommand lightOnCommand;
	private LightOffCommand lightOffCommand;
	private ImageView light;
	
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        light = (ImageView)findViewById(R.id.iv_light);//灯泡
        Button btnOn = (Button)findViewById(R.id.btn_light_on);//开
        Button btnOff = (Button)findViewById(R.id.btn_light_off);//关
        Button btnUndo = (Button)findViewById(R.id.btn_undo);//撤销
        
        //步骤1
        lightOnCommand = new LightOnCommand(light);//实例化命令
        lightOffCommand = new LightOffCommand(light);
        
        //步骤2
        remoteInvoker = new RemoteInvoker(this);
        remoteInvoker.setCommand(lightOnCommand, lightOffCommand);//设置命令
        
        btnOn.setOnClickListener(clickListener);
        btnOff.setOnClickListener(clickListener);
        btnUndo.setOnClickListener(clickListener);
    }

    private View.OnClickListener clickListener = new View.OnClickListener() {
		
		@Override
		public void onClick(View v) {
			switch (v.getId()) {
			case R.id.btn_light_on://开灯
				remoteInvoker.onButtonWasPressed();
				break;
			case R.id.btn_light_off://关灯
				remoteInvoker.offButtonWasPressed();
				break;
			case R.id.btn_undo://撤销
				remoteInvoker.undoButtonWasPressed();
				break;
			default:
				break;
			}
		}
	};
    
}
 

这样做之后,你就当你要开灯或关灯的时候,就不需要直接对电灯的实例进行操作了,而是通过RemoteInvoker对象,你只要告诉这个对象,你按下了哪个按钮,其它的事情交个这个对象去处理。那么你和电灯之间就解耦了。

还有一点好处是,你可以实现命令的撤销。

当然,命令模式还支持批量执行命令,以及批量撤销命令。比如我只要按下一个按钮,那么多个命令(打开电灯,打开空调,打开电视就都开始执行),当我按关闭按钮,那么就撤销前面所有的命令。

还有队列请求日志请求的功能,这些等到下次再结合实例讲解演示。

猜你喜欢

转载自michaelye1988.iteye.com/blog/1721439