Um dos meus padrões favoritos é o padrão de design de estado. Responda ou se comporte de maneira diferente para o mesmo conjunto de entradas fornecido.
Um dos problemas com o uso de instruções switch / case para máquinas de estado é que à medida que você cria mais estados, o switch / cases torna-se mais difícil / difícil de ler / manter, promove código espaguete desorganizado e cada vez mais difícil de alterar sem quebrar algo. Acho que usar padrões de projeto me ajuda a organizar melhor meus dados, o que é todo o ponto de abstração. Em vez de projetar seu código de estado em torno de qual estado você veio, em vez disso, estruture seu código para que ele registre o estado quando você entrar em um novo estado. Dessa forma, você obtém efetivamente um registro de seu estado anterior. Gosto da resposta de @JoshPetit e levei sua solução um passo adiante, tirada diretamente do livro GoF:
stateCtxt.h:
#define STATE (void *)
typedef enum fsmSignal
{
eEnter =0,
eNormal,
eExit
}FsmSignalT;
typedef struct fsm
{
FsmSignalT signal;
// StateT is an enum that you can define any which way you want
StateT currentState;
}FsmT;
extern int STATECTXT_Init(void);
/* optionally allow client context to set the target state */
extern STATECTXT_Set(StateT stateID);
extern void STATECTXT_Handle(void *pvEvent);
stateCtxt.c:
#include "stateCtxt.h"
#include "statehandlers.h"
typedef STATE (*pfnStateT)(FsmSignalT signal, void *pvEvent);
static FsmT fsm;
static pfnStateT UsbState ;
int STATECTXT_Init(void)
{
UsbState = State1;
fsm.signal = eEnter;
// use an enum for better maintainability
fsm.currentState = '1';
(*UsbState)( &fsm, pvEvent);
return 0;
}
static void ChangeState( FsmT *pFsm, pfnStateT targetState )
{
// Check to see if the state has changed
if (targetState != NULL)
{
// Call current state's exit event
pFsm->signal = eExit;
STATE dummyState = (*UsbState)( pFsm, pvEvent);
// Update the State Machine structure
UsbState = targetState ;
// Call the new state's enter event
pFsm->signal = eEnter;
dummyState = (*UsbState)( pFsm, pvEvent);
}
}
void STATECTXT_Handle(void *pvEvent)
{
pfnStateT newState;
if (UsbState != NULL)
{
fsm.signal = eNormal;
newState = (*UsbState)( &fsm, pvEvent );
ChangeState( &fsm, newState );
}
}
void STATECTXT_Set(StateT stateID)
{
prevState = UsbState;
switch (stateID)
{
case '1':
ChangeState( State1 );
break;
case '2':
ChangeState( State2);
break;
case '3':
ChangeState( State3);
break;
}
}
statehandlers.h:
/* define state handlers */
extern STATE State1(void);
extern STATE State2(void);
extern STATE State3(void);
statehandlers.c:
#include "stateCtxt.h:"
/* Define behaviour to given set of inputs */
STATE State1(FsmT *fsm, void *pvEvent)
{
STATE nextState;
/* do some state specific behaviours
* here
*/
/* fsm->currentState currently contains the previous state
* just before it gets updated, so you can implement behaviours
* which depend on previous state here
*/
fsm->currentState = '1';
/* Now, specify the next state
* to transition to, or return null if you're still waiting for
* more stuff to process.
*/
switch (fsm->signal)
{
case eEnter:
nextState = State2;
break;
case eNormal:
nextState = null;
break;
case eExit:
nextState = State2;
break;
}
return nextState;
}
STATE State3(FsmT *fsm, void *pvEvent)
{
/* do some state specific behaviours
* here
*/
fsm->currentState = '2';
/* Now, specify the next state
* to transition to
*/
return State1;
}
STATE State2(FsmT *fsm, void *pvEvent)
{
/* do some state specific behaviours
* here
*/
fsm->currentState = '3';
/* Now, specify the next state
* to transition to
*/
return State3;
}
Para a maioria das máquinas de estado, esp. Máquinas de estados finitos, cada estado saberá qual deve ser seu próximo estado e os critérios para fazer a transição para seu próximo estado. Para designs de estado flexível, esse pode não ser o caso, daí a opção de expor a API para estados de transição. Se você deseja mais abstração, cada manipulador de estado pode ser separado em seu próprio arquivo, que são equivalentes aos manipuladores de estado concreto no livro GoF. Se o seu design for simples com apenas alguns estados, stateCtxt.c e statehandlers.c podem ser combinados em um único arquivo para simplificar.