Rc
Когда необходимо множественное владение, может использоваться Rc
(счётчик ссылок). Rc
отслеживает количество ссылок, то есть количество владельцев значения, сохранённого внутри Rc
.
Счётчик ссылок в Rc
увеличивается на 1 каждый
раз, когда Rc
клонируется, и уменьшается на 1, когда
любой из клонов Rc
выходит из области видимости и удаляется. Когда
количество ссылок в Rc
становится равным нулю,
т.е. когда владельцев больше нет, и сам Rc
, и значение внутри него удаляются.
При клонировании Rc
никогда не делается глубокая копия. Клонирование лишь создаёт ещё один указатель на обёрнутое значение и увеличивает счётчик.
use std::rc::Rc; fn main() { let rc_examples = "Примеры с Rc".to_string(); { println!("--- Создан rc_a ---"); let rc_a: Rc<String> = Rc::new(rc_examples); println!("Счётчик ссылок в rc_a: {}", Rc::strong_count(&rc_a)); { println!("--- rc_a клонировано в rc_b ---"); let rc_b: Rc<String> = Rc::clone(&rc_a); println!("Счётчик ссылок в rc_b: {}", Rc::strong_count(&rc_b)); println!("Счётчик ссылок в rc_a: {}", Rc::strong_count(&rc_a)); // Два `Rc` равны, если равны их внутренние значения println!("rc_a и rc_b равны: {}", rc_a.eq(&rc_b)); // Мы можем напрямую использовать методы внутреннего значения println!("Размер значения внутри rc_a: {}", rc_a.len()); println!("Значение rc_b: {}", rc_b); println!("--- rc_b удаляется ---"); } println!("Счётчик ссылок в rc_a: {}", Rc::strong_count(&rc_a)); println!("--- rc_a удаляется ---"); } // Ошибка! Строка `rc_examples` уже перемещена в `rc_a` // И когда `rc_a` был удалён, `rc_examples` удалилась вместе с ним // println!("rc_examples: {}", rc_examples); // TODO ^ Попробуйте раскомментировать эту строку }
Смотрите также:
std::rc and std::sync::arc.