Command Design Pattern

Command Design Pattern

18 Sep 2024
Intermediate
109K Views
31 min read
Learn with an interactive course and practical hands-on labs

⭐ .NET Design Patterns Course: Design Patterns in C# Online Training

Understanding the Command Design Pattern

Understanding the Command Design Pattern is a crucial concept in software design. It allows for the encapsulation of a request as an object, enabling parameterization of clients with queues, logs, and operations. This decouples the sender of the request from the receiver, promoting flexibility, reuse, and scalability, which is especially valuable in undo/redo functionalities and transactional systems.

In this design pattern tutorial, we will look at the Command Design Pattern, answering the question "What is the Command Design Pattern?". In "When should I use the Command Design Pattern?" we will also explore examples of the Command Design Pattern in action. So, let us begin by addressing the question: "What is the Command Design Pattern?"

What is Command Pattern

  • The Command Design Pattern is a behavioral design pattern that converts a request into a stand-alone object, allowing clients to be parameterized with different requests, queuing requests, and supporting irreversible activities.
  • The Command Pattern wraps a request as an object, separating the sender and recipient.
  • Commands can be parameterized, which means that you can create several commands with varying parameters without modifying the invoker.
  • It separates the sender (client or invoker) from the receiver (object that performs the operation), allowing for greater flexibility and extension.
  • By saving the state or reverse commands, the pattern provides undoable operations (an action or a series of actions that can be reversed or undone in a system).

Command Pattern - UML Diagram & Implementation

The UML class diagram for the implementation of the command design pattern is given below:

The classes, interfaces, and objects in the above UML class diagram are as follows:

1. Command Interface

  • The Command Interface functions similarly to a rulebook for all command types.
  • It declares a common function, execute(), which ensures that each concrete command understands how to do its own operation.
  • It establishes the standard for all commands, allowing the remote control to manage and execute various tasks without having to know the specifics of each command.

2. Concrete Command Classes

  • Concrete Command Classes are specific commands, such as turning on the television or altering the audio volume.
  • Each class contains the details of a certain action.
  • These classes serve as executable instructions that the remote control can activate without thinking about the specifics of how each command performs its function.

3. Invoker (Remote Control)

  • The Invoker, commonly a remote control, is in charge of beginning command execution.
  • It references a command but does not go into detail on how each command works.
  • It functions similarly to a button.
  • The remote control's purpose is to coordinate and execute commands without becoming engaged in the complexity of individual activities.

4. Receiver (Devices)

  • The Receiver is the device that knows how to carry out the actual function specified by a command.
  • It could be a television, stereo, or another device. Receivers understand the exact duties specified in commands.
  • If a command specifies "turn on," the Receiver (device) understands exactly how to carry out that operation.
  • The Receiver-Command connection divides responsibilities, making it simple to add new devices or commands without interfering with existing features.

Command Design Pattern Real-Life Example

This diagram illustrates the Command Design Pattern within a smart home automation system. Here is a breakdown of the elements:

RemoteControl Class

  • Acts as the invoker that holds a reference to command objects.
  • It triggers the execution of commands without needing to know the specifics of each action.
  • It provides a method for setting and executing commands, like turning on lights or adjusting the thermostat.

SmartDevice Interface

  • Defines the interface for all smart devices, such as lights, thermostats, or speakers.
  • It ensures that each device can respond to a command in a standardized way.
  • It specifies the operations a device should perform, such as turning on or off.

ConcreteSmartDevice A and B Classes

  • These are the specific implementations of the SmartDevice interface, like SmartLight or SmartThermostat.
  • They encapsulate the actual actions, such as switching the light on or adjusting the temperature.
  • They respond to commands initiated by the RemoteControl.

Command Interface

  • Serves as the blueprint for creating various commands.
  • It specifies the execute() method that all concrete commands must implement.
  • It ensures that each command follows the same structure, making it easier to handle diverse actions.

ConcreteCommand Classes

  • These classes represent specific commands, such as turning on a light or adjusting the thermostat.
  • Each class encapsulates the logic for executing a particular operation on the smart device.
  • The execute() method is implemented to invoke the appropriate action on the device.

Key Points

  • The Command Design Pattern is useful because it decouples the invoker (remote control) from the receiver (smart devices).
  • It makes it easy to add or modify commands without altering the RemoteControl or device classes.
  • It provides flexibility in implementing additional features like undo/redo operations or logging command history.
  • It's ideal for situations where you want to handle multiple operations without tightly coupling the requester to the device logic.
Let's explore this concept in different languages, such asC# Compiler,Java Compiler,Python Compiler,TypeScript Compiler, and JavaScript Compiler.

Example



using System;

// Command Interface
public interface ICommand
{
    void Execute();
}

