Design Patterns in Rust: The State Pattern
The state pattern is a behavourial state pattern, which allows an object to change its behaviour when its internal state changes. Since this sounds quite cryptic, let’s have a look at the diagram:
A short breakdown:
- _Context _has two component in this simplified version: a field called state, which is an interface type, and an operation() method.
- When the operation() method is called, the _Context _object calls the operation() on its state field. Since the _state _field can hold any implementation of the _State _interface, the behaviour of _Context _can change.
As you can see this is not the most complex of Design patterns.
In an empty directory open your commandline or terminal and type:
cargo new state_pattern
cd state_pattern
There are several ways to implement this pattern in Rust. We will start by implementing this using smart pointers. First we define the State trait:
trait State {
fn handle(&self);
}
This quite straightforward, just one method which does some handling.
Now implement StateA:
struct StateA;
impl State for StateA {
fn handle(&self) {
println!("Handling state A");
}
}
Some explanation:
- StateA is an empty struct. This is just for simplicity’s sake.
- We explicitly define the handle() method for StateA. This does nothing more than print out a message.
The definition of StateB is similar:
struct StateB;
impl State for StateB {
fn handle(&self) {
println!("Handling state B");
}
}
Now we define the Context:
struct Context {
state:Box<dyn State>
}
impl Context {
fn new(state: Box<dyn State>)->Self {
Self {
state: state
}
}
fn set_state(&mut self,state:Box<dyn State>) {
self.state=state;
}
}
impl State for Context {
fn handle(&self) {
self.state.handle();
}
}
A line by line explanation:
- The Context is a struct which holds one field in this example, state, which is a Box smartpointer to a dyn State. This is needed because Rust needs to know the size of an object before it can be placed on the stack. We need a dyn here, because State is a trait object.
- We define a constructor, new, which gets a Box-ed dyn State object and assigns it to the state.
- We also define a set_state() method, so we can change the state.
- We also implement the State trait for Context, this allows us to pass down the call to handle() to the state.
Now, time for a test run:
fn main() {
let state=StateA;
let mut context=Context::new(Box::new(state));
context.handle();
let second_state=StateB;
context.set_state(Box::new(second_state));
context.handle();
}
We will go through this:
- We construct an object of type StateA. Remember StateA implements the State trait.
- Then we construct the context, and pass it the Box-ed instance of the state struct.
- We call handle().
- Next we construct a new state called second_state of type StateB
- We set the state of the context by passing a Box-ed instance of the second_state to the set_state method.
- Again we call handle()
I found that the Box-ed parameters clutter up the code, so we will do something about it:
We can use impl trait types to get rid of the Box-ed parameters, like this. First we rewrite the Context:
struct Context<'a> {
state:&'a dyn State
}
impl<'a> Context<'a> {
fn new(state: &'a dyn State)->Self {
Self {
state: state
}
}
fn set_state(&mut self,state:&'a impl State) {
self.state=state;
}
}
impl<'a> State for Context<'a> {
fn handle(&self) {
self.state.handle();
}
}
- Because we are using unboxed types which can go out of scope before we are done using it, we need a lifetime parameter which I simple named ‘a
- In the Context implementation the constructor is mostly the same, but with the addition of the lifetime parameter.
- The set_state() method gets an &’a impl State parameter instead of a Box-ed pointer. This means this method takes any struct which implement the State trait.
- The State implementation for the Context just gets the additional lifetime parameter.
The main method also has some changes:
fn main() {
let state=&StateA;
let mut context=Context::new(state);
context.handle();
let second_state=&StateB;
context.set_state(second_state);
context.handle();
}
Line by line:
- We create a reference to a struct of type StateA
- We create a Context object and pass it the state variable. Notice that context has the mut keyword, since the set_state() method can change it.
- We call handle() on context
- Next we create second_state of type StateB
- We set the state of context using set_state().
- We call handle() again on the context
As you can see this method has less clutter.
This pattern was quite straightforward to implement. Rust is quite flexible in this way, as you can tell from the two different implementations. Also the Rust compiler is very helpful, and it came up with many a useful suggestion while writing this code.