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

【简单模拟实现list】

在C++标准模板库(STL)中,list是一个非常强大的容器,它基于双向链表实现,支持高效的插入和删除操作。虽然我们可以直接使用STL中的list,但通过自己模拟实现一个list,可以更好地理解其背后的原理和数据结构的精髓。
在这里插入图片描述
今天我们要从三个方面来实现list:list的结构,list迭代器和list的常用接口。闲话少说,直接进入主题!

模拟实现list结点类

从上面的图片我们可以看到list中有三个成员变量,分别是prev前驱指针、next后驱指针和data数据域。

	template<class T>//建立节点结构struct Mylist_node{Mylist_node<T>* prev;Mylist_node<T>* next;T data;//构造函数Mylist_node(const T& val = T()):prev(nullptr), next(nullptr), data(val){}};

如上代码,我们分别写出了前驱指针,后驱指针和数据域。但是我们很奇怪的是,为什么要写一个构造函数呢?

初始化成员变量:构造函数可以为结构体的成员变量提供初始值。在上述的代码中,prev和next被初始化为nullptr,表示新创建的节点最初没有前驱和后继节点。
灵活性:通过构造函数,你可以灵活地为不同类型的数据成员提供不同的初始值。

构造函数中的T()是什么意思?

在构造函数中,T()是一个默认构造函数调用。这里的T是一个模板参数,表示Mylist_node可以存储任意类型的数据。T()表示调用类型T的默认构造函数来初始化data成员。

什么时候需要加上T()?

当你需要确保data成员被正确初始化,并且类型T有一个可用的默认构造函数时,就需要加上T()。
例如:
如果T是一个类类型,并且你希望data成员被初始化为该类的默认状态。
如果T是一个基本数据类型(如int、double等),T()将调用该类型的默认构造函数,通常将其初始化为0或0.0等。

模拟实现list迭代器

list的迭代器跟我们前面说的vector的迭代器大有不同,比vector的迭代器更加难实现,也就是说更加的复杂!话不多说我们来看看!
我们现在来分别实现两种迭代器普通迭代器和const迭代器。
普通迭代器:

	template <class T>struct list_iterator{typedef Mylist_node Node;Node* _node;list_iterator(Node* node):_node(node){}T& operator*(){return _node->data;}T operator->(){return &_node->data;}list_iterator<T>& operator++(){_node = _node->next;return (*this);}bool operator!=(const list_iterator<T>& it){return _node != it._node;}bool operator==(const list_iterator<T>& it){return _node == it._node;}};

const迭代器:

	template <class T>struct list_const_iterator{typedef Mylist_node Node;Node* _node;list_const_iterator(Node* node):_node(node){}const T& operator*(){return _node->data;}const T operator->(){return &_node->data;}list_const_iterator<T>& operator++(){_node = _node->next;return _node;}bool operator!=(const list_const_iterator<T>& it){return _node != it._node;}bool operator==(const list_const_iterator<T>& it){return _node == it._node;}};

我们观察上面我们实现的两个迭代器,是不是发现一个问题?就是很多代码其实重复了,那我们又该怎么办呢?当然是复用呀!我们该怎么复用上述代码呢?
如下所示:

//构造list迭代器结构
template<class T,class Ref,class Ptr>
struct list_iterator
{typedef Mylist_node<T> Node;typedef list_iterator 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++(int){Self tmp(*this);_node = _node->next;return tmp;}Self& operator--(){_node = _node->prev;return (*this);}Self operator--(int){Self tmp(*this);_node = _node->prev;return tmp;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}
};

如上代码,我们将两个迭代器的代码合二为一。那这样就会冒出很多问题了,为什么我们这样可以实现两个迭代器呢?一系列的问题。我们逐一解决!
首先我们看第一个问题:

template<class T,class Ref,class Ptr>

对于我们这里实现的模板,其中Ref和Ptr,这两个是啥玩意儿呀?
在这里插入图片描述
我们看上述红色框框里的内容,我们再往上翻,我们实现普通迭代器和const迭代器的时候,最突出的区别是不是在重载*和->的时候一个需要有const一个没有呢?
在这里插入图片描述
在这里插入图片描述
第二个问题在于前置++和后置++(前置–/后置–)
在这里插入图片描述

为什么前置++实现的时候要引用返回而后置++不用呢?