// Receiver (SmartDevice Interface)
public interface ISmartDevice
{
    void TurnOn();
    void TurnOff();
}

// ConcreteSmartDevice A - Smart Light
public class SmartLight : ISmartDevice
{
    public void TurnOn()
    {
        Console.WriteLine("Smart Light is turned ON.");
    }

    public void TurnOff()
    {
        Console.WriteLine("Smart Light is turned OFF.");
    }
}

// ConcreteSmartDevice B - Smart Thermostat
public class SmartThermostat : ISmartDevice
{
    public void TurnOn()
    {
        Console.WriteLine("Thermostat is set to heating mode.");
    }

    public void TurnOff()
    {
        Console.WriteLine("Thermostat is set to cooling mode.");
    }
}

// Concrete Command for Turning On
public class TurnOnCommand : ICommand
{
    private readonly ISmartDevice _device;

    public TurnOnCommand(ISmartDevice device)
    {
        _device = device;
    }

    public void Execute()
    {
        _device.TurnOn();
    }
}

// Concrete Command for Turning Off
public class TurnOffCommand : ICommand
{
    private readonly ISmartDevice _device;

    public TurnOffCommand(ISmartDevice device)
    {
        _device = device;
    }

    public void Execute()
    {
        _device.TurnOff();
    }
}

// Invoker (Remote Control)
public class RemoteControl
{
    private ICommand _command;

    public void SetCommand(ICommand command)
    {
        _command = command;
    }

    public void ExecuteCommand()
    {
        _command.Execute();
    }
}

// Client
public class Client
{
    public static void Main(string[] args)
    {
        // Create Receivers
        ISmartDevice light = new SmartLight();
        ISmartDevice thermostat = new SmartThermostat();

        // Create Commands
        ICommand lightOnCommand = new TurnOnCommand(light);
        ICommand lightOffCommand = new TurnOffCommand(light);
        ICommand thermostatOnCommand = new TurnOnCommand(thermostat);
        ICommand thermostatOffCommand = new TurnOffCommand(thermostat);

        // Create Invoker (Remote Control)
        RemoteControl remoteControl = new RemoteControl();

        // Turn On Smart Light
        remoteControl.SetCommand(lightOnCommand);
        remoteControl.ExecuteCommand();

        // Turn Off Smart Light
        remoteControl.SetCommand(lightOffCommand);
        remoteControl.ExecuteCommand();

        // Set Thermostat to Heating Mode
        remoteControl.SetCommand(thermostatOnCommand);
        remoteControl.ExecuteCommand();

        // Set Thermostat to Cooling Mode
        remoteControl.SetCommand(thermostatOffCommand);
        remoteControl.ExecuteCommand();
    }
}
            

interface Command {
    void execute();
}

interface SmartDevice {
    void turnOn();
    void turnOff();
}

class SmartLight implements SmartDevice {
    public void turnOn() {
        System.out.println("Smart Light is turned ON.");
    }

    public void turnOff() {
        System.out.println("Smart Light is turned OFF.");
    }
}

class SmartThermostat implements SmartDevice {
    public void turnOn() {
        System.out.println("Thermostat is set to heating mode.");
    }

    public void turnOff() {
        System.out.println("Thermostat is set to cooling mode.");
    }
}

class TurnOnCommand implements Command {
    private final SmartDevice device;

    public TurnOnCommand(SmartDevice device) {
        this.device = device;
    }

    public void execute() {
        device.turnOn();
    }
}

class TurnOffCommand implements Command {
    private final SmartDevice device;

    public TurnOffCommand(SmartDevice device) {
        this.device = device;
    }

    public void execute() {
        device.turnOff();
    }
}

class RemoteControl {
    private Command command;

    public void setCommand(Command command) {
        this.command = command;
    }

    public void executeCommand() {
        command.execute();
    }
}

public class Client {
    public static void main(String[] args) {
        // Create Receivers
        SmartDevice light = new SmartLight();
        SmartDevice thermostat = new SmartThermostat();

        // Create Commands
        Command lightOnCommand = new TurnOnCommand(light);
        Command lightOffCommand = new TurnOffCommand(light);
        Command thermostatOnCommand = new TurnOnCommand(thermostat);
        Command thermostatOffCommand = new TurnOffCommand(thermostat);

        // Create Invoker (Remote Control)
        RemoteControl remoteControl = new RemoteControl();

        // Turn On Smart Light
        remoteControl.setCommand(lightOnCommand);
        remoteControl.executeCommand();

        // Turn Off Smart Light
        remoteControl.setCommand(lightOffCommand);
        remoteControl.executeCommand();

        // Set Thermostat to Heating Mode
        remoteControl.setCommand(thermostatOnCommand);
        remoteControl.executeCommand();

        // Set Thermostat to Cooling Mode
        remoteControl.setCommand(thermostatOffCommand);
        remoteControl.executeCommand();
    }
}
            

