zerovec/map2d/
borrowed.rs

1// This file is part of ICU4X. For terms of use, please see the file
2// called LICENSE at the top level of the ICU4X source tree
3// (online at: https://github.com/unicode-org/icu4x/blob/main/LICENSE ).
4
5use crate::ZeroSlice;
6
7use core::cmp::Ordering;
8use core::fmt;
9
10use crate::map::ZeroMapKV;
11use crate::map::ZeroVecLike;
12use crate::map2d::ZeroMap2dCursor;
13
14/// A borrowed-only version of [`ZeroMap2d`](super::ZeroMap2d)
15///
16/// This is useful for fully-zero-copy deserialization from non-human-readable
17/// serialization formats. It also has the advantage that it can return references that live for
18/// the lifetime of the backing buffer as opposed to that of the [`ZeroMap2dBorrowed`] instance.
19///
20/// # Examples
21///
22/// ```
23/// use zerovec::maps::ZeroMap2dBorrowed;
24///
25/// // Example byte buffer representing the map { 1: {2: "three" } }
26/// let BINCODE_BYTES: &[u8; 51] = &[
27///     2, 0, 0, 0, 0, 0, 0, 0, 1, 0, 4, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 2, 0,
28///     0, 0, 0, 0, 0, 0, 2, 0, 11, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 116,
29///     104, 114, 101, 101,
30/// ];
31///
32/// // Deserializing to ZeroMap2d requires no heap allocations.
33/// let zero_map: ZeroMap2dBorrowed<u16, u16, str> =
34///     bincode::deserialize(BINCODE_BYTES)
35///         .expect("Should deserialize successfully");
36/// assert_eq!(zero_map.get_2d(&1, &2), Some("three"));
37/// ```
38///
39/// This can be obtained from a [`ZeroMap2d`](super::ZeroMap2d) via [`ZeroMap2d::as_borrowed`](super::ZeroMap2d::as_borrowed)
40pub struct ZeroMap2dBorrowed<'a, K0, K1, V>
41where
42    K0: ZeroMapKV<'a>,
43    K1: ZeroMapKV<'a>,
44    V: ZeroMapKV<'a>,
45    K0: ?Sized,
46    K1: ?Sized,
47    V: ?Sized,
48{
49    pub(crate) keys0: &'a K0::Slice,
50    pub(crate) joiner: &'a ZeroSlice<u32>,
51    pub(crate) keys1: &'a K1::Slice,
52    pub(crate) values: &'a V::Slice,
53}
54
55impl<'a, K0, K1, V> Copy for ZeroMap2dBorrowed<'a, K0, K1, V>
56where
57    K0: ZeroMapKV<'a>,
58    K1: ZeroMapKV<'a>,
59    V: ZeroMapKV<'a>,
60    K0: ?Sized,
61    K1: ?Sized,
62    V: ?Sized,
63{
64}
65
66impl<'a, K0, K1, V> Clone for ZeroMap2dBorrowed<'a, K0, K1, V>
67where
68    K0: ZeroMapKV<'a>,
69    K1: ZeroMapKV<'a>,
70    V: ZeroMapKV<'a>,
71    K0: ?Sized,
72    K1: ?Sized,
73    V: ?Sized,
74{
75    fn clone(&self) -> Self {
76        *self
77    }
78}
79
80impl<'a, K0, K1, V> Default for ZeroMap2dBorrowed<'a, K0, K1, V>
81where
82    K0: ZeroMapKV<'a>,
83    K1: ZeroMapKV<'a>,
84    V: ZeroMapKV<'a>,
85    K0::Slice: 'static,
86    K1::Slice: 'static,
87    V::Slice: 'static,
88    K0: ?Sized,
89    K1: ?Sized,
90    V: ?Sized,
91{
92    fn default() -> Self {
93        Self::new()
94    }
95}
96
97impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
98where
99    K0: ZeroMapKV<'a>,
100    K1: ZeroMapKV<'a>,
101    V: ZeroMapKV<'a>,
102    K0::Slice: 'static,
103    K1::Slice: 'static,
104    V::Slice: 'static,
105    K0: ?Sized,
106    K1: ?Sized,
107    V: ?Sized,
108{
109    /// Creates a new, empty `ZeroMap2dBorrowed<K0, K1, V>`.
110    ///
111    /// Note: Since [`ZeroMap2dBorrowed`] is not mutable, the return value will be a stub unless
112    /// converted into a [`ZeroMap2d`](super::ZeroMap2d).
113    ///
114    /// # Examples
115    ///
116    /// ```
117    /// use zerovec::maps::ZeroMap2dBorrowed;
118    ///
119    /// let zm: ZeroMap2dBorrowed<u16, u16, str> = ZeroMap2dBorrowed::new();
120    /// assert!(zm.is_empty());
121    /// ```
122    pub fn new() -> Self {
123        Self {
124            keys0: K0::Container::zvl_new_borrowed(),
125            joiner: Default::default(),
126            keys1: K1::Container::zvl_new_borrowed(),
127            values: V::Container::zvl_new_borrowed(),
128        }
129    }
130}
131
132impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
133where
134    K0: ZeroMapKV<'a>,
135    K1: ZeroMapKV<'a>,
136    V: ZeroMapKV<'a>,
137    K0: ?Sized,
138    K1: ?Sized,
139    V: ?Sized,
140{
141    #[doc(hidden)] // databake internal
142    pub const unsafe fn from_parts_unchecked(
143        keys0: &'a K0::Slice,
144        joiner: &'a ZeroSlice<u32>,
145        keys1: &'a K1::Slice,
146        values: &'a V::Slice,
147    ) -> Self {
148        Self {
149            keys0,
150            joiner,
151            keys1,
152            values,
153        }
154    }
155
156    /// The number of elements in the [`ZeroMap2dBorrowed`]
157    pub fn len(&self) -> usize {
158        self.values.zvl_len()
159    }
160
161    /// Whether the [`ZeroMap2dBorrowed`] is empty
162    pub fn is_empty(&self) -> bool {
163        self.values.zvl_len() == 0
164    }
165}
166
167impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
168where
169    K0: ZeroMapKV<'a> + Ord,
170    K1: ZeroMapKV<'a> + Ord,
171    V: ZeroMapKV<'a>,
172    K0: ?Sized,
173    K1: ?Sized,
174    V: ?Sized,
175{
176    /// Get the value associated with `key0` and `key1`, if it exists.
177    ///
178    /// This is able to return values that live longer than the map itself
179    /// since they borrow directly from the backing buffer. This is the
180    /// primary advantage of using [`ZeroMap2dBorrowed`](super::ZeroMap2dBorrowed) over [`ZeroMap2d`](super::ZeroMap2d).
181    ///
182    /// ```rust
183    /// use zerovec::ZeroMap2d;
184    ///
185    /// let mut map = ZeroMap2d::new();
186    /// map.insert(&1, "one", "foo");
187    /// map.insert(&2, "one", "bar");
188    /// map.insert(&2, "two", "baz");
189    ///
190    /// let borrowed = map.as_borrowed();
191    /// assert_eq!(borrowed.get_2d(&1, "one"), Some("foo"));
192    /// assert_eq!(borrowed.get_2d(&1, "two"), None);
193    /// assert_eq!(borrowed.get_2d(&2, "one"), Some("bar"));
194    /// assert_eq!(borrowed.get_2d(&2, "two"), Some("baz"));
195    /// assert_eq!(borrowed.get_2d(&3, "three"), None);
196    /// ```
197    pub fn get_2d(&self, key0: &K0, key1: &K1) -> Option<&'a V::GetType> {
198        self.get0(key0)?.get1(key1)
199    }
200}
201
202impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
203where
204    K0: ZeroMapKV<'a> + Ord,
205    K1: ZeroMapKV<'a>,
206    V: ZeroMapKV<'a>,
207    K0: ?Sized,
208    K1: ?Sized,
209    V: ?Sized,
210{
211    /// Gets a cursor for `key0`. If `None`, then `key0` is not in the map. If `Some`,
212    /// then `key0` is in the map, and `key1` can be queried.
213    ///
214    /// ```rust
215    /// use zerovec::ZeroMap2d;
216    ///
217    /// let mut map = ZeroMap2d::new();
218    /// map.insert(&1, "one", "foo");
219    /// map.insert(&2, "two", "bar");
220    /// let borrowed = map.as_borrowed();
221    /// assert!(matches!(borrowed.get0(&1), Some(_)));
222    /// assert!(matches!(borrowed.get0(&3), None));
223    /// ```
224    #[inline]
225    pub fn get0<'l>(&'l self, key0: &K0) -> Option<ZeroMap2dCursor<'a, 'a, K0, K1, V>> {
226        let key0_index = self.keys0.zvl_binary_search(key0).ok()?;
227        Some(ZeroMap2dCursor::from_borrowed(self, key0_index))
228    }
229
230    /// Binary search the map for `key0`, returning a cursor.
231    ///
232    /// ```rust
233    /// use zerovec::ZeroMap2d;
234    ///
235    /// let mut map = ZeroMap2d::new();
236    /// map.insert(&1, "one", "foo");
237    /// map.insert(&2, "two", "bar");
238    /// let borrowed = map.as_borrowed();
239    /// assert!(matches!(borrowed.get0_by(|probe| probe.cmp(&1)), Some(_)));
240    /// assert!(matches!(borrowed.get0_by(|probe| probe.cmp(&3)), None));
241    /// ```
242    pub fn get0_by<'l>(
243        &'l self,
244        predicate: impl FnMut(&K0) -> Ordering,
245    ) -> Option<ZeroMap2dCursor<'a, 'a, K0, K1, V>> {
246        let key0_index = self.keys0.zvl_binary_search_by(predicate).ok()?;
247        Some(ZeroMap2dCursor::from_borrowed(self, key0_index))
248    }
249
250    /// Returns whether `key0` is contained in this map
251    ///
252    /// ```rust
253    /// use zerovec::ZeroMap2d;
254    ///
255    /// let mut map = ZeroMap2d::new();
256    /// map.insert(&1, "one", "foo");
257    /// map.insert(&2, "two", "bar");
258    /// let borrowed = map.as_borrowed();
259    /// assert!(borrowed.contains_key0(&1));
260    /// assert!(!borrowed.contains_key0(&3));
261    /// ```
262    pub fn contains_key0(&self, key0: &K0) -> bool {
263        self.keys0.zvl_binary_search(key0).is_ok()
264    }
265}
266
267impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
268where
269    K0: ZeroMapKV<'a>,
270    K1: ZeroMapKV<'a>,
271    V: ZeroMapKV<'a>,
272    K0: ?Sized,
273    K1: ?Sized,
274    V: ?Sized,
275{
276    /// Produce an ordered iterator over keys0
277    pub fn iter0<'l>(&'l self) -> impl Iterator<Item = ZeroMap2dCursor<'a, 'a, K0, K1, V>> + '_ {
278        (0..self.keys0.zvl_len()).map(move |idx| ZeroMap2dCursor::from_borrowed(self, idx))
279    }
280}
281
282impl<'a, K0, K1, V> ZeroMap2dBorrowed<'a, K0, K1, V>
283where
284    K0: ZeroMapKV<'a> + Ord,
285    K1: ZeroMapKV<'a> + Ord,
286    V: ZeroMapKV<'a>,
287    V: Copy,
288    K0: ?Sized,
289    K1: ?Sized,
290{
291    /// For cases when `V` is fixed-size, obtain a direct copy of `V` instead of `V::ULE`
292    pub fn get_copied_2d(&self, key0: &K0, key1: &K1) -> Option<V> {
293        self.get0(key0)?.get1_copied(key1)
294    }
295}
296
297// We can't use the default PartialEq because ZeroMap2d is invariant
298// so otherwise rustc will not automatically allow you to compare ZeroMaps
299// with different lifetimes
300impl<'a, 'b, K0, K1, V> PartialEq<ZeroMap2dBorrowed<'b, K0, K1, V>>
301    for ZeroMap2dBorrowed<'a, K0, K1, V>
302where
303    K0: for<'c> ZeroMapKV<'c> + ?Sized,
304    K1: for<'c> ZeroMapKV<'c> + ?Sized,
305    V: for<'c> ZeroMapKV<'c> + ?Sized,
306    <K0 as ZeroMapKV<'a>>::Slice: PartialEq<<K0 as ZeroMapKV<'b>>::Slice>,
307    <K1 as ZeroMapKV<'a>>::Slice: PartialEq<<K1 as ZeroMapKV<'b>>::Slice>,
308    <V as ZeroMapKV<'a>>::Slice: PartialEq<<V as ZeroMapKV<'b>>::Slice>,
309{
310    fn eq(&self, other: &ZeroMap2dBorrowed<'b, K0, K1, V>) -> bool {
311        self.keys0.eq(other.keys0)
312            && self.joiner.eq(other.joiner)
313            && self.keys1.eq(other.keys1)
314            && self.values.eq(other.values)
315    }
316}
317
318impl<'a, K0, K1, V> fmt::Debug for ZeroMap2dBorrowed<'a, K0, K1, V>
319where
320    K0: ZeroMapKV<'a> + ?Sized,
321    K1: ZeroMapKV<'a> + ?Sized,
322    V: ZeroMapKV<'a> + ?Sized,
323    K0::Slice: fmt::Debug,
324    K1::Slice: fmt::Debug,
325    V::Slice: fmt::Debug,
326{
327    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
328        f.debug_struct("ZeroMap2dBorrowed")
329            .field("keys0", &self.keys0)
330            .field("joiner", &self.joiner)
331            .field("keys1", &self.keys1)
332            .field("values", &self.values)
333            .finish()
334    }
335}