前置迭代器(++it)
前置迭代器在递增操作后返回一个新的迭代器,该迭代器指向下一个元素。由于这个操作改变了迭代器的状态(即它现在指向下一个元素),因此返回类型是引用(Self&),这意味着返回的是当前对象的引用。这样做的好处是可以链式调用,例如 ++(++it),而不会创建临时对象,从而提高效率。

后置迭代器(it++)
后置迭代器在递增操作后返回一个旧的迭代器,该迭代器仍然指向原始位置。由于这个操作不改变当前对象的状态,而是创建一个新的对象来表示递增后的状态,因此返回类型是值(Self),这意味着返回的是当前对象的拷贝。这样做的好处是可以在递增操作后仍然访问原始位置的元素。

为什么前置++()内不用加int而后置++却需要加上int呢?

在C++中,后置递增操作符需要一个参数(通常是int),以区分前置递增和后置递增。这是因为C++需要一种方式来区分这两种操作,而参数类型是区分它们的一种方式。

模拟实现list中的常用接口

list的构造函数:

void empty_Mylist()
{_head = new Node;_head->next = _head;_head->prev = _head;_size = 0;
}
Mylist()
{empty_Mylist();
}

这就没什么好说的,创建一个新的结点将,将前驱指针和后驱指针都指向自己,有效数据size置为零。上述我将其分为了两部分。

list的析构函数:

		~Mylist(){clear();delete _head;_head = nullptr;}void clear(){iterator it = begin();while (it != end()){it = erase(it);}}

list的析构函数又跟vector有很大的区别,vector中的析构函数将容器内的数据置为空就好了,但是在list中需要将链表中的结点一个个清除掉,再将链表的头结点清除,所以如上我们创建一个clear函数遍历清除掉每一个结点。(上述我实现clear时用到了erase,后续我们会将它实现出来)

list的拷贝构造:
在模拟实现list中的接口的时候肯定避免不了的就是深拷贝,有深拷贝我们必然就是要实现拷贝构造。

		Mylist(const Mylist<T>& it){empty_Mylist();for (auto& e : it){push_back(e);}}

list中的赋值运算符重载:

	//赋值运算符重载void swap(Mylist<T> it){std::swap(_head, it._head);std::swap(_size, it._size);}Mylist<T>& operator= (Mylist<T> it){swap(it);return (*this);}

赋值运算符重载就不多说了,看上述代码实现。

