【C++ 】智能指针:内存管理的 “自动导航仪”
目录
一、引入
二、智能指针的两大特性:
1、RAII
特点:
好处:
2、行为像指针
三、智能指针起初的缺陷:拷贝问题
四、几种智能指针的介绍。
1、C++98出现的智能指针——auto_ptr
auto_ptr解决上述拷贝构造的问题:
2、boost库
3、unique_ptr
引用计数的实现:
赋值运算符的问题:(循环引用)
5、weak_ptr
特点:
解决循环引用问题:
五、C++智能指针的基本框架:
六、定制删除器,以及包装器的使用场景之一
七、内存泄漏:
1、什么是内存泄漏,内存泄漏的危害:
2、内存泄漏的分类
八、关于C++智能指针的相关代码:
std::unique_ptr
std::weak_ptr
前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家
点击跳转到网站
一、引入
首先通过一个使用场景来引入智能指针,如下:
class HF { public:HF(int a, int b):_a(a),_b(b){cout << "HF(int a, int b)" << endl;};~HF(){cout << "~HF()" << endl;} private:int _a = 0;int _b = 1; };void fun() {HF* h1 = new HF(1, 1);HF* h2 = new HF(1, 1);HF* h3 = new HF(1, 1);delete h1;delete h2;delete h3; }int main() {try{fun();}catch (exception& e){cout << e.what() << endl;}return 0; }
这里有一个类HF,一个子函数fun,在fun里面new了三个HF对象,然后delete,正常情况下delete会先调用析构函数,然后再释放相应的资源:
二、智能指针的两大特性:
智能指针的两大特性:
1、RAII
2、行为像指针
1、RAII
是一种利用对象生命周期来控制程序资源(如内存、文件句柄、网络连接、互斥量等等)的简单技术。(通俗来讲,就是将资源交给一个类对象来管理,通过该类的构造函数交给对象。)
特点:
在对象构造时获取资源,接着控制对资源的访问使之在对象的生命周期内始终保持有效,最后在对象析构的时候释放资源。
好处:
(1)、不需要显式地释放资源,而是通过智能指针间接帮忙释放(2)、采用这种方式,对象所需的资源在其生命期内始终保持有效
2、行为像指针
智能指针实际也是一个类,要是行为像一个指针,即要重载解引用(*),箭头(->),甚至有时还要重载方括号([ ])。
三、智能指针起初的缺陷:拷贝问题
首先我们实现一个简易版智能指针:
new了一个日期类对象交给智能指针管理,智能指针对象存在期间,资源都是存在的,最后智能指针对象生命周期结束,调用析构函数释放,同时释放掉资源(delete);
template<class T> class SmartPtr { public:SmartPtr(T* ptr = nullptr): _ptr(ptr){}~SmartPtr(){if (_ptr)delete _ptr;}T& operator*() { return *_ptr; }T* operator->() { return _ptr; } private:T* _ptr; }; struct Date {int _year;int _month;int _day; }; int main() {SmartPtr<int> sp1(new int);*sp1 = 10;cout << *sp1 << endl;SmartPtr<Date> sparray(new Date);// 需要注意的是这里应该是sparray.operator->()->_year = 2018;// 本来应该是sparray->->_year这里语法上为了可读性,省略了一个->sparray->_year = 2018;sparray->_month = 1;sparray->_day = 1;return 0; }
但是当我们用对象sp1去拷贝构造sp2时:
此时就会报错:
原因在于我们没有实现拷贝构造,此时默认拷贝构造就是浅拷贝,这样两个对象的成员变量都会指向这份资源,最后调用析构函数时,就会对这份资源delete两次,从而造成野指针的问题。如何解决这个问题,在第四点进行介绍。
四、几种智能指针的介绍。
1、C++98出现的智能指针——auto_ptr
头文件:memory
具体信息可以查看官网文档。
auto_ptr解决上述拷贝构造的问题:
auto_ptr是直接将资源的管理权转移,用对象sp1去拷贝构造sp2,那么就会将sp1的资源的管理权交给sp2管理,而sp1被置空。
大致处理方法如下:拷贝后将sp1置空就行了:
注意,C++11的移动语义也是资源的转移,但和这里是不一样的,移动语义是针对将亡值去转移资源,而这里sp1不是将亡值。
这样做是有些问题的,这里资源转移后,sp1就悬空了,此时拷贝后就不能去访问sp1,否则就会出现空指针的问题,所以很多公司都禁止使用auto_ptr;
2、boost库
提到智能指针,就得提一下boost库,boost库是C++第三方库,里面就有智能指针,而C++的智能指针就是从这个库里面引入的,然后进行了略微修改。
3、unique_ptr
该智能指针解决拷贝构造的问题的方法就是:简单粗暴,禁止拷贝,适用于不需要拷贝的场景。
底层实际就是将拷贝构造给delete了:
同时,赋值运算符重载也要禁掉,默认生成的赋值运算符重载也是浅拷贝。
4、shared_ptr
当遇到需要拷贝构造的场景时,就需要使用shared_ptr,shared_ptr解决拷贝构造的问题的方法是:引用计数,去解决多次析构的问题。
引用计数的实现:
引用计数:记录当前有几个对象参与管理这个资源,在某个对象析构时,就将引用计数减1,当最后一个对象析构时才去释放资源。
要实现引用计数,就需要一份资源对应一个计数,有人会想到定义一个静态成员count,但实则不行,因为静态成员是属于整个类的,属于所有对象。管理一个资源的时候是可以解决的,但当第二个资源出现时,就不能适用了,因为不同资源之间的引用计数都是同一个静态成员变量,所以会相互影响。
实际上的实现如下:
增加一个成员变量*pcount,即指向引用计数的指针,在构造的时候(即资源来了),就new一个计数给该指针,在拷贝构造发生的时候,除了使两个对象指向同一个资源外,两个对象的引用计数也要指向同一个,并且要记得把引用计数++一下,在某个对象析构时,就将引用计数减1,然后判断是否为最后一个对象的析构,如果是的话就释放资源。
赋值运算符的问题:(循环引用)
shared_ptr虽然解决了拷贝构造的问题,但正因为引用计数的这样实现,又会造成赋值运算符重载后出现问题。
赋值运算符简单重载:
为了分析这里的缺陷,我们引入一个场景:双向链表的赋值:
这是双向链表的简单实现
因为会将链表资源交给智能指针管理,如下:
所以链表的定义中,成员next和prev的类型也应该是智能指针,不然在赋值的时候会出现类型不同的问题,正因为需要这样设计,问题就来了。
一般情况上述实现是没有问题的,但当执行下面两句代码后,问题就来了:
这是在链接两个节点,链接完后就会这样:
首先出现两个节点分别由n1和n2指向,此时两个节点的引用计数分别都是1,当执行n1->next = n2时,n2指向的节点的引用计数就会变成2;当执行n2->prve = n1时,n1指向的节点的引用计数就会变成2。
最后当析构链表时:
这样就形成了一个闭环,导致这两个节点的内存泄漏,这个问题也叫循环引用。当两个shared_ptr互相引用就会出现循环引用的问题。
5、weak_ptr
为了解决shared_ptr的循环引用问题,所以引入了weak_ptr。
特点:
weak_ptr的特点:没有引用计数,支持默认构造,构造函数的形参没有指针,因为该智能指针不参与资源管理,但自身成员变量会有一个指针,但会被置空,weak_ptr的重点在于拷贝构造和赋值。
解决循环引用问题:
这里的不同是将链表的成员变量_next和_prev的类型变为weak_ptr,正因为weak_ptr没有增加引用计数,所以在节点链接的时候,引用计数不会增加,所以节点会正常释放。
五、C++智能指针的基本框架:
六、定制删除器,以及包装器的使用场景之一
七、内存泄漏:
1、什么是内存泄漏,内存泄漏的危害:
什么是内存泄漏:内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制,因而造成了内存的浪费。内存泄漏的危害:长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死。
2、内存泄漏的分类
C++中我们一般关心两种分类:(1)、堆内存泄漏(Heap leak)堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉。假设程序的设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak。(2)、系统资源泄漏指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。
八、关于C++智能指针的相关代码:
此小点内容来源于:豆包AI
#include <iostream> #include <memory>// 自定义类 class MyClass { public:MyClass() { std::cout << "MyClass constructor" << std::endl; }~MyClass() { std::cout << "MyClass destructor" << std::endl; }void doSomething() { std::cout << "Doing something..." << std::endl; } };class B;class A { public:std::shared_ptr<B> bPtr;~A() { std::cout << "A destructor" << std::endl; } };class B { public:std::weak_ptr<A> aPtr; // 使用 weak_ptr 避免循环引用~B() { std::cout << "B destructor" << std::endl; } };void uniquePtrExample() {// 创建一个 unique_ptr 指向 MyClass 对象std::unique_ptr<MyClass> uniquePtr = std::make_unique<MyClass>();// 调用对象的成员函数uniquePtr->doSomething();// unique_ptr 不能被复制,但可以转移所有权// std::unique_ptr<MyClass> anotherPtr = uniquePtr; // 错误,不能复制std::unique_ptr<MyClass> anotherPtr = std::move(uniquePtr);if (!uniquePtr) {std::cout << "uniquePtr is empty after move" << std::endl;}if (anotherPtr) {anotherPtr->doSomething();} }void sharedPtrExample() {// 创建一个 shared_ptr 指向 MyClass 对象std::shared_ptr<MyClass> sharedPtr1 = std::make_shared<MyClass>();std::cout << "Shared pointer 1 use count: " << sharedPtr1.use_count() << std::endl;// 复制 shared_ptr,引用计数增加std::shared_ptr<MyClass> sharedPtr2 = sharedPtr1;std::cout << "Shared pointer 1 use count after copy: " << sharedPtr1.use_count() << std::endl;std::cout << "Shared pointer 2 use count: " << sharedPtr2.use_count() << std::endl;// 调用对象的成员函数sharedPtr2->doSomething();// 释放一个 shared_ptr,引用计数减少sharedPtr2.reset();std::cout << "Shared pointer 1 use count after reset sharedPtr2: " << sharedPtr1.use_count() << std::endl; }void weakPtrExample() {std::shared_ptr<A> a = std::make_shared<A>();std::shared_ptr<B> b = std::make_shared<B>();a->bPtr = b;b->aPtr = a;std::cout << "A use count: " << a.use_count() << std::endl;std::cout << "B use count: " << b.use_count() << std::endl;// 当 a 和 b 离开作用域时,对象会被正确销毁 }int main() {std::cout << "=== Unique Ptr Example ===" << std::endl;uniquePtrExample();std::cout << std::endl;std::cout << "=== Shared Ptr Example ===" << std::endl;sharedPtrExample();std::cout << std::endl;std::cout << "=== Weak Ptr Example ===" << std::endl;weakPtrExample();std::cout << std::endl;return 0; }
在 C++ 里,手动管理动态分配的内存容易引发内存泄漏、悬空指针等问题。智能指针作为一种类模板,能有效管理动态分配的内存,避免这些问题的出现。C++ 标准库提供了三种主要的智能指针:
std::unique_ptr
、std::shared_ptr
和std::weak_ptr
。
std::unique_ptr
std::unique_ptr
属于独占式智能指针,它对所指向的对象拥有唯一的所有权。一旦std::unique_ptr
被销毁,其指向的对象也会随之被自动销毁。在
uniquePtrExample
函数中:
- 借助
std::make_unique
创建了一个std::unique_ptr
,它指向MyClass
的一个对象。- 调用
doSomething
方法来使用这个对象。- 尝试复制
std::unique_ptr
会引发编译错误,因为它不允许复制,不过可以使用std::move
转移所有权。- 转移所有权之后,原
std::unique_ptr
变为空。
std::shared_ptr
std::shared_ptr
是共享式智能指针,多个std::shared_ptr
能够指向同一个对象。它采用引用计数来管理对象的生命周期,当引用计数变为 0 时,对象就会被销毁。在
sharedPtrExample
函数中:
- 利用
std::make_shared
创建了一个std::shared_ptr
,它指向MyClass
的一个对象。- 通过
use_count
方法可以查看当前的引用计数。- 复制
std::shared_ptr
会使引用计数增加。- 调用
reset
方法可以释放std::shared_ptr
,从而使引用计数减少。
std::weak_ptr
std::weak_ptr
是弱引用智能指针,它不拥有对象的所有权,只是对std::shared_ptr
所管理的对象进行弱引用。std::weak_ptr
主要用于解决std::shared_ptr
的循环引用问题。在
weakPtrExample
函数中:
- 定义了
A
和B
两个类,其中A
类包含一个std::shared_ptr<B>
成员,B
类包含一个std::weak_ptr<A>
成员。- 创建了
A
和B
的std::shared_ptr
对象,并相互引用。- 由于
B
类使用了std::weak_ptr
,所以不会出现循环引用,当a
和b
离开作用域时,对象能够被正确销毁。
相关文章:
【C++ 】智能指针:内存管理的 “自动导航仪”
目录 一、引入 二、智能指针的两大特性: 1、RAII 特点: 好处: 2、行为像指针 三、智能指针起初的缺陷:拷贝问题 四、几种智能指针的介绍。 1、C98出现的智能指针——auto_ptr auto_ptr解决上述拷贝构造的问题:…...
macOS 上使用 Homebrew 安装和配置 frp 客户端
macOS 上使用 Homebrew 安装和配置 frp 客户端 (frpc) 指南 frp (Fast Reverse Proxy) 是一款高性能的反向代理应用,常用于内网穿透。本文将介绍在 macOS 上使用 Homebrew 安装 frpc,并进行配置和管理。 一、安装 frpc 使用 Homebrew 安装(…...
OSI参考模型
1.1 OSI参考模型 OSI(Open System Interconnection,开放式系统互联),由ISO(International Organization for Standardization,国际标准化组织)收录在ISO 7489标准中并于1984年发布。OSI参考模…...
《探秘鸿蒙分布式软总线:开启无感发现与零等待传输新时代》
在数字化浪潮中,设备之间的互联互通成为构建智能生态的关键。鸿蒙系统中的分布式软总线技术,宛如一座桥梁,让各种智能设备紧密相连。尤其是其实现的设备间无感发现和零等待传输功能,更是为用户带来了前所未有的便捷体验࿰…...
QML ListView:列表视图的数据交互与样式定制
目录 引言相关阅读项目结构示例一:ListView样式定制代码解析运行效果 示例二:ListView数据交互代码解析运行效果 总结 引言 在现代应用程序开发中,列表视图是最常见且实用的UI组件之一。Qt Quick(QML)提供了强大的Lis…...
C#集合List<T>与HashSet<T>的区别
在C#中,List和HashSet都是用于存储元素的集合,但它们在内部实现、用途、性能特性以及使用场景上存在一些关键区别。 内部实现 List:基于数组实现的,可以包含重复的元素,并且元素是按照添加的顺序存储的。 HashSet&…...
【C++进阶六】list模拟实现
【C进阶六】list模拟实现 1.list的大致框架结构2.节点类3. 迭代器4.list内各种功能的实现1.push_back(正常实现)2.insert3.begin()和end()4.push_back和push_front(复用insert)5.erase6. pop_back与pop_front (复用erase)7.clear 清空数据8.swap9.无参构造10.析构函数11.含参构…...
Spring 事件机制与观察者模式的深度解析
一、引言 在软件设计中,观察者模式(Observer Pattern)是一种非常经典且实用的设计模式。它允许一个对象(Subject)在状态发生改变时通知所有依赖它的对象(Observers),从而实现对象之…...
危险作业自动化如何重构工业安全新生态
在化工车间的反应器旁,巡检员正戴着防毒面具靠近高温罐体;在矿山深处,凿岩工人在粉尘弥漫中操控机械臂;在高架桥梁上,检修人员正攀爬至数十米高空排查隐患…… 这些场景中,每一个动作都伴随着不可预估的安全…...
Trae 下安装 Pylance 插件(仅作为实验,版权由微软所有)
目录 背景0. Trae 环境准备1. 从VSCode市场获取插件安装包2. 直接安装插件至Trae会失败3. 解压 vsix 文件4. 直接移动安装参考文献: 背景 基于 VSCode 改版但使用 Open VSX 的编辑器(如 trae、Cursor)会遇到以下限制1: 微软官方…...
多模态大语言模型arxiv论文略读(十八)
Large Model Based Referring Camouflaged Object Detection ➡️ 论文标题:Large Model Based Referring Camouflaged Object Detection ➡️ 论文作者:Shupeng Cheng, Ge-Peng Ji, Pengda Qin, Deng-Ping Fan, Bowen Zhou, Peng Xu ➡️ 研究机构: Ts…...
oracle查询锁表和解锁
oracle查询锁表和解锁 1.数据库表被锁后------解锁第一步第二步 2.查锁语句 1.数据库表被锁后------解锁 第一步 select t2.username, t2.sid, serial#, t2.logon_time from v l o c k e d o b j e c t t 1 , v locked_object t1, v lockedobjectt1,vsession t2 where t1.S…...
mysql 数据库localhost密码忘记
使用此查询语句: SELECT user, authentication_string FROM mysql.user WHERE user root; 复制对应的密码: 密码是通过md5加密后的 md5在线解密破解,md5解密加密 将密码输入进来 就可以直接破解了...
如何把pdf的内容转化成结构化数据进行存储到mysql数据库
一、PDF解析与数据提取 文本提取工具选择 使用Python的pdfplumber或PyPDF2库进行基础文本提取。 pythonCopy Code import pdfplumber def extract_text(pdf_path): with pdfplumber.open(pdf_path) as pdf: return [page.extract_text() for page in pdf.pages if page.e…...
豪越消防一体化安全管控平台:构建消防“一张图”新生态
在城市化进程加速、建筑规模与功能日益复杂的当下,消防救援工作面临着诸多严峻挑战。火灾隐患如同隐藏在暗处的“定时炸弹”,广泛分布于城市的各个角落,想要快速、精准定位绝非易事。信息传递的不顺畅更是雪上加霜,导致救援效率大…...
中和农信:以数字化与普惠理念重塑“三农”服务新生态
在乡村振兴与农业现代化的时代命题下,中国农村市场既承载传统生产模式效率不足的挑战,也面临着数字化变革带来的机遇。作为深耕农村市场多年的综合服务机构,中和农信完成了从单一小额信贷机构向综合型“三农”服务平台的跨越式转型࿰…...
Docker详细使用
Docker详细使用 文章目录 Docker详细使用使用场景docker安装常用命令帮助启动类命令镜像命令网络命令容器命令compose(服务编排) 功能列表存储(挂载本地)介绍使用⽬录挂载卷映射 网络介绍使用 DockerfileCompose介绍使用 使用场景…...
[福游宝——AI智能旅游信息查询平台]全栈AI项目-阶段二:聊天咨询业务组件开发
简言 本项目旨在构建一个以AI智能体为核心的福建省旅游信息查询系统,聚焦景点推荐、路线规划、交通天气查询等功能,为游客提供智能化、便捷化的旅游信息服务。项目采用前后端分离架构,前端基于Vite TypeScript Vue3技术栈,搭配…...
【Linux】进程的详讲(下)--进程的环境变量
目录 📖一、进程优先级 📖二、查看进程优先级 📖三、环境变量 📖四、环境变量PATH:Linux系统中的指令搜索路径 📖五、环境变量HOME 📖六、环境变量USER 📖七、命令行参数 …...
深入理解 MVC 模式在 C# 中的应用
MVC(Model-View-Controller)是一种经典的软件设计模式,广泛应用于现代应用程序开发中,尤其是在 Web 应用程序和桌面应用程序中。它通过将应用程序的不同职责进行分离,有助于提高代码的可维护性、可扩展性以及可测试性。…...
《鸿蒙软总线:基于UDP的数据传输奥秘与优势》
在鸿蒙系统构建的万物互联世界里,分布式软总线扮演着至关重要的角色,尤其是其基于UDP协议的数据传输机制,成为支撑多设备高效协同的关键技术。深入剖析这一技术的原理与优势,有助于我们理解鸿蒙系统如何实现设备间的无缝连接与流畅…...
关于 驱动开发方法 的详细分类、核心特点及对比分析,涵盖 TDD、MDD、BDD、DDD、ATDD、FDD、PDD 等主流方法
以下是关于 驱动开发方法 的详细分类、核心特点及对比分析,涵盖 TDD、MDD、BDD、DDD、ATDD、FDD、PDD 等主流方法: 一、驱动开发方法分类及详解 1. 测试驱动开发(TDD: Test-Driven Development) 定义:通过编写测试用…...
考研数据结构之二叉树(二):二叉树的遍历与线索二叉树(包含真题实战)
考研数据结构之二叉树(二):二叉树的遍历与线索二叉树 在上一篇文章中,我们详细探讨了二叉树的定义和存储结构。本文将深入讲解二叉树的核心操作——遍历,以及其重要应用形式之一——线索二叉树。这些内容不仅是考研的…...
Guava Cache的refreshAfterWrite机制
Guava Cache 的 refreshAfterWrite 机制在源码中通过 惰性刷新调度 和 细粒度锁控制 实现,核心逻辑集中在 LocalCache 类的 Segment 结构中。以下是关键源码解析: 一、核心数据结构 1. Segment 类 • 继承自 ReentrantLock:每个 Segment 独…...
邮件发送频率如何根据用户行为动态调整?
邮件发送频率,这可是个技术活儿。发多了,用户烦;发少了,机会溜。别担心,今天就给你支几招,教你如何根据用户行为,像变魔术一样灵活调整邮件发送频率。 一、盯紧用户行为数据 先别急着发邮件&a…...
HDMI与DVI接口热插拔检测
在当今数字音视频传输中,高清多媒体接口与数字视觉接口已成为主流标准。无论是电脑、蓝光播放器,还是电视与显示器,这两种接口都发挥着重要作用。而在保证系统“即插即用”和自动识别的过程中,热插拔检测(HPD)承担着关键角色。研究表明,HDMI 的第 19 引脚和 DVI 的第 16…...
Redis,RESP协议,阻塞IO 与非阻塞IO,Redis的线程模型
1.Redis 阻塞IO 与非阻塞IO Java在JDK1.4 中引入了NIO ,但是也有很多人在使用阻塞IO,这两种IO有什么区别? 在阻塞模式下,如果你从数据流读取不到指定大小的数据量,IO就会阻塞。 比如: 已知会有10个字节发送过来,但是我…...
Express学习笔记(六)——前后端的身份认证
目录 1. Web 开发模式 1.1 服务端渲染的 Web 开发模式 1.2 服务端渲染的优缺点 1.3 前后端分离的 Web 开发模式 1.4 前后端分离的优缺点 1.5 如何选择 Web 开发模式 2. 身份认证 2.1 什么是身份认证 2.2 为什么需要身份认证 2.3 不同开发模式下的身份认证 3. Sessio…...
DDoS攻防实战指南——解析企业级防护五大解决方案
一、流量清洗中心的智能化演进 云清洗服务已从被动响应转向主动防御。基于全球Anycast网络的分布式清洗节点,可在攻击发生时将流量牵引至专用清洗集群。阿里云2023年实测数据显示,其新一代清洗设备对SYN Flood的识别准确率达99.97%,误杀率控…...
2025年机电一体化、机器人与人工智能国际学术会议(MRAI 2025)
重要信息 时间:2025年4月25日-27日 地点:中国济南 官网:http://www.icmrai.org 征稿主题 机电一体化机器人人工智能 传感器和执行器 3D打印技术 智能控制 运动控制 光电系统 光机电一体化 类人机器人 人机界面 先进的运动控制 集成制造系…...
QT Sqlite数据库-教程002 查询数据-上
【1】DQL语句: DQL语句(数据查询语言),用来查询数据记录。DQL 基本结构由 SELECT FROM、WHERE、JOIN 等子句构成。DQL 语句并不会改变数据库,而是让数据库将查询结果发送结果集给客户端,返回的结果是一张虚…...
Java List流式编程全解析:从入门到实战高手
🚀 Java List流式编程全解析:从入门到实战高手 #Java8新特性 #Stream流 #集合操作 #高效编程 一、为什么需要Stream流? 传统集合操作痛点: 代码冗长:多层循环嵌套,可读性差难以并行:手动拆分…...
【Linux】深入理解线程控制
个人主页~ 深入理解线程控制 一、线程等待的原理二、线程的局部存储三、初步理解线程互斥1、互斥的概念2、需要互斥的原因 一、线程等待的原理 pthread_join的作用是线程等待,其中retval参数传递线程退出状态的原理是:当目标线程结束时,pthr…...
android面试情景题详解:android如何处理断网、网络切换或低速网络情况下的业务连续性
在移动互联网时代,Android应用已经成为人们日常生活中不可或缺的一部分。从社交媒体到在线购物,从移动办公到娱乐消费,几乎所有的服务都依赖于网络连接。然而,网络环境并非总是稳定可靠。断网、网络切换(如从Wi-Fi切换…...
基于 Qt 的 BMP 图像数据存取至 SQLite 数据库的实现
基于 Qt 的 BMP 图像数据存取至 SQLite 数据库的实现说明 本项目通过 Qt 框架实现了将 BMP 图像文件以二进制形式存入 SQLite 数据库,并可从数据库中读取还原为 BMP 图像文件的功能,适用于需要图像与结构化数据统一管理的场景。 整个流程分为两个主要部…...
Melos 发布pub.dev
🧰 Melos 全套实用教程(Flutter Monorepo 管理神器) 🚀 目录: 什么是 Melos?适合哪些项目? 安装与初始化 项目结构推荐 melos.yaml 配置详解 常用命令讲解(bootstrap、run、exe…...
基于 SSM 高校二手交易平台
收藏关注不迷路!! 🌟文末获取源码数据库🌟 感兴趣的可以先收藏起来,还有大家在毕设选题(免费咨询指导选题),项目以及论文编写等相关问题都可以给我留言咨询,希望帮助更多…...
Missashe考研日记-day20
Missashe考研日记-day20 1 高数 学习时间:2h30min学习内容: 今天当然是刷题啦,做不等式的证明板块的真题,证明题懂的都懂,难起来是真的一点思路都没有,这个板块还没做完,做完再总结题型。 2…...
Spring 单元测试核心注解全解:@InjectMocks、@MockBean、@Mock、@Autowired 的区别与实战
在编写 Spring Boot 应用的单元测试过程中,@InjectMocks、@MockBean、@Mock 和 @Autowired 是最常用的几个注解,但它们经常被混淆或误用,导致测试失败或注入错误。 本文将从本质区别、使用场景、示例代码、对比表格等多个维度,全面解析这几者的使用方法与差异,助你写出结…...
Spring Boot 项目里设置默认国区时区,Jave中Date时区配置
在 Spring Boot 项目里设置国区时区(也就是中国标准时间,即 Asia/Shanghai),可通过以下几种方式实现: 方式一:在application.properties或application.yml里设置 application.properties properties sp…...
如何使用Cloud Sync搭建群晖NAS跨设备自动备份与手机端实时监控
文章目录 前言1. 群晖NAS安装Cloud Sync1.1 安装和配置 Cloud Sync 2. 功能演示3. 群晖安装cpolar内网穿透3.1 配置Cloud Sync公网访问地址 4. 配置固定公网地址 前言 在当今这个信息爆炸的时代,个人和企业都面临着一个共同的难题:数据分散在各种设备和…...
git回退到指定版本
查看提交历史 在本地仓库中运行 git log 命令,找到需要回退到的版本对应的 commit ID。 重置本地分支 执行 git reset --hard commit_id 将本地分支回退至目标版本,commit_id为要会退的版本号。 强制推送更改 运行 git push origin HEAD --force 强制更…...
【Python爬虫】简单介绍2
目录 四、网页数据的加载方式 4.1 数据直接放在请求响应结果中 4.2 数据在异步请求响应结果中 五、爬虫 - 反爬虫 - 反反爬虫 5.1 反爬虫 5.2 反反爬虫 六、风险 6.1 合法性与合规性的重要性 6.2 违规使用爬虫面临的法律风险示例 6.3 合法合规使用爬虫的建议 四、网页…...
基于QtC++音乐播放器whisper语音转文字歌词解析
演示视频 github源码地址 gitee源码地址 通过网盘分享的文件:MySoftWare 链接: https://pan.baidu.com/s/1JdtZNoMcv7jXR_ELMuqZEg?pwd1yw6 提取码: 1yw6 –来自百度网盘超级会员v4的分享...
【QT】QT界面的美容院 -- QSS
一、背景介绍 🔥 在网页前端开发领域中,CSS 是一个至关重要的部分,描述了一个网页的 “样式”,从而起到对网页 美化 的作用。 所谓 样式 ,包括不限于大小、位置、颜色、背景、间距、字体等等。现在的网页很难找到没有…...
【AI】使用Huggingface模型实现文本内容摘要器
【AI】使用Huggingface模型实现文本内容摘要器 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 【AI】使用Huggingface模型实现文本内容摘要器什么是摘要器?摘要器的应用场景什么是…...
centOS 安装和配置docker
以下是在 CentOS 系统上安装和配置 Docker 的详细步骤: 一、安装 Docker 1. 卸载旧版本(如有) sudo yum remove docker \docker-client \docker-client-latest \docker-common \docker-latest \docker-latest-logrotate \docker-logrotate …...
现在AI大模型能帮做数据分析吗?
实际上,有了AI大模型的支持才使得AI数据分析的功能得到更好地应用,比如小浣熊AI支持数据交互功能,只要你输入具体的数据分析要求,它就能自动帮你完成数据清洗、分析、可视化全流程,而且还能生成数据分析报告。 当然&a…...
qt中,父类中有Q_OBJECT,子类中还需要加Q_OBJECT吗
在 Qt 中,关于子类是否需要添加 Q_OBJECT 宏的问题,可以总结如下: 1. 需要添加 Q_OBJECT 的情况 如果子类满足以下任一条件,必须显式添加 Q_OBJECT 宏: 定义了新的信号或槽:即使父类已有 Q_OBJECT&#…...
vue2实现在屏幕中有一个小机器人可以随意移动
第一步:创建store目录结构 src/ ├── store/ │ ├── modules/ │ │ └── robot.js # 机器人专用状态模块 │ └── index.js # Vuex 主配置文件第二步:创建机器人状态模块 创建 src/store/modules/robot.js 文件ÿ…...