list 容器常见用法及实现
文章目录
- 1. list 的介绍与使用
- 1.1 list 的介绍
- 1.2 list 的使用
- 1.2.1 list 的构造
- 1.2.2 list iterator 的使用
- 1.2.3 list capacity
- 1.2.4 list element access
- 1.2.5 list modifiers
- 1.2.6 迭代器失效问题
- 2. list 的模拟实现
- 2.1 值得注意的点:
- 2.2 std::initializer_list
- 2.3 list.h
- 2.4 迭代器模板的优化
- 2.5 list 的反向迭代器
- 3. list 与 vector 的对比
1. list 的介绍与使用
1.1 list 的介绍
list 容器底层实现就是一个带虚拟头节点的双向循环链表
1.2 list 的使用
1.2.1 list 的构造
构造函数 | 接口说明 |
---|---|
list (size_type n, const value_type& val =value_type()) | 构造的 list 中包含 n 个值为 val 的元素 |
list () | 构造空的 list |
list (const list& x) | 拷贝构造函数 |
list (InputIterator first, InputIterator last) | 用[first, last)区间中的元素构造 list |
#include <iostream>
using namespace std;
#include <list>
#include <vector>int main()
{list<int> lt1(10, 1); // 构造 10 个 1list<int> lt2; // 构造空的 listlist<int> lt3(lt1); // 拷贝构造vector<int> v(11, 2); list<int> lt4(v.begin(), v.end()); // 迭代器构造return 0;
}
1.2.2 list iterator 的使用
list 的迭代器通常被设计成一个类,它包含着一个指向当前链表节点的指针
函数声明 | 接口说明 |
---|---|
begin + end | 返回第一个元素的迭代器 + 返回最后一个元素下一个位置的迭代器 |
rbegin + rend | 返回第一个元素的 reverse_iterator,即 end 位置,返回最后一个元素下一个位置的 reverse_iterator,即 begin位置 |
auto it = lt1.begin();while (it != lt1.end()){cout << *it << " ";++it;}cout << endl;auto it1 = lt4.rbegin();while (it1 != lt4.rend()){cout << *it << " ";++it;}cout << endl;
注意:
- 反向迭代器的操作也是 ++ 让迭代器向前移动。
1.2.3 list capacity
函数声明 | 接口说明 |
---|---|
empty() | 检测 list 是否为空,空返回 true,否则返回 false |
size() | 返回 list 中有效节点个数 |
C++ 里 布尔类型的底层常用 一字节 存储
- true 通常为 0x01
- false 通常为 0x00
- 但具体实现由编译器决定,标准仅要求 false 和 true 在转换为整数时分别为 0 和 1。
cout << lt1.empty() << endl;cout << lt2.empty() << endl;cout << lt1.size() << endl;
1.2.4 list element access
函数声明 | 函数接口 |
---|---|
front() | 返回 list 的第一个节点的引用 |
back() | 返回 list 的最后一个节点的引用 |
lt1.push_front(2);lt1.push_back(3);cout << lt1.front() << endl;cout << lt1.back() << endl;
1.2.5 list modifiers
函数声明 | 接口说明 |
---|---|
push_front | 在 list 首元素前插入值为 val 的元素 |
pop_front | 删除 list 中第一个元素 |
push_back | 在 list 尾部插入值为 val 的元素 |
pop_bakc | 删除 list 中最后一个元素 |
insert | 在 list 的 pos 位置插入值为 val 的元素 |
erase | 删除 pos 位置的元素 |
swap | 交换两个 list |
clear | 清空 list 中的有效元素 |
int main()
{list<int> lt;lt.push_back(1); // 尾插lt.push_back(2);lt.push_back(3);lt.push_front(4); // 头插lt.push_front(5);lt.push_front(6);for (auto& x : lt)cout << x << " ";cout << endl;lt.pop_back(); // 尾删lt.pop_front(); // 头删for (auto& x : lt)cout << x << " ";cout << endl;auto pos = find(lt.begin(), lt.end(), 4); // 查找lt.insert(pos, 7); // 在 pos 前插入pos = find(lt.begin(), lt.end(), 1); // 查找lt.erase(pos); // 删除 pos 位置for (auto& x : lt)cout << x << " ";cout << endl;list<int> lt1 = { 1, 2, 3 }; // C++11 特性,之后再讲lt.swap(lt1); // 交换两个 listfor (auto& x : lt)cout << x << " ";cout << endl;for (auto& x : lt1)cout << x << " ";cout << endl;lt1.clear(); // 清空有效元素cout << lt1.size() << endl;return 0;
}
1.2.6 迭代器失效问题
前面提过,list 的迭代器通常被设计成一个类,包含着一个指向当前节点的指针,迭代器失效即迭代器所指向的节点的无效,即该节点被删除了。因为 list 的底层结构是 带虚拟头节点的双向循环链表,因此只有在删除时,迭代器才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不受影响。
int main()
{list<int> lt = { 1,2,3,4,5,6,7 }; // C++11 特性/* 错误写法auto it = lt.begin();while (it != lt.end()){// erase() 函数执行之后,it所指向节点已被删除,it 失效,在下一次使用之前必须给it赋值lt.erase(it);++it;}*/auto it = lt.begin();while (it != lt.end()){lt.erase(it++); // it = lt.erase(it);}return 0;
}
2. list 的模拟实现
list 容器本质是对 带头节点的双向循环链表 的封装,再加上前面说的 list 迭代器单独成一个类,我们很容易就能理解它的大致架构。
本文主要讲解对 list 的封装,要是还不熟悉 双向循环带头链表 的可以去看前面的讲解: 双向循环链表。
- list_node 两个指针域与一个数据域
- list_iterator 用于封装迭代器的操作
- list 用于组织 list_node
2.1 值得注意的点:
-
迭代器进行拷贝的时候使用编译器默认生成的浅拷贝就行了,因为资源是被链表管理的,而非迭代器,迭代器只是用来模拟指针的行为
-
因为要适配迭代器的迭代,所以这里的 insert 和 erase 的返回值都是 迭代器类型
-
一个奇怪的实现 operator->(),迭代器模拟的是指针的行为,结构体的时候才需要使用 ->,我们知道结构体可以使用 (*it)._a ,也可以使用 it->_a,所以需要重载 ->,而 -> 是访问指针类型的,所以返回值为指针,但是它的使用就有点奇怪了
T* operator->() {return &_node->_data; }// 使用时 cout << it->_a << endl; // 实际上是两个 -> ,但是太难看了,所以省略了一个 cout << it.operator->()->_a << endl; // 第一个是运算符重载,第二个是原生指针的解引用
2.2 std::initializer_list
它是 C++11 引入的一个标准库特性,用于简化对象通过花括号初始化列表(如 {1, 2, 3})进行初始化或赋值的操作。它是 <initializer_list> 头文件中定义的模板类。
用途:
-
允许容器、自定义类或函数接受花括号列表作为参数,例如:
// 直接构造 list<int> lt0({ 1,2,3,4,5,6 }); // 隐式类型转换 list<int> lt1 = { 1,2,3,4,5,6,7,8 };
特性:
- 元素是只读的(通过 const T* 访问)。
- 底层数据是浅拷贝(仅复制指针,不复制元素本身)。
- 通常用于临时初始化场景,需注意底层数据的生命周期。
其他地方和之前讲解的 vector 的迭代器类似,这里便不做过多赘述。
2.3 list.h
#pragma once#include <assert.h>
#include <iostream>
using namespace std;namespace zkp
{template<class T>struct list_node{T _data;list_node<T>* _next;list_node<T>* _prev;list_node(const T& data = T()):_data(data), _next(nullptr), _prev(nullptr){}};template<class T>struct list_iterator{typedef list_node<T> Node;typedef list_iterator<T> Self;Node* _node;list_iterator(Node* node):_node(node){}T& operator*(){return _node->_data;}T* operator->(){return &_node->_data;}Self& operator++(){_node = _node->_next; // 将迭代器指向的 node 指向下一个 nodereturn *this; // 返回这个迭代器}Self& operator--(){_node = _node->_prev; // 将迭代器指向的 node 指向上一个 nodereturn *this;}bool operator!=(const Self& s) const{return _node != s._node;}bool operator==(const Self& s) const{return _node == s._node;}};template<class T>struct list_const_iterator{typedef list_node<T> Node;typedef list_const_iterator<T> Self;Node* _node;list_const_iterator(Node* node):_node(node){}const T& operator*(){return _node->_data;}Self& operator++(){_node = _node->_next; // 将迭代器指向的 node 指向下一个 nodereturn *this; // 返回这个迭代器}Self& operator--(){_node = _node->_prev; // 将迭代器指向的 node 指向上一个 nodereturn *this;}bool operator!=(const Self& s) const{return _node != s._node;}bool operator==(const Self& s) const{return _node == s._node;}};template<class T>class list{typedef list_node<T> Node;public:typedef list_iterator<T> iterator;typedef list_const_iterator<T> const_iterator;iterator begin(){return _head->_next;}iterator end(){return _head;}const_iterator begin() const{return _head->_next;}const_iterator end() const{return _head;}void empty_init(){_head = new Node;_head->_next = _head;_head->_prev = _head;_size = 0;}list(){empty_init();}list(initializer_list<T> il){empty_init();for (auto& e : il){push_back(e);}}// lt2(lt1)list(const list<T>& lt){empty_init();for (auto& e : lt){push_back(e);}}void swap(list<T>& lt){std::swap(_head, lt._head);std::swap(_size, lt._size);}// lt1 = lt3list<T>& operator=(list<T> lt){swap(lt);return *this;}void clear(){auto it = begin();while (it != end()){it = erase(it);}}~list(){clear();delete _head;_head = nullptr;}void push_back(const T& x){/*Node* newnode = new Node(x);newnode->_next = _head;newnode->_prev = _head->_prev;_head->_prev->_next = newnode;_head->_prev = newnode;++_size;*/insert(end(), x);}void push_front(const T& x){insert(begin(), x);}iterator insert(iterator pos, const T& x){Node* cur = pos._node;Node* prev = cur->_prev;Node* newnode = new Node(x);// prev newnode curnewnode->_next = cur;cur->_prev = newnode;newnode->_prev = prev;prev->_next = newnode;++_size;return newnode;}void pop_back(){erase(end());}void pop_front(){erase(begin());}iterator erase(iterator pos){assert(pos != end());Node* prev = pos._node->_prev;Node* next = pos._node->_next;prev->_next = next;next->_prev = prev;delete pos._node;--_size;return next;}size_t size() const{return _size;}bool empty() const{return _size == 0;}private:Node* _head;size_t _size;};struct AA{int _a1 = 1;int _a2 = 1;};// 按需实例化// T* const ptr1// const T* ptr2template<class Container>void print_container(const Container& con){// const iterator -> 迭代器本身不能修改// const_iterator -> 指向内容不能修改typename Container::const_iterator it = con.begin();//auto it = con.begin();while (it != con.end()){//*it += 10;cout << *it << " ";++it;}cout << endl;for (auto e : con){cout << e << " ";}cout << endl;}void test_list1(){list<int> lt;lt.push_back(1);lt.push_back(2);lt.push_back(3);lt.push_back(4);list<int>::iterator it = lt.begin();while (it != lt.end()){*it += 10;cout << *it << " ";++it;}cout << endl;for (auto e : lt){cout << e << " ";}cout << endl;print_container(lt);list<AA> lta;lta.push_back(AA());lta.push_back(AA());lta.push_back(AA());lta.push_back(AA());list<AA>::iterator ita = lta.begin();while (ita != lta.end()){//cout << (*ita)._a1 << ":" << (*ita)._a2 << endl;// 特殊处理,本来应该是两个->才合理,为了可读性,省略了一个->cout << ita->_a1 << ":" << ita->_a2 << endl;cout << ita.operator->()->_a1 << ":" << ita.operator->()->_a2 << endl;++ita;}cout << endl;}void test_list2(){list<int> lt;lt.push_back(1);lt.push_back(2);lt.push_back(3);lt.push_back(4);// insert以后迭代器不失效list<int>::iterator it = lt.begin();lt.insert(it, 10);*it += 100;print_container(lt);// erase以后迭代器失效// 删除所有的偶数it = lt.begin();while (it != lt.end()){if (*it % 2 == 0){it = lt.erase(it);}else{++it;}}print_container(lt);}void test_list3(){list<int> lt1;lt1.push_back(1);lt1.push_back(2);lt1.push_back(3);lt1.push_back(4);list<int> lt2(lt1);print_container(lt1);print_container(lt2);list<int> lt3;lt3.push_back(10);lt3.push_back(20);lt3.push_back(30);lt3.push_back(40);lt1 = lt3;print_container(lt1);print_container(lt3);}void func(const list<int>& lt){print_container(lt);}
}
2.4 迭代器模板的优化
我们不难发现,普通迭代器和 const迭代器的代码只有一点点区别,重合度太高了,有没有什么能解决这种情况呢? 有的,兄弟有的,让我们看一下 库中是怎么实现的。
这里把原来一个模板参数 T 增加到了三个 T, T&, T*
- 普通迭代器:传入<T, T&, T*>
list_iterator<T, T&, T*> it; // operator* 返回 T&,operator-> 返回 T*
- const迭代器:传入 <T, const T&, const T*>
list_iterator<T, const T&, const T*> const_it; // operator* 返回 const T&
这样,同一个迭代器模板就可以通过传入不同的模板参数生成普通迭代器和 const 迭代器,大大减少了重复代码
要修改的代码其实也不多,只不过是把原来迭代器参数模板增加到三个,再在 list 类模板中传入不同的模板参数生成两种迭代器。
直接来看修改后的迭代器模板代码吧
template<class T, class Ref, class Ptr>
struct list_iterator
{typedef list_node<T> Node;typedef list_iterator<T, Ref, Ptr> Self;Node* _node;list_iterator(Node* node):_node(node){}Ref operator*(){return _node->_data;}Ptr operator->(){return &_node->_data;}Self& operator++(){_node = _node->_next;return *this;}Self& operator--(){_node = _node->_prev;return *this;}Self operator++(int){Self tmp(*this);_node = _node->_next;return tmp;}Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}bool operator==(const Self& s) const{return _node == s._node;}bool operator!=(const Self& s) const{return _node != s._node;}
};
2.5 list 的反向迭代器
前面我们说过不管是正向迭代器还是反向迭代器,它向后移动的操作都是 ++,反向迭代器的++就是正向迭代器的–,反向迭代器的–就是正向迭代器的++,因此反向迭代器的实现可以借助正向迭代器,即:反向迭代器内部可以包含一个正向迭代器,对正向迭代器的接口进行包装即可。
template<class Iterator>class ReverseListIterator{// 注意:此处typename的作用是明确告诉编译器,Ref是Iterator类中的类型,而不是静态成员变量// 否则编译器编译时就不知道Ref是Iterator中的类型还是静态成员变量// 因为静态成员变量也是按照 类名::静态成员变量名 的方式访问的public:typedef typename Iterator::Ref Ref;typedef typename Iterator::Ptr Ptr;typedef ReverseListIterator<Iterator> Self;public:// 构造ReverseListIterator(Iterator it) : _it(it) {}// 具有指针类似行为Ref operator*() {Iterator temp(_it);--temp;return *temp;} Ptr operator->() { return &(operator*()); }// 迭代器支持移动Self& operator++() {--_it;return *this;} Self operator++(int) {Self temp(*this);--_it;return temp;} Self& operator--() {++_it;return *this;} Self operator--(int){Self temp(*this);++_it;return temp;}// 迭代器支持比较bool operator!=(const Self& l)const { return _it != l._it; }bool operator==(const Self& l)const { return _it != l._it; }Iterator _it;};
3. list 与 vector 的对比
其实就是双向循环链表和动态数组的对比
- | vector | list |
---|---|---|
底层结构 | 动态数组 | 带头双向循环链表 |
随机访问 | 支持随机访问,访问某个元素效率O(1) | 不支持随机访问,访问某个元素效率O(N) |
插入和删除 | 任意位置插入和删除效率低,需要搬移元素,时间复杂度为O(N),插入时有可能需要增容,增容:开辟新空间,拷贝元素,释放旧空间,导致效率更低 | 任意位置插入和删除效率高,不需要搬移元素,时间复杂度为O(1) |
空间利用率 | 底层为连续空间,不容易造成内存碎片,空间利用率高,缓存利用率高 | 底层节点动态开辟,小节点容易造成内存碎片,空间利用率低,缓存利用率低 |
迭代器 | 原生态指针 | 对原生态指针(节点指针)进行封装 |
迭代器失效 | 在插入元素时,要给所有的迭代器重新赋值,因为插入元素有可能会导致重新扩容,致使原来迭代器失效,删除时,当前迭代器需要重新赋值否则会失效 | 插入元素不会导致迭代器失效,删除元素时,只会导致当前迭代器失效,其他迭代器不受影响 |
使用场景 | 需要高效存储,支持随机访问,不关心插入删除效率 | 大量插入和删除操作,不关心随机访问 |
相关文章:

list 容器常见用法及实现
文章目录 1. list 的介绍与使用1.1 list 的介绍1.2 list 的使用1.2.1 list 的构造1.2.2 list iterator 的使用1.2.3 list capacity1.2.4 list element access1.2.5 list modifiers1.2.6 迭代器失效问题 2. list 的模拟实现2.1 值得注意的点:2.2 std::initializer_li…...

iOS视频编码详细步骤(视频编码器,基于 VideoToolbox,支持硬件编码 H264/H265)
iOS视频编码详细步骤流程 1. 视频采集阶段 视频采集所使用的代码和之前的相同,所以不再过多进行赘述 初始化配置: 通过VideoCaptureConfig设置分辨率1920x1080、帧率30fps、像素格式kCVPixelFormatType_420YpCbCr8BiPlanarFullRange设置摄像头位置&am…...

浅析 Golang 内存管理
文章目录 浅析 Golang 内存管理栈(Stack)堆(Heap)堆 vs. 栈内存逃逸分析内存逃逸产生的原因避免内存逃逸的手段 内存泄露常见的内存泄露场景如何避免内存泄露?总结 浅析 Golang 内存管理 在 Golang 当中,堆…...
记录: Windows下远程Liunx 系统xrdp 用到的一些小问题(免费踩坑 记录)
采用liunx Ubuntu22.04版本以下,需要安装 xrdp 或者VNC 具体过程就是下载 在linux命令行里 首先更新软件包:sudo apt update 安装xrdp服务:sudo apt install xrdp 启动XRDP:sudo systemctl start xrdp(如果在启动的…...

C++ 并发编程(1)再学习,为什么子线程不调用join方法或者detach方法,程序会崩溃? 仿函数的线程启动问题?为什么线程参数默认传参方式是值拷贝?
本文的主要学习点,来自 这哥们的视频内容,感谢大神的无私奉献。你可以根据这哥们的视频内容学习,我这里只是将自己不明白的点,整理记录。 C 并发编程(1) 线程基础,为什么线程参数默认传参方式是值拷贝?_哔…...

【Python 算法零基础 2.模拟 ④ 基于矩阵】
目录 基于矩阵 Ⅰ、 2120. 执行所有后缀指令 思路与算法 ① 初始化结果列表 ② 方向映射 ③ 遍历每个起始位置 ④ 记录结果 Ⅱ、1252. 奇数值单元格的数目 思路与算法 ① 初始化矩阵 ② 处理每个操作 ③ 统计奇数元素 Ⅲ、 832. 翻转图像 思路与算法 ① 水平翻转图像 ② 像素值…...

【教程】Docker方式本地部署Overleaf
转载请注明出处:小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你,欢迎[点赞、收藏、关注]哦~ 目录 背景说明 下载仓库 初始化配置 修改监听IP和端口 自定义网站名称 修改数据存放位置 更换Docker源 更换Docker存储位置 启动Overleaf 创…...

3337|3335. 字符串转换后的长度 I(||)
1.字符串转换后的长度 I 1.1题目 3335. 字符串转换后的长度 I - 力扣(LeetCode) 1.2解析 递推法解析 思路框架 我们可以通过定义状态变量来追踪每次转换后各字符的数量变化。具体地,定义状态函数 f(i,c) 表示经过 i 次转换后࿰…...

PHP黑白胶卷底片图转彩图功能 V2025.05.15
关于底片转彩图 传统照片底片是摄影过程中生成的反色图像,为了欣赏照片,需要通过冲印过程将底片转化为正像。而随着数字技术的发展,我们现在可以使用数字工具不仅将底片转为正像,还可以添加色彩,重现照片原本的色彩效…...

字符串检索算法:KMP和Trie树
目录 1.引言 2.KMP算法 3.Trie树 3.1.简介 3.2.Trie树的应用场景 3.3.复杂度分析 3.4.Trie 树的优缺点 3.5.示例 1.引言 字符串匹配,给定一个主串 S 和一个模式串 P,判断 P 是否是 S 的子串,即找到 P 在 S 中第一次出现的位置。暴力匹…...
Java大师成长计划之第22天:Spring Cloud微服务架构
📢 友情提示: 本文由银河易创AI(https://ai.eaigx.com)平台gpt-4o-mini模型辅助创作完成,旨在提供灵感参考与技术分享,文中关键数据、代码与结论建议通过官方渠道验证。 随着企业应用的不断扩展,…...
瀑布模型VS敏捷模型VS喷泉模型
目录 1. 瀑布模型(Waterfall Model) 2. 敏捷模型(Agile Model) 3. 喷泉模型(Fountain Model)...

基于.Net开发的网络管理与监控工具
从零学习构建一个完整的系统 平常项目上线后,不仅意味着开发的完成,更意味着项目正式进入日常运维阶段。在这个阶段,网络的监控与管理也是至关重要的,这时候就需要一款网络管理工具,可以协助运维人员用于日常管理&…...

Python并发编程:开启性能优化的大门(7/10)
1.引言 在当今数字化时代,Python 已成为编程领域中一颗璀璨的明星,占据着编程语言排行榜的榜首。无论是数据科学、人工智能,还是 Web 开发、自动化脚本编写,Python 都以其简洁的语法、丰富的库和强大的功能,赢得了广大…...
Linux 中 open 函数的本质与细节全解析
一、open简介 在 Linux 下,一切皆文件。而对文件的读写,离不开文件的“打开”操作。虽然 C 语言标准库提供了方便的 fopen,但更底层、更强大的是系统调用 open,掌握它能让你对文件系统控制更细致,在系统编程、驱动开发…...
llama.cpp无法使用gpu的问题
使用cuda编译llama.cpp后,仍然无法使用gpu。 ./llama-server -m ../../../../../model/hf_models/qwen/qwen3-4b-q8_0.gguf -ngl 40 报错如下 ggml_cuda_init: failed to initialize CUDA: forward compatibility was attempted on non supported HW warning: n…...
Python Unicode字符串和普通字符串转换
Unicode 是一种字符编码标准,旨在为世界上所有书写系统的每个字符提供一个唯一的数字标识(称为码点)。 码点: 每个 Unicode 字符被分配一个唯一的数字,称为码点表示形式:u 后跟 4-6 位十六进制数…...
Ansible Roles 是一种用于层次化和结构化组织 Ansible Playbook 的机制。
Ansible Roles 是一种用于层次化和结构化组织 Ansible Playbook 的机制。它通过将变量、文件、任务、模板和处理器等放置在单独的目录中,简化了 Playbook 的管理和复用。Roles 自 Ansible 1.2 版本引入,极大地提高了代码的可维护性和可重用性。 目录结构 一个标准的 Ansibl…...

易学探索助手-个人记录(十)
在现代 Web 应用中,用户体验的重要性不断上升。近期我完成了两个功能模块 —— 语音播报功能 与 用户信息修改表单,分别增强了界面交互与用户自管理能力。 一、语音播报功能(SpeechSynthesis) 功能特点 支持播放、暂停、继续、停…...
Linux基础 -- SSH 流式烧录与压缩传输笔记
Linux SSH 流式烧录与压缩传输指南 一、背景介绍 在嵌入式开发和维护中,常常需要通过 SSH 从 PC 向设备端传输大文件(如系统镜像、固件)并将其直接烧录到指定磁盘(如 /dev/mmcblk2)。然而,设备端存储空间…...

学习51单片机01(安装开发环境)
新学期新相貌.......哈哈哈,我终于把贪吃蛇结束了,现在我们来学stc51单片机! 要求:c语言的程度至少要到函数,指针尽量!如果c语言不好的,可以回去看看我的c语言笔记。 1.开发环境的安装&#x…...
事件驱动reactor的原理与实现
fdset 集合:(就是说) fd_set是一个位图(bitmap)结构 每个位代表一个文件描述符 0表示不在集合中,1表示在集合中 fd_set结构(简化): [0][1][2][3][4][5]...[1023] …...
大模型训练简介
在人工智能蓬勃发展的当下,大语言模型(LLM)成为了众多应用的核心驱动力。从智能聊天机器人到复杂的内容生成系统,LLM 的卓越表现令人瞩目。而这背后,大模型的训练过程充满了奥秘。本文将深入探讨 LLM 训练的各个方面&a…...
深度解析 MySQL 与 Spring Boot 长耗时进程:从故障现象到根治方案(含 Tomcat 重启必要性分析)
一、典型故障现象与用户痛点 在高并发业务场景中,企业级 Spring Boot 应用常遇到以下连锁故障: 用户侧:网页访问超时、提交表单无响应,报错 “服务不可用”。运维侧:监控平台报警 “数据库连接池耗尽”,To…...
More Effective C++:改善编程与设计(上)
More Effective C: 目录 More Effective C: 条款1:仔细区别pointers和 references 条款2:最好使用C转型操作符 条款3:绝对不要以多态方式处理数组 条款4:非必要不要提供default constructor 条款5:对定制的“类型转换函数”保持警觉 …...
TNNLS-2020《Autoencoder Constrained Clustering With Adaptive Neighbors》
核心思想分析 该论文提出了一种名为ACC_AN(Autoencoder Constrained Clustering with Adaptive Neighbors)的深度聚类方法,旨在解决传统子空间聚类方法在处理非线性数据分布和高维数据时的局限性。核心思想是将深度自编码器(Auto…...

SpringAI
机器学习: 定义:人工智能的子领域,通过数据驱动的方法让计算机学习规律,进行预测或决策。 核心方法: 监督学习(如线性回归、SVM)。 无监督学习(如聚类、降维)。 强化学…...

lua 作为嵌入式设备的配置语言
从lua的脚本中获取数据 lua中栈的索引 3 | -1 2 | -2 1 | -3 可以在lua的解释器中加入自己自定的一些功能,其实没啥必要,就是为了可以练习下lua...

ERP系统源码,小型工厂ERP系统源码,CRM+OA+进销存+财务
ERP系统源码,小型工厂ERP系统源码,ERP计划管理系统源码,CRMOA进销存财务 对于ERP来说,最为主要的作用就是能够强调企业的计划性,通过以业务订单以及客户的相关需求来作为企业计划的基础,并且还能够对企业现…...

基于EFISH-SCB-RK3576/SAIL-RK3576的矿用本安型手持终端技术方案
(国产化替代J1900的矿山智能化解决方案) 一、硬件架构设计 本安型结构设计 防爆防护体系: 采用铸镁合金外壳复合防爆玻璃(抗冲击能量>20J),通过GB 3836.1-2021 Ex ib I Mb认证 全密闭IP68接口…...