list中的尾插实现:

		void push_back(const T& val){//创建一个新节点Node* newnode = new Node(val);//插入数据Node* tail = _head->prev;newnode->prev = tail;newnode->next = _head;tail->next = newnode;_head->prev = newnode;}

如上代码是list尾插代码实现,但是我们可以在实现insert之后,用insert来复用push_back,复用如下:

void push_back(const T& val)
{insert(end(), val);
}

list的insert实现:

		void insert(iterator pos, const T& val){Node* cur = pos._node;Node* prev = cur->prev;//创建一个新的节点Node* newnode = new Node(val);//将新的节点插入链表中prev->next = newnode;newnode->prev = prev;newnode->next = cur;cur->prev = newnode;//有效数字++_size++;}

list的erase实现:

		iterator erase(iterator pos){assert(pos != end());Node* cur = pos._node;Node* nextnode = cur->next;Node* prevnode = cur->prev;prevnode->next = nextnode;nextnode->prev = prevnode;delete cur;--_size;return iterator(nextnode);}

上述代码,就是实现erase。那么我们会有问题,就是为什么erase返回的类型是迭代器呢?什么原因呢?如果返回的不是迭代器会出现什么问题吗?

原因:返回一个指向下一个元素的迭代器可以避免重新获取迭代器的开销,特别是在循环中删除元素时。这样可以提高代码的执行效率。在删除元素后,返回下一个元素的迭代器可以保持迭代器的连续性,使得在删除元素后可以无缝地继续遍历列表。如果删除操作失败(例如,尝试删除一个不存在的元素),返回一个有效的迭代器可以避免返回一个无效的迭代器,从而减少错误处理的复杂性。

如果不这样使用会出现什么问题:如果你在遍历容器时删除元素,返回类型不是迭代器会导致你无法获取到被删除元素之后的那个元素的迭代器。这将使得你难以继续遍历容器,因为你失去了对下一个元素的引用。

其他模式实现接口:

		//有效数据size_t size() const{return _size;}//头插void push_front(const T& val){insert(begin(), val);}//尾删void pop_back(){erase(end());}//头删void pop_front(){erase(begin());}

模拟实现list代码总览

Mylist.h:

#pragma once
#include <iostream>
#include <assert.h>namespace cjc
{template<class T>//建立节点结构struct Mylist_node{Mylist_node<T>* prev;Mylist_node<T>* next;T data;//构造函数Mylist_node(const T& val = T()):prev(nullptr), next(nullptr), data(val){}};//构造list迭代器结构template<class T,class Ref,class Ptr>struct list_iterator{typedef Mylist_node<T> Node;typedef list_iterator 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++(int) //后置++需要int区分一下{Self tmp(*this); //这里最开始的类型不对_node = _node->next;return tmp;}Self& operator--(){_node = _node->prev;return (*this);}Self operator--(int){Self tmp(*this);_node = _node->prev;return tmp;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}};/*template <class T>struct list_iterator{typedef Mylist_node Node;Node* _node;list_iterator(Node* node):_node(node){}T& operator*(){return _node->data;}T operator->(){return &_node->data;}list_iterator<T>& operator++(){_node = _node->next;return (*this);}bool operator!=(const list_iterator<T>& it){return _node != it._node;}bool operator==(const list_iterator<T>& it){return _node == it._node;}};template <class T>struct list_const_iterator{typedef Mylist_node Node;Node* _node;list_const_iterator(Node* node):_node(node){}const T& operator*(){return _node->data;}const T operator->(){return &_node->data;}list_const_iterator<T>& operator++(){_node = _node->next;return _node;}bool operator!=(const list_const_iterator<T>& it){return _node != it._node;}bool operator==(const list_const_iterator<T>& it){return _node == it._node;}};*/template <class T>class Mylist{typedef Mylist_node<T> Node;public:/*typedef list_iterator<T> iterator;typedef list_const_iterator<T> const_iterator;*/typedef list_iterator<T, T&, T*> iterator;typedef list_iterator<T, const T&, const T*> const_iterator;iterator begin(){return iterator(_head->next);}iterator end(){return iterator(_head);}const_iterator begin() const{return iterator(_head->next);}const_iterator end() const{return iterator(_head);}//构造函数void empty_Mylist(){_head = new Node;_head->next = _head;_head->prev = _head;_size = 0;}Mylist(){empty_Mylist();}//析构函数~Mylist(){clear();delete _head;_head = nullptr;}void clear(){iterator it = begin();while (it != end()){it = erase(it);}}//拷贝构造Mylist(const Mylist<T>& it){empty_Mylist();for (auto& e : it){push_back(e);}}//赋值运算符重载void swap(Mylist<T> it){std::swap(_head, it._head);std::swap(_size, it._size);}Mylist<T>& operator= (Mylist<T> it){swap(it);return (*this);}//有效数据size_t size() const{return _size;}//尾插实现void push_back(const T& val){创建一个新节点//Node* newnode = new Node(val);插入数据//Node* tail = _head->prev;//newnode->prev = tail;//newnode->next = _head;//tail->next = newnode;//_head->prev = newnode;insert(end(), val);}//头插void push_front(const T& val){insert(begin(), val);}//尾删void pop_back(){erase(end());}//头删void pop_front(){erase(begin());}//插入数据void insert(iterator pos, const T& val){Node* cur = pos._node;Node* prev = cur->prev;//创建一个新的节点Node* newnode = new Node(val);//将新的节点插入链表中prev->next = newnode;newnode->prev = prev;newnode->next = cur;cur->prev = newnode;//有效数字++_size++;}//删除数据iterator erase(iterator pos){assert(pos != end());Node* cur = pos._node;Node* nextnode = cur->next;Node* prevnode = cur->prev;prevnode->next = nextnode;nextnode->prev = prevnode;delete cur;--_size;return iterator(nextnode);}private:Node* _head;size_t _size;};
}

test.cpp:

#define  _CRT_SECURE_NO_WARNINGS 1
#include "Mylist.h"
#include <iostream>void Test1()
{cjc::Mylist<int> l1;l1.push_back(1);l1.push_back(2);l1.push_back(3);l1.push_back(4);for (auto& e : l1){std::cout << e << " ";}std::cout << std::endl;
}void Test2()
{cjc::Mylist<int>l2;l2.push_front(1);l2.push_front(2);l2.push_front(3);l2.push_front(4);for (auto& e : l2){std::cout << e << " ";}std::cout << std::endl;
}void Test3()
{cjc::Mylist<int> l1;l1.push_back(1);l1.push_back(2);l1.push_back(3);l1.push_back(4);l1.erase(l1.begin());l1.erase(l1.begin());l1.erase(l1.begin());for (auto& e : l1){std::cout << e << " ";}std::cout << std::endl;
}void Test4()
{cjc::Mylist<int>l2;l2.push_front(1);l2.push_front(2);l2.push_front(3);l2.push_front(4);l2.insert(l2.begin(), 0);l2.insert(l2.begin(), 0);l2.insert(l2.begin(), 0);l2.insert(l2.begin(), 0);for (auto& e : l2){std::cout << e << " ";}std::cout << std::endl;
}void Test5()
{cjc::Mylist<int>l2;l2.push_front(1);l2.push_front(2);l2.push_front(3);l2.push_front(4);for (auto& e : l2){std::cout << e << " ";}std::cout << std::endl;l2.clear();for (auto& e : l2){std::cout << e << " ";}std::cout << std::endl;
}int main()
{尾插测试//Test1();头插测试//Test2();尾删测试//Test3();头删测试//Test4();//删除测试 Test5();return 0;
}

相关文章:

【简单模拟实现list】

在C标准模板库&#xff08;STL&#xff09;中&#xff0c;list是一个非常强大的容器&#xff0c;它基于双向链表实现&#xff0c;支持高效的插入和删除操作。虽然我们可以直接使用STL中的list&#xff0c;但通过自己模拟实现一个list&#xff0c;可以更好地理解其背后的原理和数…...

深入解析ZAB协议:ZooKeeper的分布式一致性核心

引言 在分布式系统中&#xff0c;如何高效、可靠地实现多节点间的数据一致性是核心挑战之一。ZAB协议&#xff08;ZooKeeper Atomic Broadcast&#xff09;作为 ZooKeeper的核心算法&#xff0c;被广泛应用于分布式协调服务&#xff08;如Kafka、HBase、Dubbo等&#xff09;。…...

交叉熵损失函数,KL散度, Focal loss

交叉熵损失函数&#xff08;Cross-Entropy Loss&#xff09; 交叉熵损失函数&#xff0c;涉及两个概念&#xff0c;一个是损失函数&#xff0c;一个是交叉熵。 首先&#xff0c;对于损失函数。在机器学习中&#xff0c;损失函数就是用来衡量我们模型的预测结果与真实结果之间…...

k8s部署实战-springboot应用部署

在 Kubernetes 上部署 SpringBoot 应用实战指南 前言 本文将详细介绍如何将一个 SpringBoot 应用部署到 Kubernetes 集群中,包括制作镜像、编写部署文件、创建服务等完整步骤。 准备工作 1. 示例 SpringBoot 应用 假设我们有一个简单的 SpringBoot 应用,提供 REST API 服…...

快速选择算法:优化大数据中的 Top-K 问题

在处理海量数据时&#xff0c;经常会遇到这样的需求&#xff1a;找出数据中最大的前 K 个数&#xff0c;而不必对整个数据集进行排序。这种场景下&#xff0c;快速选择算法&#xff08;Quickselect&#xff09;就成了一个非常高效的解决方案。本文将通过一个 C 实现的快速选择算…...

uniapp-商城-60-后台 新增商品(属性的选中和页面显示)

前面添加了属性&#xff0c;添加属性的子级项目。也分析了如何回显&#xff0c;但是在添加新的商品的时&#xff0c;我们也同样需要进行选择&#xff0c;还要能正常的显示在界面上。下面对页面的显示进行分析。 1、界面情况回顾 属性显示其实是个一嵌套的数据显示。 2、选中的…...

利用 Amazon Bedrock Data Automation(BDA)对视频数据进行自动化处理与检索

当前点播视频平台搜索功能主要是基于视频标题的关键字检索。对于点播平台而言&#xff0c;我们希望可以通过优化视频搜索体验满足用户通过模糊描述查找视频的需求&#xff0c;从而提高用户的搜索体验。借助 Amazon Bedrock Data Automation&#xff08;BDA&#xff09;技术&…...

项目QT+ffmpeg+rtsp(一)——Qt的安装和rtsp的测试

文章目录 一、Qt安装二、插件配置tool与卸载三、下载ffmpeg四、查看能否使用(视频)五、代码复现5.1 rtsp申请5.2 rtsp在线测试5.3代码修改六、结果一、Qt安装 对于QT中5.12版本之后,都是使用在线版本,如果你想安装某一个的历史在线版本,一定要点击archive,不然显示不出来…...

高速光耦在通信行业的应用(五) | 5Mbps通信光耦的特性

针对5MBd速率光耦市场&#xff0c;晶台推出KL2200、KL2201和KL2202系列光耦 ,对标大部分国外品牌产品的应用&#xff1b;它分别由一个红外发射二极管和一个高速集成光电检测器逻辑门组成。 它采用 8 引脚 DIP 封装&#xff0c;并提供 SMD 选项。KL2200 的检测器具有一个三态输出…...

#跟着若城学鸿蒙# web篇-运动和方向传感器监测

前言 有些前端业务场景需要用到一些传感器&#xff0c;比如运动传感器和方向传感器来实现摇一摇功能。这就需要前端能够直接获取到相关数据&#xff0c;而不是通过 js 调用客户端代码来实现。 权限 还是需要在模块的module.json5文件中添加相关权限 {"name" : &qu…...

【匹配】Hirschberg

Hirschberg 文章目录 Hirschberg1. 算法介绍2. 公式及原理3. 伪代码 1. 算法介绍 背景与目标 Hirschberg 算法由 Dan Hirschberg 于1975年提出&#xff0c;是对 Needleman–Wunsch 全局比对的内存优化&#xff0c;通过分治策略将空间复杂度从 O ( m n ) O(mn) O(mn) 降到 O (…...

如何在 Windows 上安装类似 Synaptic 的 Chocolatey GUI 包管理器

如果你正在寻找类似 Linux 中 APT 的 Windows 包管理器&#xff0c;那么没有什么比 Chocolatey 更好的了。它是 Windows 10 上可用的最佳包管理器之一&#xff0c;可以通过命令行界面安装所有流行的软件和工具。然而&#xff0c;这并不意味着如果你不喜欢命令行&#xff0c;你就…...

激活函数全解析:定义、分类与 17 种常用函数详解

一、激活函数的定义与作用 定义&#xff1a; 激活函数是添加到人工神经网络中的函数&#xff0c;用于帮助网络学习数据中的复杂模式&#xff0c;决定神经元的输出。 核心作用&#xff1a; 为神经网络引入非线性&#xff0c;增强模型表达能力。需可微分&#xff08;或近似可微&…...

1-10 目录树

在ZIP归档文件中&#xff0c;保留着所有压缩文件和目录的相对路径和名称。当使用WinZIP等GUI软件打开ZIP归档文件时&#xff0c;可以从这些信息中重建目录的树状结构。请编写程序实现目录的树状结构的重建工作。 输入格式: 输入首先给出正整数N&#xff08;≤104&#xff09;…...

Python OOP核心技巧:如何正确选择实例方法、类方法和静态方法

Python方法类型全解析&#xff1a;实例方法、类方法与静态方法的使用场景 一、三种方法的基本区别二、访问能力对比表三、何时使用实例方法使用实例方法的核心场景&#xff1a;具体应用场景&#xff1a;1. 操作实例属性2. 对象间交互3. 实现特定实例的行为 四、何时使用类方法使…...

RK3588 ADB使用

安卓adb操作介绍 adb&#xff08;Android Debug Bridge&#xff09;是一个用于与安卓设备进行通信和控制的工具。adb可以通过USB或无线网络连接安卓设备&#xff0c;执行各种命令&#xff0c;如安装和卸载应用&#xff0c;传输文件&#xff0c;查看日志&#xff0c;运行shell命…...

ubuntu环境下 基于Python 打包的 批量命令行可视化操作工具 GUI

文章目录 一.需求&#xff1a;二.原理支撑&#xff1a;三.简单Demo四.封装成GUI1.依赖库2.代码 五.打包成可执行文件六.命令行的配置七.运行效果 一.需求&#xff1a; 作为测试工程师&#xff0c;为了到现场高效的调试&#xff0c;部署工作&#xff0c;需要一个可视化的工具&a…...

大语言模型 10 - 从0开始训练GPT 0.25B参数量 补充知识之模型架构 MoE、ReLU、FFN、MixFFN

写在前面 GPT&#xff08;Generative Pre-trained Transformer&#xff09;是目前最广泛应用的大语言模型架构之一&#xff0c;其强大的自然语言理解与生成能力背后&#xff0c;是一个庞大而精细的训练流程。本文将从宏观到微观&#xff0c;系统讲解GPT的训练过程&#xff0c;…...

SkyWalking的工作原理和搭建过程

SkyWalking 是一个开源的 应用性能监控系统&#xff08;APM&#xff09;&#xff0c;专为云原生、微服务架构设计。其核心原理基于 分布式追踪&#xff08;Distributed Tracing&#xff09;、指标收集&#xff08;Metrics Collection&#xff09; 和 日志关联&#xff08;Log C…...

CMS(plone / joomla 搭建测试)

开源选择 wordpress 用得最多 也是最容易有漏洞被攻击 被挂木马的 joomla &#xff08;JMS多站点&#xff1a;商业扩展&#xff09; — 多站点需付费 Drupal ProcessWire Plone因其内置的强大安全特性和较少的用户基础&#xff08;相比 WordPress 和 Joomla&#xff09;&#…...

基于 Flink 的实时推荐系统:从协同过滤到多模态语义理解

基于 Flink 的实时推荐系统&#xff1a;从协同过滤到多模态语义理解 嘿&#xff0c;各位技术小伙伴们&#xff01;在这个信息爆炸的时代&#xff0c;你是不是常常惊叹于各大平台仿佛能 “读懂你的心”&#xff0c;精准推送你感兴趣的内容呢&#xff1f;今天&#xff0c;小编就…...

Flink SQL、Hudi 、Doris在数据上的组合应用

Flink SQL、Hudi 和 Doris 是大数据领域中不同定位的技术组件&#xff0c;各自解决不同的问题&#xff0c;以下从核心定位、关键特性和典型场景三个维度展开说明&#xff1a; 1. Flink SQL&#xff1a;流批统一的实时计算引擎 核心定位&#xff1a;Flink 是 Apache 顶级的流批…...

Flink运维要点

一、Flink 运维核心策略 1. 集群部署与监控 资源规划 按业务优先级分配资源&#xff1a;核心作业优先保障内存和 CPU&#xff0c;避免资源竞争。示例&#xff1a;为实时风控作业分配专用 TaskManager&#xff0c;配置 taskmanager.memory.process.size8g。 监控体系 集成 Prom…...

VSCode + Cline AI辅助编程完全指南

VSCode Cline AI辅助编程完全指南 在当今AI快速发展的时代&#xff0c;程序员可以通过AI工具极大地提高工作效率。本教程将详细介绍如何使用VSCode结合Cline&#xff08;Claude AI助手&#xff09;进行AI辅助编程&#xff0c;帮助你提高开发效率&#xff0c;解决复杂问题。 …...

【源码级开发】Qwen3接入MCP,企业级智能体开发实战!

Qwen3接入MCP智能体开发实战&#xff08;上&#xff09; 一、MCP技术与Qwen3原生MCP能力介绍 1.智能体开发核心技术—MCP 1.1 Function calling技术回顾 如何快速开发一款智能体应用&#xff0c;最关键的技术难点就在于如何让大模型高效稳定的接入一些外部工具。而在MCP技术…...

回调函数应用示例

回调函数是一种通过函数指针&#xff08;或引用&#xff09;调用的函数&#xff0c;它在特定事件或条件发生时被另一个函数调用。回调函数的核心思想是将函数作为参数传递&#xff0c;以便在适当的时候执行自定义逻辑&#xff0c;常用于异步编程、事件驱动架构等场景。 业务场景…...

R语言如何解决导出pdf中文不显示的问题

前言 以前绘图都默认英文&#xff0c;突然要求都改成中文&#xff0c;呆住。。。。。。。。。 标题代码实现 ### 导入工具包 ### library(readr) library(dplyr) library(corrplot)df <- read_csv("./clinical.csv") df <- df %>% select(-id, -label)##…...

国产linux系统(银河麒麟,统信uos)使用 PageOffice自定义Word模版中的数据区域

​ PageOffice 国产版 &#xff1a;支持信创系统&#xff0c;支持银河麒麟V10和统信UOS&#xff0c;支持X86&#xff08;intel、兆芯、海光等&#xff09;、ARM&#xff08;飞腾、鲲鹏、麒麟等&#xff09;、龙芯&#xff08;Mips、LoogArch&#xff09;芯片架构。 在实际的Wor…...

llamafactory SFT 从断点恢复训练

背景 我使用llamafactory sft 微调模型的时候。gpu停止运行了。日志文件没有任何的报错信息。 显存还是占用状态。 查看llamafactory的进程是下述信息&#xff1a; 151312 151306 91 17:42 ? 03:58:10 [llamafactory-cl] 既然如此&#xff0c;那就只能从断点恢复训练了。 …...

C#里使用Prism.Core的例子

由于使用WPF来开发应用程序, 那么就会使用一些框架程序来加速开发,一般会使用Prism.Core来加速。 这个应用最后运行的显示如下: 第一步需要安装下面的包: <?xml version="1.0" encoding="utf-8"?> <packages><package id="Mi…...

【MySQL】数据库三大范式

目录 一. 什么是范式 二. 第一范式 三. 第二范式 不满足第二范式时可能出现的问题 四. 第三范式 一. 什么是范式 在数据库中范式其实就是一组规则&#xff0c;在我们设计数据库的时候&#xff0c;需要遵守不同的规则要求&#xff0c;设计出合理的关系型数据库&#xff0c;…...

window 显示驱动开发-分页视频内存资源

与 Microsoft Windows 2000 显示驱动程序模型不同&#xff0c;Windows Vista 显示驱动程序模型允许创建比可用物理视频内存总量更多的视频内存资源&#xff0c;然后根据需要分页进出视频内存。 换句话说&#xff0c;并非所有视频内存资源都同时位于视频内存中。 GPU 的管道中可…...

炼丹学习笔记3---ubuntu2004部署运行openpcdet记录

前言 环境 cuda 11.3 python 3.8 ubuntu2004 一、cuda环境检测 ylhy:~/code_ws/OpenPCDet/tools$ nvcc -V nvcc: NVIDIA (R) Cuda compiler driver Copyright (c) 2005-2021 NVIDIA Corporation Built on Sun_Mar_21_19:15:46_PDT_2021 Cuda compilation tools, release 11.3…...

美创科技针对《银行保险机构数据安全管理办法》解读

在数字化浪潮席卷下&#xff0c;银行保险业的运营模式发生了翻天覆地的变化&#xff0c;数据已然成为行业发展的核心驱动力。从客户基本信息、交易记录&#xff0c;到业务运营的关键数据、市场分析报告&#xff0c;海量数据背后潜藏巨大价值。然而&#xff0c;数据安全风险也随…...

activeMq 限制用户接收topic范围

1、在conf配置文件中找到jetty-realm.properties文件&#xff0c;添加用户信息 2、在broker标签中加入topic限制权限信息 <plugins><simpleAuthenticationPlugin><users><authenticationUser username"admin" password"admin" group…...

LIIGO ❤️ RUST 12 YEARS

LIIGO &#x1f496; RUST 12 YEARS 今天是RUST语言1.0发布十周年纪念日。十年前的今天&#xff0c;2015年的今天&#xff0c;Rust 1.0 正式发行。这是值得全球Rust支持者隆重纪念的日子。我借此机会衷心感谢Rust语言创始人Graydon Hoare&#xff0c;Mozilla公司&#xff0c;以…...

增量学习:机器学习领域中的资源高效利用秘籍

前言 在机器学习的广袤天地中&#xff0c;增量学习宛如一颗冉冉升起的新星&#xff0c;正逐渐展现出其独特的魅力和巨大的潜力。 它是一种能让 AI 模型像人类一样&#xff0c;逐步学习并不断强化自身知识&#xff0c;同时不会遗忘过往所学信息的学习方法。随着时代的飞速发展&a…...

OpenCV 背景建模详解:从原理到实战

在计算机视觉领域&#xff0c;背景建模是一项基础且重要的技术&#xff0c;它能够从视频流中分离出前景目标&#xff0c;广泛应用于运动目标检测、视频监控、人机交互等场景。OpenCV 作为计算机视觉领域最受欢迎的开源库之一&#xff0c;提供了多种高效的背景建模算法。本文将深…...

makefile细节说明

在 Makefile中&#xff0c;依赖关系的左右两部分有特定的名称&#xff1a; ​​左边部分&#xff08;冒号左侧&#xff09;​​ 称为 ​​目标&#xff08;Target&#xff09;​​ ​​右边部分&#xff08;冒号右侧&#xff09;​​ 称为 ​​依赖项&#xff08;Prerequisite…...

计算机图形学编程(使用OpenGL和C++)(第2版)学习笔记 10.增强表面细节(二)法线贴图

1. 法线贴图&#xff08;Normal Mapping&#xff09; 法线贴图是一种在3D图形渲染中广泛使用的表面细节增强技术。它通过存储每个像素的法线信息来模拟表面的细微凹凸细节&#xff0c;而无需增加实际的几何复杂度。 1.1. 工作原理 纹理存储 使用RGB通道存储法线向量的XYZ分量…...

使用 OpenCV 将图像中标记特定颜色区域

在计算机视觉任务中&#xff0c;颜色替换是一种常见的图像处理操作&#xff0c;广泛用于视觉增强、目标高亮、伪彩色渲染等场景。本文介绍一种简单而高效的方式&#xff0c;基于 OpenCV 检测图像中接近某种颜色的区域&#xff0c;并将其替换为反色&#xff08;对比色&#xff0…...

Service Mesh

目录 一、Service Mesh 的核心特点 二、Service Mesh 的典型架构 1. Sidecar 模式 2. 控制平面与数据平面分离 三、Service Mesh 解决的核心问题 四、典型应用场景 五、主流 Service Mesh 框架对比 六、挑战与局限性 七、未来趋势 总结 Istio 一、Istio 核心组件与…...

反射机制详细说明

反射机制详细说明 1. 反射的基本概念 反射(Reflection)是Java提供的一种在运行时(Runtime)动态获取类信息并操作类属性、方法和构造器的机制。通过反射,程序可以在运行时检查类、接口、字段和方法,并且可以实例化对象、调用方法、访问或修改字段值,甚至操作私有成员,…...

基于Mongodb的分布式文件存储实现

分布式文件存储的方案有很多&#xff0c;今天分享一个基于mongodb数据库来实现文件的存储&#xff0c;mongodb支持分布式部署&#xff0c;以此来实现文件的分布式存储。 基于 MongoDB GridFS 的分布式文件存储实现&#xff1a;从原理到实战 一、引言 当系统存在大量的图片、…...

相机Camera日志分析之九:高通相机Camx 基于预览1帧的ConfigureStreams二级日志分析详解

【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:高通相机Camx 日志分析之三:camx hal预览1帧logcat日志opencamera详解 相机Camera日志分析之三:不想输出每秒30帧巨量日志,如何只输出1帧日志作为学习使用? 这一篇我们开始讲: 高通相机Camx 日志…...

neo4j框架:ubuntu系统中neo4j安装与使用教程

在使用图数据库的时候&#xff0c;经常需要用到neo4j这一图数据库处理框架。本文详细介绍了neo4j安装使用过程中的问题与解决方法。 一、安装neo4j 在安装好了ubuntu系统、docker仓库和java的前提下 在ubuntu系统命令行依次输入如下命令&#xff1a; # 安装依赖库 sudo apt-…...

k8s灰度发布

基于 Traefik 的加权灰度发布-腾讯云开发者社区-腾讯云 Traefik | Traefik | v1.7 Releases traefik/traefik GitHub 从上面连接下载后上传到harbor虚拟机 vagrant upload /C/Users/HP280/Downloads/traefik 下载配置文件 wget -c http://raw.githubusercontent.com/conta…...

K8S从Harbor拉取镜像

参考 配置cri-docker使kubernetes1.24以docker作为运行时_启动cirdocker_跳跃音符#3712的博客-CSDN博客 部署Harbor私有容器镜像仓库并配置Kubernetes从Harbor拉取镜像的方法_运维个西瓜的博客-CSDN博客 K8S连接Harbor私有仓库_k8s harbor 登录-CSDN博客 K8S集群配置使用私…...

【Spring Boot后端组件】mybatis-plus使用

文章目录 mybatis-plus使用一、依赖引入二、添加相关配置项三、功能详解1.自增主键2.逻辑删除3.操作时间自动填充4.其他字段自动填充5.分页查询6.自定义动态查询7.代码生成器8.代码生成器(自定义模板) mybatis-plus使用 一、依赖引入 pom.xml文件 <?xml version"1.…...

Oc语言学习 —— 重点内容总结与拓展(下)

类别&#xff08;分类&#xff09;和拓展 分类&#xff1a; 专门用来给类添加新方法 不能给类添加成员属性&#xff0c;添加成员属性也无法取到 注意&#xff1a;其实可与通过runtime 给分类添加属性&#xff0c;即属性关联&#xff0c;重写setter&#xff0c;getter方法 分类…...