mirror of
https://github.com/erg-lang/erg.git
synced 2025-09-28 20:14:45 +00:00
64 lines
1.5 KiB
Rust
64 lines
1.5 KiB
Rust
use std::rc::Rc;
|
|
use std::cell::RefCell;
|
|
use std::borrow::{Borrow, ToOwned};
|
|
use std::hash::Hash;
|
|
|
|
use crate::set::Set;
|
|
use crate::{Str, RcArray};
|
|
|
|
#[derive(Debug)]
|
|
pub struct Cache<T: ?Sized>(RefCell<Set<Rc<T>>>);
|
|
|
|
impl<T: ?Sized> Default for Cache<T> {
|
|
fn default() -> Self { Self::new() }
|
|
}
|
|
|
|
impl<T: ?Sized> Cache<T> {
|
|
pub fn new() -> Self { Self(RefCell::new(Set::new())) }
|
|
}
|
|
|
|
impl Clone for Cache<str> {
|
|
fn clone(&self) -> Self {
|
|
Self(self.0.clone())
|
|
}
|
|
}
|
|
|
|
impl<T: Hash + Eq> Clone for Cache<T> {
|
|
fn clone(&self) -> Self {
|
|
Self(self.0.clone())
|
|
}
|
|
}
|
|
|
|
impl Cache<str> {
|
|
pub fn get(&self, s: &str) -> Str {
|
|
if let Some(cached) = self.0.borrow().get(s) {
|
|
return cached.clone().into()
|
|
} // &self.0 is dropped
|
|
let s = Str::rc(s);
|
|
self.0.borrow_mut().insert(s.clone().into_rc());
|
|
s
|
|
}
|
|
}
|
|
|
|
impl<T: Hash + Eq + Clone> Cache<[T]> {
|
|
pub fn get(&self, q: &[T]) -> Rc<[T]> {
|
|
if let Some(cached) = self.0.borrow().get(q) {
|
|
return cached.clone()
|
|
} // &self.0 is dropped
|
|
let s = RcArray::from(q);
|
|
self.0.borrow_mut().insert(s.clone());
|
|
s
|
|
}
|
|
}
|
|
|
|
impl<T: Hash + Eq> Cache<T> {
|
|
pub fn get<Q: ?Sized + Hash + Eq>(&self, q: &Q) -> Rc<T>
|
|
where Rc<T>: Borrow<Q>, Q: ToOwned<Owned = T> {
|
|
if let Some(cached) = self.0.borrow().get(q) {
|
|
return cached.clone()
|
|
} // &self.0 is dropped
|
|
let s = Rc::from(q.to_owned());
|
|
self.0.borrow_mut().insert(s.clone());
|
|
s
|
|
}
|
|
}
|