当前位置: 首页 > news >正文

卓商网站建设临淄网站建设

卓商网站建设,临淄网站建设,秦洁婷seo博客,要查询一个网站在什么公司做的推广怎么查导航 循环引用一、现象二、解决 循环引用 循环引用出现的一个场景就是你指向我,我指向你,导致程序崩溃 解决方式可以通过弱指针,而Rust中的弱指针就是Weak 在Rc中,可以实现,对一个变量,持有多个不可变引…

导航

  • 循环引用
    • 一、现象
    • 二、解决

循环引用

循环引用出现的一个场景就是你指向我我指向你,导致程序崩溃

解决方式可以通过弱指针,而Rust中的弱指针就是Weak

Rc中,可以实现,对一个变量,持有多个不可变引用,并且都拥有该变量的所有权,这种情况下就可能会发生循环引用的现象

一、现象

从代码中来,到代码中去,先上个图
在这里插入图片描述

use crate::List::{Cons, Nil};
use std::cell::RefCell;
use std::rc::Rc;#[derive(Debug)]
enum List {Cons(i32, RefCell<Rc<List>>),Nil,
}impl List {fn tail(&self) -> Option<&RefCell<Rc<List>>> {match self {Cons(_, item) => Some(item),Nil => None,}}
}
fn main() {//Cons(5, RefCell::new(Rc::new(Nil)))已经有一个拥有者alet a = Rc::new(Cons(5, RefCell::new(Rc::new(Nil))));println!("a的初始化rc计数 = {}", Rc::strong_count(&a));println!("a指向的节点 = {:?}", a.tail());// 创建`b`到`a`的引用//b指向的结构体Cons(10, RefCell::new(Rc::clone(&a)),结构体字段也拥有了Cons(5, RefCell::new(Rc::new(Nil))//对于Cons(10, RefCell::new(Rc::clone(&a)结构体,只有一个拥有者blet b = Rc::new(Cons(10, RefCell::new(Rc::clone(&a))));//所以对于Cons(5, RefCell::new(Rc::new(Nil)))已经有了两个拥有者println!("在b创建后,a的rc计数 = {}", Rc::strong_count(&a));//这里是2println!("b的初始化rc计数 = {}", Rc::strong_count(&b));//这里是1println!("b指向的节点 = {:?}", b.tail());// 利用RefCell的可变性,创建了`a`到`b`的引用if let Some(link) = a.tail() {//因为 Cons(i32, RefCell<Rc<List>>)内部定义的是RefCell,可以修改指向,指向另一个Rc*link.borrow_mut() = Rc::clone(&b);//指向b}//对于Cons(10, RefCell::new(Rc::clone(&a)已经有b和a结构体中的Cons字段,两个所有者println!("在更改a后,b的rc计数 = {}", Rc::strong_count(&b));//对于Cons(5, RefCell::new(Rc::clone(&b)已经有a和b结构体中的Cons字段,两个所有者println!("在更改a后,a的rc计数 = {}", Rc::strong_count(&a));// 下面一行println!将导致循环引用//Rc指向的对象被drop的原则是,所有拥有者已经失去所有权,如果不主动drop,就无法解开引用// 下面一行println!将导致循环引用// 我们可怜的8MB大小的main线程栈空间将被它冲垮,最终造成栈溢出,Debug特征会一直打印每个结构体的字段,所以会递归// println!("a next item = {:?}", a.tail());
}

二、解决

Weak 非常类似于 Rc,但是与 Rc 持有所有权不同,Weak 不持有所有权,借用的值可以在生命周期结束后正常drop掉,就不会出现循环引用了.

WeakRc不同的地方是:

  1. 获取一个没有所有权的引用方式是,Rc::downgrade(&)
  2. 获取借用值是upgrade(),返回的是Option<Rc> ,值存在就返回引用,不存在就返回None
