Java/Collections Data Structure/State Machine
The representation of a finite state machine
/*
* JBoss, Home of Professional Open Source
* Copyright 2005, JBoss Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
/**
* The representation of a finite state machine.
*
* @author Scott.Stark@jboss.org
* @version $Revision: 2787 $
*/
@SuppressWarnings("unchecked")
public class StateMachine implements Cloneable {
/** A description of the state machine */
private String description;
/** The set of states making up the state machine */
private HashSet states;
/** The starting state */
private State startState;
/** The current state of the state machine */
private State currentState;
/**
* Create a state machine given its states and start state.
*
* @param states -
* Set<State> for the state machine
* @param startState -
* the starting state
*/
public StateMachine(Set states, State startState) {
this(states, startState, null);
}
/**
* Create a state machine given its states and start state.
*
* @param states -
* Set<State> for the state machine
* @param startState -
* the starting state
* @param description -
* an optional description of the state machine
*/
public StateMachine(Set states, State startState, String description) {
this.states = new HashSet(states);
this.startState = startState;
this.currentState = startState;
this.description = description;
}
/**
* Make a copy of the StateMachine maintaining the current state.
*
* @return a copy of the StateMachine.
*/
public Object clone() {
StateMachine clone = new StateMachine(states, startState, description);
clone.currentState = currentState;
return clone;
}
/**
* Get the state machine description.
*
* @return an possibly null description.
*/
public String getDescription() {
return description;
}
/**
* Get the current state of the state machine.
*
* @return the current state.
*/
public State getCurrentState() {
return currentState;
}
/**
* Get the start state of the state machine.
*
* @return the start state.
*/
public State getStartState() {
return startState;
}
/**
* Get the states of the state machine.
*
* @return the machine states.
*/
public Set getStates() {
return states;
}
/**
* Transition to the next state given the name of a valid transition.
*
* @param actionName -
* the name of transition that is valid for the current state.
* @return the next state
* @throws IllegalTransitionException
*/
public State nextState(String actionName) throws IllegalTransitionException {
Transition t = currentState.getTransition(actionName);
if (t == null) {
String msg = "No transition for action: "" + actionName + "" from state: ""
+ currentState.getName() + """;
throw new IllegalTransitionException(msg);
}
State nextState = t.getTarget();
System.out.println("nextState(" + actionName + ") = " + nextState);
currentState = nextState;
return currentState;
}
/**
* Reset the state machine back to the start state
*
* @return the start state
*/
public State reset() {
this.currentState = startState;
return currentState;
}
public String toString() {
StringBuffer tmp = new StringBuffer("StateMachine[:\n");
tmp.append("\tCurrentState: " + currentState.getName());
Iterator i = states.iterator();
while (i.hasNext()) {
tmp.append("\n").append(i.next());
}
tmp.append("]");
return tmp.toString();
}
}
/*
* JBoss, Home of Professional Open Source Copyright 2005, JBoss Inc., and
* individual contributors as indicated by the @authors tag. See the
* copyright.txt in the distribution for a full listing of individual
* contributors.
*
* This is free software; you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at your option)
* any later version.
*
* This software is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
* details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this software; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA, or see the FSF
* site: http://www.fsf.org.
*/
/**
* The respresentation of a state in a state machine.
*
* @author Scott.Stark@jboss.org
* @version $Revision: 2787 $
*/
@SuppressWarnings("unchecked")
class State {
/** The name of the state */
private String name;
/** HashMap<String, Transition> */
private HashMap allowedTransitions = new HashMap();
/** Arbitrary state data */
private Object data;
public State(String name) {
this(name, null);
}
public State(String name, Map transitions) {
this.name = name;
if (transitions != null) {
allowedTransitions.putAll(transitions);
}
}
/**
* Get the state name.
*
* @return the name of the state.
*/
public String getName() {
return name;
}
public Object getData() {
return data;
}
public void setData(Object data) {
this.data = data;
}
/**
* An accept state is indicated by no transitions
*
* @return true if this is an accept state, false otherwise.
*/
public boolean isAcceptState() {
return allowedTransitions.size() == 0;
}
/**
* Add a transition to the allowed transition map.
*
* @param transition
* @return this to allow chained addTransition calls
*/
public State addTransition(Transition transition) {
allowedTransitions.put(transition.getName(), transition);
return this;
}
/**
* Lookup an allowed transition given its name.
*
* @param name -
* the name of a valid transition from this state.
* @return the valid transition if it exists, null otherwise.
*/
public Transition getTransition(String name) {
Transition t = (Transition) allowedTransitions.get(name);
return t;
}
/**
* Get the Map<String, Transition> of allowed transitions for this state.
*
* @return the allowed transitions map.
*/
public Map getTransitions() {
return allowedTransitions;
}
public String toString() {
StringBuffer tmp = new StringBuffer("State(name=");
tmp.append(name);
Iterator i = allowedTransitions.entrySet().iterator();
while (i.hasNext()) {
Map.Entry e = (Map.Entry) i.next();
tmp.append("\n\t on: ");
tmp.append(e.getKey());
Transition t = (Transition) e.getValue();
tmp.append(" go to: ");
tmp.append(t.getTarget().getName());
}
tmp.append(")");
return tmp.toString();
}
}
/*
* JBoss, Home of Professional Open Source Copyright 2005, JBoss Inc., and
* individual contributors as indicated by the @authors tag. See the
* copyright.txt in the distribution for a full listing of individual
* contributors.
*
* This is free software; you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at your option)
* any later version.
*
* This software is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
* details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this software; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA, or see the FSF
* site: http://www.fsf.org.
*/
/**
* A representation of a transition from a state to another state.
*
* @author Scott.Stark@jboss.org
* @version $Revision: 1958 $
*/
class Transition {
private String name;
private State target;
public Transition(String name, State target) {
this.name = name;
this.target = target;
}
public String getName() {
return name;
}
public State getTarget() {
return target;
}
}
/*
* JBoss, Home of Professional Open Source Copyright 2005, JBoss Inc., and
* individual contributors as indicated by the @authors tag. See the
* copyright.txt in the distribution for a full listing of individual
* contributors.
*
* This is free software; you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at your option)
* any later version.
*
* This software is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
* details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this software; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA, or see the FSF
* site: http://www.fsf.org.
*/
/**
* An exception thrown when an invalid transition is attempted from a state.
*
* @author Scott.Stark@jboss.org
* @version $Revision: 2800 $
*/
class IllegalTransitionException extends Exception {
/** The serialVersionUID */
private static final long serialVersionUID = -3392564168782896452L;
public IllegalTransitionException(String msg) {
super(msg);
}
}