class ICommand:
    def execute(self):
        pass

class ISmartDevice:
    def turn_on(self):
        pass

    def turn_off(self):
        pass

class SmartLight(ISmartDevice):
    def turn_on(self):
        print("Smart Light is turned ON.")

    def turn_off(self):
        print("Smart Light is turned OFF.")

class SmartThermostat(ISmartDevice):
    def turn_on(self):
        print("Thermostat is set to heating mode.")

    def turn_off(self):
        print("Thermostat is set to cooling mode.")

class TurnOnCommand(ICommand):
    def __init__(self, device: ISmartDevice):
        self.device = device

    def execute(self):
        self.device.turn_on()

class TurnOffCommand(ICommand):
    def __init__(self, device: ISmartDevice):
        self.device = device

    def execute(self):
        self.device.turn_off()

class RemoteControl:
    def __init__(self):
        self.command = None

    def set_command(self, command: ICommand):
        self.command = command

    def execute_command(self):
        self.command.execute()

# Client
light = SmartLight()
thermostat = SmartThermostat()

light_on_command = TurnOnCommand(light)
light_off_command = TurnOffCommand(light)
thermostat_on_command = TurnOnCommand(thermostat)
thermostat_off_command = TurnOffCommand(thermostat)

remote_control = RemoteControl()

# Turn On Smart Light
remote_control.set_command(light_on_command)
remote_control.execute_command()

# Turn Off Smart Light
remote_control.set_command(light_off_command)
remote_control.execute_command()

# Set Thermostat to Heating Mode
remote_control.set_command(thermostat_on_command)
remote_control.execute_command()

# Set Thermostat to Cooling Mode
remote_control.set_command(thermostat_off_command)
remote_control.execute_command()
            

interface ICommand {
    execute(): void;
}

interface ISmartDevice {
    turnOn(): void;
    turnOff(): void;
}

class SmartLight implements ISmartDevice {
    turnOn() {
        console.log("Smart Light is turned ON.");
    }

    turnOff() {
        console.log("Smart Light is turned OFF.");
    }
}

class SmartThermostat implements ISmartDevice {
    turnOn() {
        console.log("Thermostat is set to heating mode.");
    }

    turnOff() {
        console.log("Thermostat is set to cooling mode.");
    }
}

class TurnOnCommand implements ICommand {
    private device: ISmartDevice;

    constructor(device: ISmartDevice) {
        this.device = device;
    }

    execute() {
        this.device.turnOn();
    }
}

class TurnOffCommand implements ICommand {
    private device: ISmartDevice;

    constructor(device: ISmartDevice) {
        this.device = device;
    }

    execute() {
        this.device.turnOff();
    }
}

class RemoteControl {
    private command: ICommand;

    setCommand(command: ICommand) {
        this.command = command;
    }

    executeCommand() {
        this.command.execute();
    }
}

// Client
const light = new SmartLight();
const thermostat = new SmartThermostat();

const lightOnCommand = new TurnOnCommand(light);
const lightOffCommand = new TurnOffCommand(light);
const thermostatOnCommand = new TurnOnCommand(thermostat);
const thermostatOffCommand = new TurnOffCommand(thermostat);

const remoteControl = new RemoteControl();

// Turn On Smart Light
remoteControl.setCommand(lightOnCommand);
remoteControl.executeCommand();

// Turn Off Smart Light
remoteControl.setCommand(lightOffCommand);
remoteControl.executeCommand();

// Set Thermostat to Heating Mode
remoteControl.setCommand(thermostatOnCommand);
remoteControl.executeCommand();

// Set Thermostat to Cooling Mode
remoteControl.setCommand(thermostatOffCommand);
remoteControl.executeCommand();
            

class Command {
    execute() {}
}

class SmartLight {
    turnOn() {
        console.log("Smart Light is turned ON.");
    }

    turnOff() {
        console.log("Smart Light is turned OFF.");
    }
}

class SmartThermostat {
    turnOn() {
        console.log("Thermostat is set to heating mode.");
    }

    turnOff() {
        console.log("Thermostat is set to cooling mode.");
    }
}

class TurnOnCommand extends Command {
    constructor(device) {
        super();
        this.device = device;
    }

    execute() {
        this.device.turnOn();
    }
}

class TurnOffCommand extends Command {
    constructor(device) {
        super();
        this.device = device;
    }

    execute() {
        this.device.turnOff();
    }
}

class RemoteControl {
    setCommand(command) {
        this.command = command;
    }