use std::rc::Rc;
use std::rc::Weak;
use std::cell::RefCell;// 主人
struct Owner {name: String,gadgets: RefCell<Vec<Weak<Gadget>>>,
}// 工具
struct Gadget {id: i32,owner: Rc<Owner>,
}fn main() {// 创建一个 Owner// 需要注意,该 Owner 也拥有多个 `gadgets`let gadget_owner : Rc<Owner> = Rc::new(Owner {name: "Gadget Man".to_string(),gadgets: RefCell::new(Vec::new()),});// 创建工具,同时与主人进行关联:创建两个 gadget,他们分别持有 gadget_owner 的一个引用。let gadget1 = Rc::new(Gadget{id: 1, owner: gadget_owner.clone()});let gadget2 = Rc::new(Gadget{id: 2, owner: gadget_owner.clone()});// 为主人更新它所拥有的工具// 因为之前使用了 `Rc`,现在必须要使用 `Weak`,否则就会循环引用gadget_owner.gadgets.borrow_mut().push(Rc::downgrade(&gadget1));gadget_owner.gadgets.borrow_mut().push(Rc::downgrade(&gadget2));// 遍历 gadget_owner 的 gadgets 字段for gadget_opt in gadget_owner.gadgets.borrow().iter() {// gadget_opt 是一个 Weak<Gadget> 。 因为 weak 指针不能保证他所引用的对象// 仍然存在。所以我们需要显式的调用 upgrade() 来通过其返回值(Option<_>)来判// 断其所指向的对象是否存在。// 当然,Option 为 None 的时候这个引用原对象就不存在了。let gadget = gadget_opt.upgrade().unwrap();println!("Gadget {} owned by {}", gadget.id, gadget.owner.name);}// 在 main 函数的最后,gadget_owner,gadget1 和 gadget2 都被销毁。// 具体是,因为这几个结构体之间没有了强引用(`Rc<T>`),所以,当他们销毁的时候。//!!!!!!// 首先 gadget2 和 gadget1 被销毁。// 然后因为 gadget_owner 的引用数量为 0,所以这个对象可以被销毁了。// 循环引用问题也就避免了
}

在这里插入图片描述

http://www.yayakq.cn/news/630649/

相关文章:

  • 化妆品网站建设网站上海殷行建设网站
  • 徐州网站二次开发网站中的游戏是怎么做的
  • 发布消息做任务的网站做物流行业网站的开发公司
  • 专业网站制作咨询设计官网入口
  • 元芳重庆网站建设网站显示正在建设中
  • 怎么设置网站字体wordpress文章缩略图
  • 有哪些专门做写字楼的网站专业的建网站公司地址
  • 宁波专业网站建设公司济南模板网站设计
  • 购物网站代码模板免费网店有哪些平台
  • 无障碍网站建设《电子商务网站开发》实验报告
  • 建设网站需要哪些步骤wordpress添加文章内容目录
  • 网站的icp备案网站 扩展
  • 网站自己怎么做的网站前端是什么
  • 淘宝做网站的店wordpress 4.9 升级
  • 怎么做那些盗号网站pc 手机自适应网站用什么做
  • 建立一个网站平台需要多少钱新能源汽车价格排名
  • 上海浦东网站设计公司wordpress精美免费主题
  • 网站域名解析失败汽车网站模板免费下载
  • 怎样给装修公司做网站wordpress插件引入js
  • 公司做的网站列入什么科目iis网站服务器 建立出现问题
  • 室内效果图网站太原网络推广代理公司
  • 网站搭建技术都有啥抖音seo公司
  • 局域网内的网站建设东莞市寮步镇防疫新闻最新
  • 域名注册好怎么建设网站东营网站建设公司 网络易
  • 怎样先做网站后买域名牙膏的网站建设方案
  • 免费自助建站系统有哪些如何做购物网站
  • 伊利网站设计做网站就是做app
  • 网站内图片变换怎么做成都手机网站
  • 企业网站托管后果中国建设银行安徽省招聘信息网站
  • 怎么注册网站挣流量wordpress中文标题