首页
题库
公司真题
专项练习
面试题库
在线编程
面试
面试经验
AI 模拟面试
简历
求职
学习
基础学习课
实战项目课
求职辅导课
专栏&文章
竞赛
搜索
我要招人
发布职位
发布职位、邀约牛人
更多企业解决方案
AI面试、笔试、校招、雇品
HR免费试用AI面试
最新面试提效必备
登录
/
注册
土拨猫
武汉理工大学 C++
发布于湖北
关注
已关注
取消关注
@牛啊牛の:
智能指针
C++智能指针:防止内存泄漏,动态管理内存,分配-自动释放内存》》unique_ptr, shared_ptr, weak_ptr;属于封装类。 unique_ptr:unique_ptr实现独占式拥有或严格拥有概念,保证同一时间内只有一个智能指针可以指向该对象。它对于避免资源泄露,例如,以new创建对象后因为发生异常而忘记调用delete时的情形特别有用。 实现原理:将拷贝构造函数和赋值拷贝构造函数申明为private或delete。不允许拷贝构造函数和赋值操作符,但是支持移动构造函数,通过std:move把一个对象指针变成右值之后可以移动给另一个unique_ptr shared_ptr:shared_ptr实现共享式拥有概念。多个智能指针可以指向相同对象,该对象和其相关资源会在“最后一个引用被销毁”时候释放。从名字share就可以看出了资源可以被多个指针共享,它使用计数机制来表明资源被几个指针共享。可以通过成员函数use_count()来查看资源的所有者个数。除了可以通过new来构造,还可以通过传入auto_ptr, unique_ptr,weak_ptr来构造。当我们调用release()时,当前指针会释放资源所有权,计数减一。当计数等于0时,资源会被释放。 shared_ptr 是为了解决 auto_ptr 在对象所有权上的局限性(auto_ptr 是独占的), 在使用引用计数的机制上提供了可以共享所有权的智能指针。 use_count 返回引用计数的个数 ; unique 返回是否是独占所有权( use_count 为 1) ; swap 交换两个 shared_ptr 对象(即交换所拥有的对象) ; reset 放弃内部对象的所有权或拥有对象的变更, 会引起原有对象的引用计数的减少 ; get 返回内部对象(指针), 由于已经重载了()方法, 因此和直接使用对象是一样的.如 shared_ptr sp(new int(1)); sp 与 sp.get()是等价的 ; 初始化:因为带有参数的 shared_ptr 构造函数是 explicit 类型的,所以不能像这样std::shared_ptr<int> p1 = new int();隐式调用它构造函数。创建新的shared_ptr对象的最佳方法是使用std :: make_shared: std::shared_ptr<int> p1 = std::make_shared<int>(); SharedPtr<string> pstr(new string("abc")); std::make_shared 一次性为int对象和用于引用计数的数据都分配了内存,而new操作符只是为int分配了内存。 -------共享所有权如何在参考计数的帮助下工作的? 1、当新的 shared_ptr 对象与指针关联时,则在其构造函数中,将与此指针关联的引用计数增加1。 2、当任何 shared_ptr 对象超出作用域时,则在其析构函数中,它将关联指针的引用计数减1。如果引用计数变为0,则表示没有其他 shared_ptr 对象与此内存关联,在这种情况下,它使用delete函数删除该内存。 ----------创建 shared_ptr 时注意事项:不要使用同一个原始指针构造 shared_ptr;创建多个 shared_ptr 的正常方法是使用一个已存在的shared_ptr 进行创建,而不是使用同一个原始指针进行创建。 int *num = new int(23); std::shared_ptr<int> p1(num); std::shared_ptr<int> p2(p1); // 正确使用方法 std::shared_ptr<int> p3(num); // 不推荐 std::cout << "p1 Reference = " << p1.use_count() << std::endl; // 输出 2 std::cout << "p2 Reference = " << p2.use_count() << std::endl; // 输出 2 std::cout << "p3 Reference = " << p3.use_count() << std::endl; // 输出 1 假如使用原始指针num创建了p1,又同样方法创建了p3,当p1超出作用域时会调用delete释放num内存,此时num成了悬空指针,当p3超出作用域再次delete的时候就可能会出错。 weak_ptr:weak_ptr 是一种不控制对象生命周期的智能指针, 它指向一个 shared_ptr 管理的对象。进行该对象的内存管理的是那个强引用的 shared_ptr。weak_ptr只是提供了对管理对象的一个访问手段。weak_ptr 设计的目的是为配合 shared_ptr 而引入的一种智能指针来协助 shared_ptr 工作,它只可以从一个 shared_ptr 或另一个 weak_ptr 对象构造, 它的构造和析构不会引起引用记数的增加或减少。weak_ptr是用来解决shared_ptr相互引用时的死锁问题,如果说两个shared_ptr相互引用,那么这两个指针的引用计数永远不可能下降为0,资源永远不会释放。它是对对象的一种弱引用,不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr可以直接赋值给它,它可以通过调用lock函数来获得shared_ptr。 class B;class A{public: shared_ptr<B> pb_; ~A(){ cout<<"A delete\n";}};class B{public: shared_ptr<A> pa_; ~B(){ cout<<"B delete\n";}};void fun(){ shared_ptr<B> pb(new B()); shared_ptr<A> pa(new A()); pb->pa_ = pa; pa->pb_ = pb; cout<<pb.use_count()<<endl; cout<<pa.use_count()<<endl;}int main(){ fun(); return 0;} 可以看到fun函数中pa ,pb之间互相引用,两个资源的引用计数为2,当要跳出函数时,智能指针pa,pb析构时两个资源引用计数会减一,但是两者引用计数还是为1,导致跳出函数时资源没有被释放(A B的析构函数没有被调用),如果把其中一个改为weak_ptr就可以了,我们把类A里面的shared_ptr pb_; 改为weak_ptr pb; 运行结果如下,这样的话,资源B的引用开始就只有1,当pb析构时,B的计数变为0,B得到释放,B释放的同时也会使A的计数减一,同时pa析构时使A的计数减一,那么A的计数为0,A得到释放。 注意:我们不能通过weak_ptr直接访问对象的方法,比如B对象中有一个方法print(),我们不能这样访问,pa->pb->print(); 英文pb是一个weak_ptr,应该先把它转化为shared_ptr,如:shared_ptr p = pa->pb_.lock(); p->print(); ----------1.5.6 weak_ptr 能不能知道对象计数为 0,为什么?参考回答: 不能。 weak_ptr是一种不控制对象生命周期的智能指针,它指向一个shared_ptr管理的对象。进行该对象管理的是那个引用的shared_ptr。weak_ptr只是提供了对管理 对象的一个访问手段。weak_ptr设计的目的只是为了配合shared_ptr而引入的一种智能指针,配合shared_ptr工作,它只可以从一个shared_ptr或者另一个weak_ptr对象构造,它的构造和析构不会引起计数的增加或减少。 --------1.5.7 weak_ptr 如何解决 shared_ptr 的循环引用问题? 参考回答: 为了解决循环引用导致的内存泄漏,引入了弱指针weak_ptr,weak_ptr的构造函数不会修改引用计数的值,从而不会对对象的内存进行管理,其类似一个普通指针,但是不会指向引用计数的共享内存,但是可以检测到所管理的对象是否已经被释放,从而避免非法访问。 ------------线程安全性 多线程环境下,调用不同shared_ptr实例的成员函数是不需要额外的同步手段的,即使这些shared_ptr拥有的是同样的对象。但是如果多线程访问(有写操作)同一个shared_ptr,则需要同步,否则就会有race condition 发生。也可以使用 shared_ptr overloads of atomic functions来防止race condition的发生。多个线程同时读同一个shared_ptr对象是线程安全的,但是如果是多个线程对同一个shared_ptr对象进行读和写,则需要加锁。 多线程读写shared_ptr所指向的同一个对象,不管是相同的shared_ptr对象,还是不同的shared_ptr对象,也需要加锁保护。例子如下: shared_ptr<long> global_instance = make_shared<long>(0);std::mutex g_i_mutex; void thread_fcn(){ //std::lock_guard<std::mutex> lock(g_i_mutex); //shared_ptr<long> local = global_instance; for(int i = 0; i < 100000000; i++) { *global_instance = *global_instance + 1; //*local = *local + 1; }} int main(int argc, char** argv){ thread thread1(thread_fcn); thread thread2(thread_fcn); thread1.join(); thread2.join(); cout << "*global_instance is " << *global_instance << endl; return 0;} 在线程函数thread_fcn的for循环中,2个线程同时对global_instance进行加1的操作。这就是典型的非线程安全的场景,最后的结果是未定的,运行结果如下: *global_instance is 197240539 如果使用的是每个线程的局部shared_ptr对象local,因为这些local指向相同的对象,因此结果也是未定的,运行结果如下: *global_instance is 160285803 因此,这种情况下必须加锁,将thread_fcn中的第一行代码的注释去掉之后,不管是使用global_instance,还是使用local,得到的结果都是: *global_instance is 200000000。 [考得全会][蒙的全对][面试必过][心想事成][成功上岸][offer+1][内推成功]
点赞 2
评论 3
全部评论
推荐
最新
楼层
暂无评论,快来抢首评~
相关推荐
04-17 16:59
科大讯飞_教育BG_后端开发(准入职员工)
从投递到OC,我用了三个月
在2025年秋招,我是从八月份开始陆续投递的,是的,没听错,26届的秋招不在是金九银十了,而是金八银九铜十铁十一了。我秋招一共投递300+,美团、京东、滴滴、快手等中大厂,九月份和十月份一直在面试,中间有一个非互联网厂过了,一直催我签三方,好在我拖了两周,最终在十月底拿了offer!
从投递到OC,你用了多久
点赞
评论
收藏
分享
04-17 22:46
已编辑
中南大学 测试开发
记录第一次寻找实习经历
背景background 本人bg双九非科班,传统工科,无实习经历。本科时没有学过java、数据库、计网等相关内容,对于第一份实习的寻找是迷茫的,无论是学什么(八股,项目),还是岗位的选择(前后端,开发,AI相关),都不知道怎么下手。 在和AI聊天、让其根据自身本科与研究生的经历推荐一些比较合适的岗位后,最终选择了测试开发这个岗位。于是,认准这个岗位后,在三月初返校后,我便开始了寻找暑期实习的学习之路。八股、项目、刷题learning 最开始我也不知道测开该怎么学习,但是根据查看一些牛客求职建议帖子得知通用的八股是必备的,于是我开始系统性学习计算机网络、计算机系统与数据库。同时在这段时间在b站...
点赞
评论
收藏
分享
04-09 14:12
浙江工业大学之江学院 Java
春招0面
学院本进不了开发岗吗
饼子吃到撑:
学院本是这样的,找工作拼运气,你技术再好人家筛选学历照样沉入海底
,海投就行了
点赞
评论
收藏
分享
03-26 19:49
京东_后端开发(实习员工)
腾讯暑期一面,诡异。
暑期后端,面了1h10min,无八股,没提过点评。好歹是最想去的厂,看了网上的面经直接吓哭了,准备了半天计网机遇。本来以为项目很对口部门,想着被好好拷打一番进步的心态进面试了,问到最后只有一脑门子问号。1️⃣聊天十分钟,问实习时间,稍微说了说AI。2️⃣然后开始对着项目的犄角旮旯问,完全不问技术栈,也不问任何知识点,就纯问,问这一个项目到底是干啥的。值得提的: 问了这个项目有没有需求,要是想让我用你会怎么推销。3️⃣其中盯着分片上传断点续传一直叽里咕噜问了半小时,感觉我吟唱太快了,面试官没听懂怎么个流程,没怎么对上脑电波。每个阶段的每个小细点都要细问一下分片丢了咋办,一层还行,多层套多层丢失我就晕了,不知道在问什么,只能说说终归到底是对账的思想。算法整了道非hot100,说了思路,让写个函数行了。这块也是聊天,嘴都没停过。你说是kpi吧,问了我一小时。你说不是,复盘我空茫然半天,也不知道到底是问了点啥。面试官很严肃,就是我总想笑。高情商来说,其实很考察了我有没有思考和临场发挥能力,有一种应试教育了半天突然给我塞了一口你要被快乐教育的感觉。反问: 说我简历挺好的,算法还得练,有一定思考但还不够深。
elliot19:
确实是这样,上次就纯聊天,问项目实习做啥,最后掏了一道非hot100的hard
第二天挂,今晚复活重新一面
查看6道真题和解析
点赞
评论
收藏
分享
04-21 11:47
北京工业大学 硬件开发
腾讯测开暑实一面,整理一下题目和感受
今天腾讯测试开发一面,27届找暑实,整理一下,有一题卡了说出来别重蹈。 【考点流水账】Linux日志查找:给一个日志文件,找出ERROR级别记录并统计次数。我用grep + awk写的,面试官没有异议。Python编程:写一个函数,判断一个字符串是否是合法的IP地址(四段数字,每段0-255)。我用split + 判断写的,面试官说逻辑没问题,问了有没有更简洁的写法(正则,我当时没想到,事后想到了……)SQL:联表查询,group by,统计某字段数量,难度不高。接口测试设计:测试一个登录接口,说说测试点。(我答了:正常流程/边界值/异常场景/安全测试,面试官说不错,但追问了:你说的"...
查看8道真题和解析
点赞
评论
收藏
分享
评论
点赞成功,聊一聊 >
点赞
收藏
分享
评论
提到的真题
返回内容
全站热榜
更多
1
...
和室友聊完我才知道自己输在信息差...
1.7W
2
...
面到崩溃后,我接受保底offer去旅游了
1.3W
3
...
字节抖音前端暑期二面
7535
4
...
从S到A,从夯到拉,ai项目盘点
5864
5
...
腾讯云智二面(已挂)
5247
6
...
同学去了腾讯、蔚来,而我去了一个小厂,后来我也...
5036
7
...
约不到面的日子
4474
8
...
招银网络科技春招
4474
9
...
ai项目要有差异化
4380
10
...
26届后端java/go求职心得
4261
创作者周榜
更多
正在热议
更多
#
哪些AI项目值得做?
#
15511次浏览
422人参与
#
秋招笔试记录
#
397532次浏览
2193人参与
#
华泰星战营,提前锁定校招offer
#
11554次浏览
353人参与
#
实习时最怕听到的一句话
#
14244次浏览
135人参与
#
90后北漂现状
#
38656次浏览
222人参与
#
找不到大厂实习可以去小厂吗?
#
12270次浏览
108人参与
#
机械人,说说你的烦心事
#
143914次浏览
1150人参与
#
应届生初入职场,求建议
#
332469次浏览
2916人参与
#
简历上如何体现你的“AI”能力?
#
7047次浏览
167人参与
#
你简历上最心虚的一句话
#
14471次浏览
154人参与
#
没有面试的日子里,你在做什么
#
8338次浏览
229人参与
#
携程笔试
#
162291次浏览
903人参与
#
如果有时光机,你最想去到哪个年纪?
#
77069次浏览
858人参与
#
你总挂在第__面?
#
5114次浏览
47人参与
#
ai智能作图
#
682380次浏览
5726人参与
#
汉得笔试
#
3893次浏览
23人参与
#
24届秋招同行攻略分享
#
1478659次浏览
14432人参与
#
你知道最慷慨和最抠的公司分别是
#
7164次浏览
59人参与
#
绿盟笔试
#
3400次浏览
24人参与
#
大厂无回复,继续等待还是奔赴小厂
#
356721次浏览
2024人参与
#
机械人还在等华为开奖吗?
#
333853次浏览
1628人参与
牛客网
牛客网在线编程
牛客网题解
牛客企业服务