    executeCommand() {
        this.command.execute();
    }
}

// Client
const light = new SmartLight();
const thermostat = new SmartThermostat();

const lightOnCommand = new TurnOnCommand(light);
const lightOffCommand = new TurnOffCommand(light);
const thermostatOnCommand = new TurnOnCommand(thermostat);
const thermostatOffCommand = new TurnOffCommand(thermostat);

const remoteControl = new RemoteControl();

// Turn On Smart Light
remoteControl.setCommand(lightOnCommand);
remoteControl.executeCommand();

// Turn Off Smart Light
remoteControl.setCommand(lightOffCommand);
remoteControl.executeCommand();

// Set Thermostat to Heating Mode
remoteControl.setCommand(thermostatOnCommand);
remoteControl.executeCommand();

// Set Thermostat to Cooling Mode
remoteControl.setCommand(thermostatOffCommand);
remoteControl.executeCommand();
            

Output

Smart Light is turned ON.
Smart Light is turned OFF.
Thermostat is set to heating mode.
Thermostat is set to cooling mode.

Explanation

  • This code demonstrates the Command pattern with smart devices.
  • It is where the ICommand interface defines the command structure, and ISmartDevice represents devices like a smart light and thermostat.
  • The TurnOnCommand and TurnOffCommand execute actions for these devices.
  • The RemoteControl class acts as the invoker, allowing the client to execute commands to turn devices on or off.

When to use the Command Design Pattern

The Command Design Pattern should be used in these situations:
  • Undo/redo functionality:It is ideal when you need to implement undo/redo actions, allowing you to store and reverse commands easily.
  • Parameterizing actions: It is useful when you want to pass actions (commands) as parameters to other objects, enabling flexible system design.
  • Queueing operations: This is effective when commands need to be queued or logged for later execution, such as in task scheduling systems.
  • Encapsulating requests: It is helpful when you need to encapsulate a request as an object, making it easier to manage, log, or execute actions at a later time.
When not to use the Command Design Pattern
The Command Design Pattern should not be used in these situations:
  • Simple actions: It is unnecessary when actions are straightforward and do not need to be encapsulated into objects or queued for later execution.
  • Limited flexibility: It is overkill if your application does not require extensive undo/redo functionality or complex request handling.
  • Increased complexity: It is better to avoid when the pattern adds excessive complexity without significant benefits, especially for small, simple tasks.
Summary

The Command Design Pattern encapsulates requests as objects, allowing for parameterization of clients with different requests and supporting undoable operations. It promotes loose coupling between the sender and receiver of commands and is ideal for implementing queues, logging, and transactional systems. However, it may not be suitable for simple operations due to the added complexity. For a deeper understanding and practical applications of design patterns, consider enrolling in the Software Architecture and Design Certification Training offered by Scholarhat.

FAQs

The Command Design Pattern encapsulates a request as an object, allowing for parameterization, request queueing, and action recording. It separates the duty for issuing commands from that of executing them, promoting decoupling and flexibility in command processing.

The Command Pattern solves the problem of separating a request's sender and recipient, allowing for more flexible command execution, undo/redo procedures, and request logging. It facilitates request management as objects, allowing for easy parameterization, queueing, and command execution.

The Command Pattern is intended to isolate the sender of a request from the object that processes the request, allowing for more flexible and scalable command execution. It allows you to parameterize queries, perform undo/redo actions, and simplify command logging and queuing.

The Command Pattern provides undo/redo capabilities by storing commands as objects that may be run or reversed as needed, allowing activities to be undone or redone by executing or reverting commands.

The Invoker in the Command Pattern triggers the execution of commands but does not know the details of how each command operates, allowing it to manage commands and execute them without being concerned with the specifics of their implementation. 
Share Article
About Author
Shailendra Chauhan (Microsoft MVP, Founder & CEO at ScholarHat)

Shailendra Chauhan, Founder and CEO of ScholarHat by DotNetTricks, is a renowned expert in System Design, Software Architecture, Azure Cloud, .NET, Angular, React, Node.js, Microservices, DevOps, and Cross-Platform Mobile App Development. His skill set extends into emerging fields like Data Science, Python, Azure AI/ML, and Generative AI, making him a well-rounded expert who bridges traditional development frameworks with cutting-edge advancements. Recognized as a Microsoft Most Valuable Professional (MVP) for an impressive 9 consecutive years (2016–2024), he has consistently demonstrated excellence in delivering impactful solutions and inspiring learners.

Shailendra’s unique, hands-on training programs and bestselling books have empowered thousands of professionals to excel in their careers and crack tough interviews. A visionary leader, he continues to revolutionize technology education with his innovative approach.
Accept cookies & close this