lets consider following code:
```rust use once_cell::sync::OnceCell;
trait X{ fn string() -> String; }
// having to recompute string() over and over might be expensive (not in this example, but still)
// so we use lazy initialization
fn generic
VALUE.get_or_init(||{
T::string()
})
}
// And now it can be used like this struct A; impl X for A{ fn string() -> String{ "A".to_string() } }
struct B; impl X for B{ fn string() -> String{ "B".to_string() } }
fn main(){ asserteq!(generic::(), "A"); asserteq!(generic::(), "A"); // Wait what? // Not completely behaviour I was expecting // This is due to fact that static variable placed inside of generic function // wont be cloned into each version of function, but will be shared // Thus second call does not initialize value for B, but takes value // initialized in previous call. } ```
This crate was designed to solve this particular problem.
Lets make some changes:
```rust use genericstatic::StaticTypeMap; use oncecell::sync::OnceCell;
trait X{ fn string() -> String; }
// having to recompute string() over and over might be expensive (not in this example, but still)
// so we use lazy initialization
fn generic
map.call_once::<T, _>(||{
T::string()
})
}
// And now it can be used like this struct A; impl X for A{ fn string() -> String{ "A".to_string() } }
struct B; impl X for B{ fn string() -> String{ "B".to_string() } }
fn main(){ asserteq!(generic::(), "A"); asserteq!(generic::(), "B"); } ```
Current implementation uses RwLock to make it safe in concurrent applications, which will be slightly slower then regular