// Copyright 2012-2013 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! A mutable, nullable memory location #[missing_doc]; use cast::transmute_mut; use unstable::finally::Finally; use prelude::*; /* A dynamic, mutable location. Similar to a mutable option type, but friendlier. */ #[no_freeze] #[deriving(Clone, DeepClone, Eq)] #[allow(missing_doc)] pub struct Cell { priv value: Option } impl Cell { /// Creates a new full cell with the given value. pub fn new(value: T) -> Cell { Cell { value: Some(value) } } /// Creates a new empty cell with no value inside. pub fn new_empty() -> Cell { Cell { value: None } } /// Yields the value, failing if the cell is empty. pub fn take(&self) -> T { let this = unsafe { transmute_mut(self) }; if this.is_empty() { fail!("attempt to take an empty cell"); } this.value.take_unwrap() } /// Yields the value if the cell is full, or `None` if it is empty. pub fn take_opt(&self) -> Option { let this = unsafe { transmute_mut(self) }; this.value.take() } /// Returns the value, failing if the cell is full. pub fn put_back(&self, value: T) { let this = unsafe { transmute_mut(self) }; if !this.is_empty() { fail!("attempt to put a value back into a full cell"); } this.value = Some(value); } /// Returns true if the cell is empty and false if the cell is full. pub fn is_empty(&self) -> bool { self.value.is_none() } /// Calls a closure with a reference to the value. pub fn with_ref(&self, op: &fn(v: &T) -> R) -> R { do self.with_mut_ref |ptr| { op(ptr) } } /// Calls a closure with a mutable reference to the value. pub fn with_mut_ref(&self, op: &fn(v: &mut T) -> R) -> R { let mut v = Some(self.take()); do (|| { op(v.get_mut_ref()) }).finally { self.put_back(v.take_unwrap()); } } } #[test] fn test_basic() { let value_cell = Cell::new(~10); assert!(!value_cell.is_empty()); let value = value_cell.take(); assert!(value == ~10); assert!(value_cell.is_empty()); value_cell.put_back(value); assert!(!value_cell.is_empty()); } #[test] #[should_fail] fn test_take_empty() { let value_cell: Cell<~int> = Cell::new_empty(); value_cell.take(); } #[test] #[should_fail] fn test_put_back_non_empty() { let value_cell = Cell::new(~10); value_cell.put_back(~20); } #[test] fn test_with_ref() { let good = 6; let c = Cell::new(~[1, 2, 3, 4, 5, 6]); let l = do c.with_ref() |v| { v.len() }; assert_eq!(l, good); } #[test] fn test_with_mut_ref() { let good = ~[1, 2, 3]; let v = ~[1, 2]; let c = Cell::new(v); do c.with_mut_ref() |v| { v.push(3); } let v = c.take(); assert_eq!(v, good); }