1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
//! Traits involving the length of a collection. /// A trait for describing whether a collection is empty, i.e., its length is zero. /// /// Checking whether a collection is empty should take a constant amount of time and space. pub trait Empty { /// Returns whether the collection is empty. /// /// # Examples /// /// ```rust /// use len_trait::Empty; /// /// fn check_empty<C: ?Sized + Empty>(collection: &C) { /// assert!(collection.is_empty()); /// } /// /// check_empty(""); /// check_empty(&[1, 2, 3][..0]); /// check_empty(&"".to_string()); /// ``` fn is_empty(&self) -> bool; } /// A trait for describing the length of a collection. /// /// The amount of data stored in a collection, i.e. the amount of space it requires in memory, is /// directly proportional to its length. For this reason, `str` and other types measure their /// lengths in code values (e.g. `u8`), not code points (e.g. `char`). /// /// Obtaining the length of the collection must take a constant amount of time and space. #[cfg_attr(feature = "cargo-clippy", allow(len_without_is_empty))] // TODO: https://github.com/Manishearth/rust-clippy/issues/1740 pub trait Len: Empty { /// Returns the length of the collection. /// /// # Examples /// /// ```rust /// use len_trait::Len; /// /// fn print_len<C: ?Sized + Len>(collection: &C) { /// println!("{} units long", collection.len()); /// } /// /// print_len("中文"); // 6 units long /// print_len(&[1, 2, 3][..]); // 3 units long /// ``` fn len(&self) -> usize; } /// A trait for clearing collections. /// /// A collection is cleared by dropping all of its data. After `clear` is called, the collection is /// guaranteed to be empty. /// /// Clearing a collection must take at most a linear amount of time and space. pub trait Clear: Len { /// Clears the collection, dropping its elements and returning its length to zero. /// /// # Examples /// /// ```rust /// use len_trait::Clear; /// /// struct Value(u32); /// impl Drop for Value { /// fn drop(&mut self) { /// println!("Dropped a {}", self.0) /// } /// } /// /// fn check_clear<C: Clear>(mut collection: C) { /// collection.clear(); /// assert!(collection.is_empty()); /// } /// /// check_clear(vec![Value(12)]); // Prints "Dropped a 12" /// check_clear("Hello, world!".to_string()); /// ``` fn clear(&mut self); } /// A trait for modifying the length of a collection. /// /// These methods must take at most a linear amount of time and space with respect to the number of /// elements which are moved or dropped. pub trait LenMut: Default + Clear { /// Truncates the collection to be no greater than `len` long, dropping elements as needed. /// /// If the collection is less than `len` long, do nothing. /// /// # Panics /// /// Panics if `len` is not valid according to the collection. For example, the implementation /// for `String` will panic if `len` does not lie on a character boundary. /// /// # Examples /// /// ```rust /// use len_trait::LenMut; /// /// fn check_truncate<C: LenMut>(mut collection: C) { /// let old_len = collection.len(); /// collection.truncate(5); /// if old_len >= 5 { /// assert_eq!(collection.len(), 5); /// } else { /// assert_eq!(collection.len(), old_len); /// } /// } /// /// check_truncate("Hello, world!".to_string()); /// check_truncate(vec![1, 2, 3]); /// ``` fn truncate(&mut self, len: usize); /// Splits off the collection at the given index, returning the data past the index. /// /// # Panics /// /// Panics if `index > len`. /// /// # Examples /// /// ```rust /// use len_trait::LenMut; /// /// fn check_split_off<C: LenMut>(mut collection: C) { /// let old_len = collection.len(); /// let split = collection.split_off(5); /// assert_eq!(collection.len(), 5); /// assert_eq!(split.len(), old_len - 5); /// } /// /// check_split_off("Hello, world!".to_string()); /// check_split_off(vec![1, 2, 3, 4, 5, 6, 7, 8, 9]); /// ``` fn split_off(&mut self, index: usize) -> Self; }