Wzorzec projektowy polecenie należy do czynnościowych wzorców projektowych. Jego celem jest umieszczenie w jednej klasie zarządzającej wszystkich poleceń, które mogą być wywołane np. przy wciśnięciu przycisku klawiatury. W tejże klasie przechowywane są interfejsy obiektów klas, które wykonują określone polecenie.
Przykład diagramu UML wzorca projektowego polecenie
Na poniższym diagramie UML klasa KeyboardCommands zawiera pole commands, które jest zdolne do przechowywania pary danych: klucz (char) → polecenie (iCommand). Klucz określa jednocześnie pod jaki przycisk klawiatury podpięty został dany interfejs polecenia.
Interfejs iCommand jest dziedziczony przez klasy, będące jednocześnie poleceniami programu. Są to klasy:
CommandLeft - wykonująca operację przemieszczenia samochodu w lewo;
CommandRight - wykonująca operację przemieszczenia samochodu w prawo;
CommandForward - wykonująca operację przemieszczenia samochodu do przodu;
CommandBack - wykonująca operację przemieszczenia samochodu do tyłu;
CommandExit - wyjście z programu
Pierwsze cztery klasy przechowują wskaźnik do obiektu klasy Car wywołując jego metody wewnętrzne odpowiedzialne za przemieszczanie się samochodu.
Przykładowa implementacja wzorca projektowego polecenie w C++
#include <iostream>
#include <string>
#include <map>
class Car{
int x;
int y;
public:
Car(const int x, const int y): x(x), y(y){};
void moveLeft(){
x--;
std::cout<<"Moved left" << std::endl;
};
void moveRight(){
x++;
std::cout<<"Moved right" << std::endl;
};
void moveForward(){
y++;
std::cout<<"Moved forward" << std::endl;
};
void moveBack(){
y--;
std::cout<<"Moved back" << std::endl;
};
void writePos(){
std::cout<<"x "<<x<<" y "<<y<<std::endl;
};
};
class iCommand{
protected:
std::string commandName;
public:
iCommand(std::string commandName): commandName(commandName){};
virtual void doCommand() = 0;
inline std::string getCommand(){return commandName;};
};
class CommandLeft : public iCommand{
Car *car;
public:
CommandLeft(Car *car):iCommand("Turn left"), car(car){};
virtual void doCommand(){
if(car != NULL){
car->moveLeft();
}
}
};
class CommandRight : public iCommand{
Car *car;
public:
CommandRight(Car *car):iCommand("Turn right"), car(car){};
virtual void doCommand(){
if(car != NULL){
car->moveRight();
}
}
};
class CommandForward : public iCommand{
Car *car;
public:
CommandForward(Car *car):iCommand("Move forward"), car(car){};
virtual void doCommand(){
if(car != NULL){
car->moveForward();
}
}
};
class CommandBack : public iCommand{
Car *car;
public:
CommandBack(Car *car):iCommand("Move back"), car(car){};
virtual void doCommand(){
if(car != NULL){
car->moveBack();
}
}
};
class CommandExit : public iCommand{
public:
CommandExit():iCommand("Exit"){};
virtual void doCommand(){
std::cout<<this->getCommand()<<std::endl;
}
};
class KeyboardCommands{
std::map<char, iCommand*> commands;
public:
bool addCommand(char key, iCommand* ic){
if(commands.find(key) == commands.end()){
commands[key] = ic;
return true;
}
return false;
};
void writeCommandsList(){
for(std::map<char, iCommand*>::iterator i = commands.begin(); i != commands.end(); i++){
std::cout << i->second->getCommand() << " ["<<i->first<<"]" << std::endl;
}
}
bool doCommand(char key){
if(commands.find(key) != commands.end()){
commands[key]->doCommand();
return true;
}
return false;
}
bool changeCommandKey(const char newKey, char &oldKey){
if(commands.find(newKey) == commands.end()){
commands[newKey] = commands[oldKey];
commands.erase(oldKey);
oldKey = newKey;
return true;
}
return false;
}
~KeyboardCommands(){
for(std::map<char, iCommand*>::iterator i = commands.begin(); i != commands.end(); i++){
if(i->second != NULL){
delete i->second;
i->second = NULL;
}
}
}
};
int main(){
std::map<char, iCommand*> commands;
Car *car = new Car(10,10);
KeyboardCommands keyboardGameMode;
char exitKey = 'e';
keyboardGameMode.addCommand('a', new CommandLeft(car));
keyboardGameMode.addCommand('d', new CommandRight(car));
keyboardGameMode.addCommand('w', new CommandForward(car));
keyboardGameMode.addCommand('s', new CommandBack(car));
keyboardGameMode.addCommand(exitKey, new CommandExit());
KeyboardCommands *keyboard = &keyboardGameMode;
car->writePos();
char key = 'w';
if(keyboard->changeCommandKey('x',exitKey)){
std::cout<<"Changed" << std::endl << std::endl;
}
do{
std::cout<<"Live or die, make your move:" << std::endl;
keyboard->writeCommandsList();
std::cin>>key;
if(keyboard->doCommand(key)){
car->writePos();
}
}while(key != exitKey);
if(car){
delete car;
car = NULL;
}
std::cin.ignore(1);
std::cin.get();
return 0;
}
Przykładowy wynik działania programu:
x 10 y 10
Changed
Live or die, make your move:
Turn left [a]
Turn right [d]
Move back [s]
Move forward [w]
Exit [x]
a
Moved left
x 9 y 10
Live or die, make your move:
Turn left [a]
Turn right [d]
Move back [s]
Move forward [w]
Exit [x]
d
Moved right
x 10 y 10
Live or die, make your move:
Turn left [a]
Turn right [d]
Move back [s]
Move forward [w]
Exit [x]
w
Moved forward
x 10 y 11
Live or die, make your move:
Turn left [a]
Turn right [d]
Move back [s]
Move forward [w]
Exit [x]
s
Moved back
x 10 y 10
Live or die, make your move:
Turn left [a]
Turn right [d]
Move back [s]
Move forward [w]
Exit [x]
x
Exit
x 10 y 10