From: Martin Pitt Date: Sun, 29 Aug 2021 13:55:13 +0000 (+0200) Subject: concepts: Dynamic trait objects X-Git-Url: https://piware.de/gitweb/?p=learn-rust.git;a=commitdiff_plain;h=c182c6166ae3971320daaeadba3ea718124303e7 concepts: Dynamic trait objects --- diff --git a/concepts/src/lib.rs b/concepts/src/lib.rs index b7e5eb8..ca06241 100644 --- a/concepts/src/lib.rs +++ b/concepts/src/lib.rs @@ -108,3 +108,86 @@ impl Iterator for Counter5 { } } } + +pub struct Post { + state: Option>, + content: String, +} + +impl Post { + pub fn new() -> Post { + Post { + state: Some(Box::new(Draft {})), + content: String::new(), + } + } + + pub fn add_text(&mut self, text: &str) { + self.content.push_str(text); + } + + pub fn content(&self) -> &str { + // as_ref() converts Option> to Option<&Box> + // state can never be None, all state transitions return a new one + self.state.as_ref().unwrap().content(self) + } + + pub fn request_review(&mut self) { + if let Some(s) = self.state.take() { + self.state = Some(s.request_review()); + } + } + + pub fn approve(&mut self) { + if let Some(s) = self.state.take() { + self.state = Some(s.approve()); + } + } +} + +trait State { + fn request_review(self: Box::) -> Box; + fn approve(self: Box::) -> Box; + + #[allow(unused_variables)] + fn content<'a>(&self, post: &'a Post) -> &'a str { + "" + } +} + +struct Draft {} +impl State for Draft { + fn request_review(self: Box::) -> Box { + Box::new(PendingReview {}) + } + + fn approve(self: Box::) -> Box { + self + } +} + +struct PendingReview {} +impl State for PendingReview { + fn request_review(self: Box::) -> Box { + self + } + + fn approve(self: Box::) -> Box { + Box::new(Published {}) + } +} + +struct Published {} +impl State for Published { + fn request_review(self: Box::) -> Box { + self + } + + fn approve(self: Box::) -> Box { + Box::new(Published {}) + } + + fn content<'a>(&self, post: &'a Post) -> &'a str { + &post.content + } +} diff --git a/concepts/src/main.rs b/concepts/src/main.rs index 94bdb96..f9eafe9 100644 --- a/concepts/src/main.rs +++ b/concepts/src/main.rs @@ -247,6 +247,19 @@ fn test_threads() { println!("counter: {}", *counter.lock().unwrap()); } +fn test_dyn_traits() { + let text = "I ate a salad for lunch today"; + let mut post = Post::new(); + post.add_text(text); + assert_eq!("", post.content()); + + post.request_review(); + assert_eq!("", post.content()); + + post.approve(); + assert_eq!(text, post.content()); +} + fn main() { test_strings(); test_vectors(); @@ -256,4 +269,5 @@ fn main() { test_closures(); test_iterators(); test_threads(); + test_dyn_traits(); }