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

C++智能指针:高效管理内存的利器

上篇文章C算法一维/二维前缀和算法模板题目录1.智能指针的使用场景2.RAII和智能指针的设计思路3.C标准库智能指针的使用4.智能指针的原理5.shared_ptr和weak_ptr5.1shared_ptr循环引用问题5.2weak_ptr6.shared_ptr的线程安全问题7.C11和boost中智能指针的关系8.内存泄漏8.1什么是内存泄漏及其危害8.2检测内存泄漏8.3如何避免内存泄漏1.智能指针的使用场景如下方的代码new了之后我们也delete了但由于抛异常导致之后的delete并没有执行因此产生了内存泄漏问题。所以我们需要new以后捕获异常捕获到异常后delete内存再把异常抛出不过由于new本身也可能抛异常连续两个new和下面的Divide都可能会抛异常会让我们处理起来很麻烦。而智能指针放在这样的场景中就会让问题简单很多。double Divide(int a, int b) { // 当b0时抛出异常 if (b 0) { throwDivide by zero condition; } else { return (double)a / (double)b; } } void Func() { // 这里可以看到如果发生除0错误抛出异常另外下面的array和array2没有得到释放。 // 所以这里捕获异常后并不处理异常异常还是交给外面处理这里捕获了再重新抛出去。 // 但是如果array2 new的时候抛异常就还需要套一层捕获释放逻辑这里更好解决方案 // 是智能指针 int* array1 new int[10]; int* array2 new int[10]; // 如果抛异常呢 try { int len, time; cin len time; cout Divide(len, time) endl; } catch (...) { cout delete [] array1 endl; cout delete [] array2 endl; delete[] array1; delete[] array2; throw; // 异常重新抛出捕获到说明就抛出什么 } // ... cout delete [] array1 endl; delete[] array1; cout delete [] array2 endl; delete[] array2; } int main() { try { Func(); } catch (const char* errmsg) { cout errmsg endl; } catch (const exception e) { cout e.what() endl; } catch (...) { cout 未知异常 endl; } return 0; }2.RAII和智能指针的设计思路RAll是Resource Acquisition Is Initialization的缩写他是一种管理资源的类的设计思想本质是一种利用对象生命周期来管理获取到的动态资源避免资源泄漏这里的资源可以是内存、文件指针、网络连接、互斥锁等等。RAII在获取资源时把资源委托给一个对象接着控制对资源的访问资源在对象的生命周期内始终保持有效最后在对象析构的时候释放资源这样保障了资源的正常释放避免资源泄漏问题。智能指针类除了满足RAII的设计思路还要方便资源的访问所以智能指针类还会像迭代器类一样重载operator* / operator- / operator [ ] 等运算符方便访问资源。template class T class SmartPtr { public: // RAII SmartPtr(T* ptr) :_ptr(ptr) { } ~SmartPtr() { cout delete[] _ptr endl; delete[] _ptr; } // 重载运算符模拟指针行为方便访问资源 T* operator-() { return _ptr; } T operator*() { return *_ptr; } T operator[](size_t i) { return _ptr[i]; } private: T* _ptr; }; double Divide(int a, int b) { // 当b 0时抛出异常 if (b 0) { throwDivide by zero condition; } else { return (double)a / (double)b; } } void Func() { // 这里使用RAII的智能指针类管理new出来的数组 SmartPtrint sp1 new int[10]; SmartPtrint sp2 new int[10]; for (size_t i 0; i 10; i) { sp1[i] sp2[i] i; } int len, time; cin len time; cout Divide(len, time) endl; } int main() { try { Func(); } catch (const char* errmsg) { cout errmsg endl; } catch (const exception e) { cout e.what() endl; } catch (...) { cout 未知异常 endl; } return 0; }3.C标准库智能指针的使用C标准库中的智能指针都在 memory 这个头文件下面我们包含 memory 就可以使用了智能指针有好几种除了weak_ptr他们都符合RAII和像指针一样访问的行为原理上而言主要是解决智能指针拷贝时的思路不同。auto_ptr是C98时设计出来的智能指针他的特点是拷贝时把被拷贝对象的资源的管理权转让给拷贝对象这是一个非常糟糕的设计因为他会到被拷贝对象悬空访问报错的问题C11设计出新的智能指针后强烈建议不要使用auto_ptr。其他C11出来之前很多公司也是明令禁止使用这个智能指针的。struct Date { public: int _year; int _month; int _day; Date(int year 1, int month 1, int day 1) :_year(year) ,_month(month) ,_day(day) { cout A() endl; } ~Date() { cout ~Date() endl; } }; int main() { // C17标准直接移除auto_ptr //auto_ptrDate ap1(new Date); // 拷贝时管理权限转移被拷贝对象ap1悬空 //auto_ptrDate ap2(ap1); // 空指针访问ap1对象已经悬空 //ap1-_year; return 0; }unique_ptr是C11设计出来的智能指针他的名字翻译出来是唯一指针他的特点是不支持拷贝只支持移动。如果不需要拷贝的场景就非常建议使用他。int main() { unique_ptrDate up1(new Date); // 不支持拷贝 //unique_ptrDate up2(up1); // 支持移动但是移动后up1也悬空 unique_ptrDate up3(move(up1)); return 0; }shared_ptr是C11设计出来的智能指针他的名字翻译出来就是共享指针他的特点是支持拷贝也支持移动。如果需要拷贝的场景就需要使用他。底层是用引用计数的方式实现的。int main() { shared_ptrDate sp1(new Date); // 支持拷贝 shared_ptrDate sp2(sp1); shared_ptrDate sp3(sp2); cout sp1.use_count() endl; sp1-_year; cout sp1-_year endl; cout sp2-_year endl; cout sp3-_year endl; // 支持移动但是移动后sp1也悬空 shared_ptrDate sp4(move(sp1)); return 0; }weak_ptr是C11设计出来的智能指针他的名字翻译出来是弱指针他完全不同于上面的智能指针他不支持RAII也就意味着不能用它直接管理资源weak_ptr的产生本质是要解决shared_ptr的一个循环引用导致内存泄漏问题。智能指针析构时默认是进行delete释放资源的也就意味着如果不是new出来的资源交给智能指针管理析构时就会崩溃。智能指针支持在构造时给一个删除器其本质就是一个可调用对象这个可调用对象中实现你想要的释放资源的方式当构造智能指针时给了定制的删除器在智能指针析构时就会调用删除器去释放资源。因为new[]经常使用所以为了简洁unique_ptr和shared_ptr都特化了一份[]的版本使用时unique_ptrDate[ ] up1(new Date[5]); shared_ptrDate[] sp1(new Date[5]); 就可以管理new[]的资源。templateclass T void DeleteArrayFunc(T* ptr) { delete[] ptr; } templateclass T class DeleteArray { public: void operator()(T* ptr) { delete[] ptr; } }; class Fclose { public: void operator()(FILE* ptr) { cout fclose: ptr endl; fclose(ptr); } }; int main() { // 这样实现程序会崩溃 // unique_ptrDate up1(new Date[10]); // shared_ptrDate sp1(new Date[10]); // 解决方案1 // 因为new[]经常使用所以unique_ptr和shared_ptr // 实现了⼀个特化版本这个特化版本析构时用的delete[] unique_ptrDate[] up1(new Date[5]); shared_ptrDate[] sp1(new Date[5]); // 解决方案2 // 仿函数对象做删除器 unique_ptrDate, DeleteArrayDate up2(new Date[5], DeleteArrayDate()); // unique_ptr和shared_ptr支持删除器的方式有所不同 // unique_ptr是在类模板参数支持的shared_ptr是构造函数参数支持的 // 这里没有使用相同的方式还是挺坑的 // 使用仿函数unique_ptr可以不在构造函数传递因为仿函数类型构造的对象直接就可以调用 // 但是下面的函数指针和lambda的类型不可以 unique_ptrDate, DeleteArrayDate up3(new Date[5]); shared_ptrDate sp2(new Date[5], DeleteArrayDate()); // 函数指针做删除器 unique_ptrDate, void(*)(Date*) up4(new Date[5], DeleteArrayFuncDate); shared_ptrDate sp3(new Date[5], DeleteArrayFuncDate); // lambda表达式做删除器 auto delArrOBJ [](Date* ptr) {delete[] ptr; }; unique_ptrDate, void(*)(Date*) up5(new Date[5], DeleteArrayFuncDate); shared_ptrDate sp4(new Date[5], delArrOBJ); // 实现其他资源管理的删除器 shared_ptrFILE sp5(fopen(Test.cpp, r), Fclose()); shared_ptrFILE sp6(fopen(Test.cpp, r), [](FILE* ptr) { cout fclose: ptr endl; fclose(ptr); }); return 0; }template class T, class... Argsshared_ptrTmake_shared(Args... args);shared_ptr 除了支持用指向资源的指针构造还支持make_shared用初始化资源对象的值直接构造。shared_ptr 和unique_ptr 都支持了operator bool的类型转换如果智能指针对象是一个空对象没有管理资源则返回false否则返回true意味着我们可以直接把智能指针对象给if判断是否为空。shared_ptr 和unique_ptr 都得构造函数都使用explicit 修饰防止普通指针隐式类型转换成智能指针对象。int main() { shared_ptrDate sp1(new Date(2026, 3, 3)); shared_ptrDate sp2 make_sharedDate(2026, 3, 3); auto sp3 make_sharedDate(2024, 9, 11); shared_ptrDate sp4; // if (sp1.operator bool()) if (sp1) cout sp1 is not nullptr endl; if (!sp4) cout sp1 is nullptr endl; // 报错 //shared_ptrDate sp5 new Date(2024, 9, 11); //unique_ptrDate sp6 new Date(2024, 9, 11); return 0; }4.智能指针的原理模拟实现auto_ptr和unique_ptr的核心功能这两个智能指针的实现比较简单。auto_ptr的思路是拷贝时转移资源管理权给被拷贝对象这种思路是不被认可的也不建议使用。unique_ptr的思路是不支持拷贝。大家重点要看看shared_ptr是如何设计的尤其是引用计数的设计主要这里一份资源就需要一个引用计数所以引用计数才用静态成员的方式是无法实现的要使用堆上动态开辟的方式构造智能指针对象时来一份资源就要new一个引用计数出来。多个shared_ptr指向资源时就引用计数shared_ptr对象析构时就--引用计数引用计数减到0时代表当前析构的shared_ptr是最后一个管理资源的对象则析构资源。auto_ptrtemplateclass T class auto_ptr { public: auto_ptr(T* ptr) :_ptr(ptr) { } auto_ptr(auto_ptrT sp) :_ptr(sp._ptr) { sp._ptr nullptr; } auto_ptrT operator(auto_ptrT ap) { // 检测是否为自己给自己赋值 if (this ! ap) { // 释放当前对象中的资源 if (_ptr) delete _ptr; // 转移ap中资源到当前对象中 _ptr ap._ptr; ap._ptr NULL; } return *this; } ~auto_ptr() { if (_ptr) { cout delete: _ptr endl; delete _ptr; } } T operator*() { return *_ptr; } T* operator-() { return _ptr; } private: T* _ptr; };unique_ptrtemplateclass T class unique_ptr { public: explicit unique_ptr(T* ptr) :_ptr(ptr) { } ~unique_ptr() { if (_ptr) { cout delete: _ptr endl; delete _ptr; } } T operator*() { return *_ptr; } T* operator-() { return _ptr; } unique_ptr(const unique_ptrT up) delete; unique_ptrT operator(const unique_ptrT up) delete; unique_ptr(unique_ptrT sp) :_ptr(sp._ptr) { sp._ptr nullptr; } unique_ptrT operator(unique_ptrT sp) { delete _ptr; _ptr sp._ptr; sp._ptr nullptr; } private: T* _ptr; };shared_ptrtemplateclass T class shared_ptr { public: explicit shared_ptr(T* ptr nullptr) :_ptr(ptr) ,_pcount(new int(1)) { } templateclass D shared_ptr(T* ptr, D del) : _ptr(ptr) , _pcount(new int(1)) ,_del(del) { } shared_ptr(const shared_ptrT sp) :_ptr(sp._ptr) , _pcount(sp._pcount) , _del(sp._del) { (*_pcount); } void release() { if (--(*_pcount) 0) { // 最后一个管理的对象释放资源 _del(_ptr); delete _pcount; _ptr nullptr; _pcount nullptr; } } shared_ptrT operator(const shared_ptrT sp) { if (_ptr ! sp._ptr) { release(); _ptr sp._ptr; _pcount sp._pcount; (*_pcount); _del sp._del; } return *this; } ~shared_ptr() { release(); } T* get() const { return _ptr; } int use_count() const { return *_pcount; } T operator*() { return *_ptr; } T* operator-() { return _ptr; } private: T* _ptr; int* _pcount; //atomicint* _pcount; functionvoid(T*) _del [](T* ptr) {delete ptr; }; };weak_ptrtemplateclass T class weak_ptr { public: weak_ptr() { } weak_ptr(const shared_ptrT sp) :_ptr(sp.get()) { } weak_ptrT operator(const shared_ptrT sp) { _ptr sp.get(); return *this; } private: T* _ptr nullptr; };5.shared_ptr和weak_ptr5.1shared_ptr循环引用问题shared_ptr大多数情况下管理资源非常合适支持RAll也支持拷贝。但是在循环引用的场景下会导致资源没得到释放内存泄漏所以我们要认识循环引用的场景和资源没释放的原因并且学会使用weak_ptr解决这种问题。如下图所述场景n1和n2析构后管理两个节点的引用计数减到1右边的节点什么时候释放是由左边节点中的_next管着_next析构后右边的节点就释放了。_next什么时候析构 _next是左边节点的的成员左边节点释放_next就析构了。左边节点什么时候释放 左边节点由右边节点中的_prev管着_prev析构后左边的节点就释放了。_prev什么时候析构 _prev是右边节点的成员右边节点释放_prev就析构了。至此逻辑上成功形成回旋镖似的循环引用谁都不会释放就形成了循环引用导致内存泄漏把ListNode结构体中的_next和_prev改成weak_ptrweak_ptr绑定到shared_ptr时不会增加它的引用计数_next和_prev不参与资源释放管理逻辑就成功打破了循环引用解决了这里的问题struct ListNode { int _data; //std::shared_ptrListNode _next; //std::shared_ptrListNode _prev; // 这里改为weak_ptr当n1-_next n2;绑定shared_ptr时 // 不增加n2的引用计数不参与资源释放的管理就不会形成循环引用了 std::weak_ptrListNode _next; std::weak_ptrListNode _prev; ~ListNode() { cout ~ListNode() endl; } }; int main() { // 循环引用 -- 内存泄漏 std::shared_ptrListNode n1(new ListNode); std::shared_ptrListNode n2(new ListNode); cout n1.use_count() endl; cout n2.use_count() endl; n1-_next n2; n2-_prev n1; cout n1.use_count() endl; cout n2.use_count() endl; // weak_ptr不支持管理资源不支持RAII // weak_ptr是专⻔绑定shared_ptr不增加他的引用计数作为⼀些场景的辅助管理 //std::weak_ptrListNode wp(new ListNode); return 0; }5.2weak_ptrweak_ptr不支持RAll也不支持访问资源所以我们看文档发现weak_ptr构造时不支持绑定到资源只支持绑定到shared_ptr绑定到shared_ptr时不增加shared_ptr的引用计数那么就可以解决上述的循环引用问题。weak_ptr也没有重载operator*和operator-等因为他不参与资源管理那么如果他绑定的shared_ptr已经释放了资源那么他去访问资源就是很危险的。weak_ptr支持expired检查指向的资源是否过期use_count也可获取shared_ptr的引用计数weak_ptr想访问资源时可以调用lock返回一个管理资源的shared_ptr如果资源已经被释放返回的shared_ptr是一个空对象如果资源没有释放则通过返回的shared_ptr访问资源是安全的。int main() { std::shared_ptrstring sp1(new string(111111)); std::shared_ptrstring sp2(sp1); std::weak_ptrstring wp sp1; cout wp.expired() endl; cout wp.use_count() endl; // sp1和sp2都指向了其他资源则weak_ptr就过期了 sp1 make_sharedstring(222222); cout wp.expired() endl; cout wp.use_count() endl; sp2 make_sharedstring(333333); cout wp.expired() endl; cout wp.use_count() endl; wp sp1; //std::shared_ptrstring sp3 wp.lock(); auto sp3 wp.lock(); cout wp.expired() endl; cout wp.use_count() endl; *sp3 ###; cout *sp1 endl; return 0; }结果6.shared_ptr的线程安全问题shared_ptr的引用计数对象在堆上如果多个shared_ptr对象在多个线程中进行shared_ptr的拷贝析构时会访问修改引用计数就会存在线程安全问题所以shared_ptr引用计数是需要加锁或者原子操作保证线程安全的。shared_ptr指向的对象也是有线程安全的问题的但是这个对象的线程安全问题不归shared_ptr管它也管不了应该有外层使用shared_ptr的人进行线程安全的控制。下面的程序会崩溃或者A资源没释放xxx::shared_ptr引l用计数从int*改成atomicint*就可以保证引用计数的线程安全问题或者使用互斥锁加锁也可以。struct AA { int _a1 0; int _a2 0; ~AA() { cout ~AA() endl; } }; int main() { xxx::shared_ptrAA p(new AA); const size_t n 100000; mutex mtx; auto func []() { for (size_t i 0; i n; i) { // 这⾥智能指针拷⻉会计数 xxx::shared_ptrAA copy(p); { unique_lockmutex lk(mtx); copy-_a1; copy-_a2; } } }; thread t1(func); thread t2(func); t1.join(); t2.join(); cout p-_a1 endl; cout p-_a2 endl; cout p.use_count() endl; return 0; }7.C11和boost中智能指针的关系Boost库是为C语言标准库提供扩展的一些C程序库的总称Boost社区建立的初衷之一就是为C的标准化工作提供可供参考的实现Boost社区的发起人Dawes本人就是C标准委员会的成员之一。在Boost库的开发中Boost社区也在这个方向上取得了丰硕的成果C11及之后的新语法和库有很多都是从Boost中来的。C98中产生了第一个智能指针auto_ptr。C boost给出了更实用的scoped_ptr/scoped_array和shared_ptr/shared_array和weak_ptr等CTR1引入了shared_ptr等不过注意的是TR1并不是标准版。C11引入了unique_ptr和shared_ptr和weak_ptr。 需要注意的是unique_ptr对应boost的scoped_ptr。并且这些智能指针的实现原理是参考boost中的实现的。8.内存泄漏8.1什么是内存泄漏及其危害什么是内存泄漏内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存一般是忘记释放或者发生异常释放程序未能执行导致的。内存泄漏并不是指内存在物理上的消失而是应用程序分配某段内存后因为设计错误失去了对该段内存的控制因而造成了内存的浪费。内存泄漏的危害普通程序运行一会就结束了出现内存泄漏问题也不大进程正常结束页表的映射关系解除物理内存也可以释放。长期运行的程序出现内存泄漏影响很大如操作系统、后台服务、长时间运行的客户端等等不断出现内存泄漏会导致可用内存不断变少各种功能响应越来越慢最终卡死。int main() { // 申请⼀个1G未释放这个程序多次运⾏也没啥危害 // 因为程序⻢上就结束进程结束各种资源也就回收了 char* ptr new char[1024 * 1024 * 1024]; cout (void*)ptr endl; return 0; }8.2检测内存泄漏linux下内存泄漏检测https://blog.csdn.net/gatieme/article/details/51959654windows下第三方软件https://blog.csdn.net/lonely1047/article/details/1200389298.3如何避免内存泄漏工程前期良好的设计规范养成良好的编码规范申请的内存空间记着匹配的去释放。ps这个理想状态。但是如果碰上异常时就算注意释放了还是可能会出问题。需要下一条智能指针来管理才有保证。尽量使用智能指针来管理资源如果自己场景比较特殊采用RAII思想自己造个轮子管理。定期使用内存泄漏工具检测尤其是每次项目快上线前不过有些工具不够靠谱或者是收费。总结一下内存泄漏非常常见解决方案分为两种1、事前预防型。如智能指针等。2、事后查错型。如泄漏检测工具。

相关文章:

C++智能指针:高效管理内存的利器

上篇文章:C算法:一维/二维前缀和算法模板题 目录 1.智能指针的使用场景 2.RAII和智能指针的设计思路 3.C标准库智能指针的使用 4.智能指针的原理 5.shared_ptr和weak_ptr 5.1shared_ptr循环引用问题 5.2weak_ptr 6.shared_ptr的线程安全问题 7…...

Linux系列四:SSH工具安装

由于我们企业开发时,Linux服务器一般都是在远程的机房部署的,我们要操作服务 器,不会每次都跑到远程的机房里面操作,而是会直接通过SSH连接工具进行连接操作。 进入正文: https://dl.hostbuf.com/finalshell3/finalsh…...

【跟韩工学Hadoop系列第3篇】Hadoop 单节点集群搭建(优化版)-003篇

文章目录Hadoop 单节点集群搭建(优化版)一、环境前置优化(基础层)1. 系统级优化(Linux)2. JDK 环境优化二、核心配置优化(功能层)1. 通用配置(core-site.xml)…...

Excel高级查询工具Pro版v3.0|跨平台浏览器端轻量级数据引擎

温馨提示:文末有联系方式为什么推荐升级至Excel高级查询工具Pro版v3.0? 突破传统Excel筛选局限,这不仅是一款增强型查询插件,更是一个嵌入浏览器的轻量级本地数据库系统——无需联网、不依赖Office,开箱即用。跨平台无…...

PTA 习题9-3 平面向量加法

本题要求编写程序,计算两个二维平面向量的和向量。输入格式:输入在一行中按照“x1​ y1​ x2​ y2​”的格式给出两个二维平面向量v1​(x1​,y1​)和v2​(x2​,y2​)的分量。输出格式:在一行中按照(x, y)的格式输出和向量,坐标输出小数点后一位&#xff…...

生产环境部署Spring boot时优化启动速度和内存的技巧

核心思路是减少类加载、精简依赖、优化JVM。一、 提升启动速度惰性初始化在 application.yml或启动命令中开启,让 Bean 只在首次使用时创建,大幅缩短启动时间:spring:main:lazy-initialization: true注意:可能隐藏依赖问题&#x…...

ADRC自抗扰控制器的Simulink仿真之旅:S函数构建ESO与TD模块

ADRC自抗扰控制器的simulink仿真,ESO和TD等模块使用S函数开发最近在研究控制算法这块,ADRC(自抗扰控制器)着实引起了我的兴趣。今天就来跟大家分享一下ADRC在Simulink中的仿真实现,特别是ESO(扩张状态观测器&#xff0…...

ServerBox安卓版SSH客户端推荐|高效远程服务器管理工具

温馨提示:文末有联系方式ServerBox安卓版:专为移动场景优化的SSH管理利器 ServerBox安卓版是一款面向Android平台深度优化的SSH终端工具,集安全性、稳定性与易用性于一体,让服务器管理不再局限于电脑端。核心功能全覆盖&#xff0…...

探索滑模控制的多元世界

滑模控制,SMC,滑模抖振削弱,模糊滑模,动态滑模,高阶滑模,反馈线性化滑模,滑模控制器参数寻优。在控制领域,滑模控制(SMC)就像一颗璀璨的明珠,以其…...

关于网络故障排查

ping和traceroute命令都是常用的工具ping命令主要用于检测网络联通性。通过发送ICMP(Internet Control Message Protocol)回显请求(echo request)报文并等待目标主机回应,可以确定目标主机是否在线以及网络传输的延迟和…...

2026 全球 AI 大模型实力榜:国产崛起与国际争锋的全新格局

2026 年,全球 AI 大模型行业迈入技术深水区与产业落地期双轨并行的新阶段,国际头部模型在超大参数、全模态能力上持续突破,国产模型则凭借本土化适配、垂直领域深耕、算力自主化实现弯道超车,不仅在核心性能上与国际顶尖模型差距持…...

WrenAI 深度解析:算法视角:wren-ai-service 如何利用 RAG 与 Metadata 提升 SQL 准确率?

在 Text-to-SQL 领域,尽管 GPT-4 等大语言模型(LLM)已经展现了惊人的代码生成能力,但在面对复杂的企业级数据库时,直接生成 SQL 的准确率往往不尽如人意。核心难点不在于 SQL 语法,而在于**“语义鸿沟”**&…...

2.斐波那契数列的题解

##题目的分析首先他是关于一个数列,第一项为1,第二项为1,从第三项起为前两项的和1.输入#测试数据的组数n#输入n组测试数据(每组数据代表数列的第几项)2.输出#对应第n项的数字(这个数字是前两项数字的和&…...

基于NI Linux RT和Veristand的实时仿真探索

基于Ni linux RT和veristand的实时仿真在工程与科研领域,实时仿真扮演着至关重要的角色,它能帮助我们在实际系统搭建之前,对各种设计进行验证和优化。今天咱们就来聊聊基于NI Linux RT和Veristand的实时仿真。 NI Linux RT简介 NI Linux RT是…...

字节面试官怒怼:RAG只会检索?大模型意图识别实战(非常详细),从入门到精通,收藏这一篇就够了!

直到上周一个学员面试的经历让我觉得必须补上这篇了:Query 理解与路由。 面试官问他:“用户在你们系统里输入’帮我算一下 A 款保险的理赔金额’,你的系统怎么处理的?” 他说:“先做 Embedding,然后去向量…...

【基于GasTurb的不同构型发动机性能对比】 GasTurb软件 1、涡桨、涡扇发动机等构型

【基于GasTurb的不同构型发动机性能对比】 GasTurb软件 1、涡桨、涡扇发动机等构型 2、在一样的推力需求下对比NOx排放差异 3、在不同的delta_T和高度下对比性能差异在航空发动机设计领域,GasTurb软件就像机械工程师的瑞士军刀。今天咱们拿它来折腾点有意思的——把…...

玩转T-Mats库:航空发动机气路故障仿真那些事儿

T-Mats库 涡扇发动机气路故障 数据 仿真模型】 1、包含部件流量、效率及压比故障在内的13类故障植入,故障程序和组合可自定义;航空发动机,典型气路故障仿真; 2、基于软阈值去噪处理后的信号序列提取了真实的运行扰动; 3、输出数据符合CMAPASS的排列要求。…...

电导增量法INC仿真模型,作为目前实际光伏发电系统中最常用的mppt算法,可以用于学习研究

电导增量法INC仿真模型,作为目前实际光伏发电系统中最常用的mppt算法,可以用于学习研究,才用了输出参考电压的方式来进行pwm调制。电导增量法的仿真模型在光伏圈子里都快被玩出花了,但说实话这算法确实稳得一批。今天咱们直接撸代…...

基于企鹅优化算法的机器人轨迹规划(Matlab代码实现)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...

手把手玩转变压器辐射传热仿真】今天带大家用COMSOL做个会“发热发光“的变压器模型,全程保姆级操作演示,连参数设置时容易手滑的地方都标出来了

有限元仿真模型二:视频教程仿真模型-基于comsol有限元的变压器辐射传热数值分析 1、分析变压器内外辐射传热机理,利用传热方程建立辐射传热数值计算模型 2、提供详细的讲解指导,带你实现从零到有,超级适合新手小白学习&#xff01…...

2026年春日活动海报复盘:如何快速敲定桃树主题视觉方案

作为一名社群运营,我经常需要为各种小型活动快速制作宣传物料。上周三,我接到了一个紧急任务:为周末的“社区踏春赏桃”活动设计一张宣传海报,要求周四上午就要发到业主群和朋友圈里预热。时间紧,任务急,而…...

Java 编程中的运算符探寻

Java 编程中的运算符探寻 🚀 Java 运算符是表达式计算的核心,掌握它们能让你写出高效、清晰的代码,避免优先级陷阱。本文从基础到进阶,带你全面探寻 Java 运算符(基于 Java 21/23 最新特性,2026 年仍适用&…...

带你了解Java中的Mono接口

带你了解 Java 中的 Mono 接口 🚀 在 Java 响应式编程(Reactive Programming)中,Mono 是 Project Reactor 库的核心类型之一。它不是 Java 标准库(JDK)中的接口,而是 Reactor Core 提供的响应式…...

西门子s71200 总线控制V90伺服程序模板 ,包含plc程序,西门子触摸屏程序

西门子s71200 总线控制V90伺服程序模板 ,包含plc程序,西门子触摸屏程序,eplan电气图纸一份 有两种控制模式 1.基于111报文自己编写的PN通讯控制V90伺服程序。 2.基于工艺对象轴程序PROFIdrive配置的PN通讯控制V90伺服程序 程序可以直接复制使…...

单北斗GNSS形变监测是什么?主要有如何应用于大坝监测?

单北斗GNSS形变监测是一种利用卫星技术进行位移监测的高精度系统,广泛应用于大坝、桥梁等基础设施的安全监测。该系统通过接收GPS信号,能够实时获取目标点的三维位置变化,提供可靠的数据支持。在应用过程中,用户可以根据具体监测需…...

东华Oj101-103

101. 找出质数 作者: SunCiHai 时间限制: 10s 章节: 字符串 问题描述 明明学习数学已经有一段时间了。一次老师在课上讲了什么叫质数。质数就是大于等于2且只能被1和其本身整除的整数。明明觉得这很简单,以为这很容易掌握,于是就不多做练习。明的爸…...

2. LangGraph 安装指南

Langgraph 系统教程(基于 1.1.X 版本) LangGraph 综述 LangGraph 安装指南 LangGraph 快速入门 用 LangGraph 的思维构建智能体 LangGraph 工作流与智能体详解...

学Simulink——基于Simulink的通信延迟下多机轨迹一致性分析

目录 手把手教你学Simulink ——基于Simulink的通信延迟下多机轨迹一致性分析 一、问题背景 二、多智能体系统建模 1. 单智能体动力学(简化为一阶积分器) 2. 分布式一致性协议(无延迟) 3. 引入通信延迟 三、稳定性分析理论基础 四、Simulink 建模仿真步骤 第一步:…...

Qwen3-ASR-1.7B语音转写教程:音频切片策略+长语音分段识别最佳实践

Qwen3-ASR-1.7B语音转写教程:音频切片策略长语音分段识别最佳实践 你是不是也遇到过这种情况:一段长达一小时的会议录音,上传到语音识别工具后,要么识别速度慢得让人抓狂,要么中间卡住直接报错,最后只能得…...

3D Face HRN保姆级教程:如何用Pillow预处理图像提升人脸检测成功率

3D Face HRN保姆级教程:如何用Pillow预处理图像提升人脸检测成功率 1. 为什么图像预处理如此重要 当你使用3D Face HRN人脸重建模型时,可能会遇到这样的问题:上传了一张看起来不错的人脸照片,系统却提示"未检测到人脸"…...