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

C++《list的模拟实现》

在上一篇C++《list》专题当中我们了解了STL当中list类当中的各个成员函数该如何使用,接下来在本篇当中我们将试着模拟实现list,在本篇当中我们将通过模拟实现list过程中深入理解list迭代器和之前学习的vector和string迭代器的不同,接下来就开始本篇的学习吧!!!


1.实现各个函数之前的工作 

在list模拟实现中由于是使用类模板来模拟实现list,因此在此list内的函数声明与定义就不分离,这里的原因接下来在模板进阶篇章中会进行讲解。

所以只需要两个文件list.cpp与test.cpp;在list.cpp内实现list类,在test测试实现的list各个成员函数是否满足我们的要求。并且为了避免我们模拟实现的list和std命名空间内的std冲突,在此要将模拟实现的list放在我们新创建的命名空间内

完成了程序文件的实现接下来来实现list类内的成员变量,由于STL内的list其实是一个双链表也就是带头双向循环链表,因此和之前在数据结构内实现链表一样先要实现一个结构体来表示链表的节点

注:链表的节点也是用模板来实现这样就可以使得链表的节点可以存储任意类型的数据。并且由于要实现的是双链表因此链表的节点当中有三个数据分别是存储的数据、指向前一个节点的指针、指向后一个节点的指针

#include<iostream>
using namespace std;namespace zhz
{template<class T>struct list_node{T date;list_node<T>* prev;list_node<T>* next;list_node(const T& x = T()):date(x), prev(nullptr), next(nullptr){}};}

由于以上的类在之后创建的list类当中使用到list_node内的成员函数以及成员变量,因此就直接将该类用struct来创建,这样就会使得内部的成员函数以及成员变量默认都是公有的,这是你可能有会有疑惑这样不会使得用户可以修改程序底层的数据破坏原有的封装了吗?

在此其实时不会出现这样的问题,这是因为用户在使用时时无法感受到容器底层的结构的,就比如在学习list底层之前用户是无法感知到list底层是带头的链表,无法感知到底层的节点存储着什么数据,其实在此就是一种隐形的封装。所以就算程序底层的一些是公有的,但是对应用户来说也是属于封闭的,就像是“黑箱”一样

在以上链表节点的结构体当中我们还实现了默认构造函数,这样就可以让之后每创建一个节点都能在定义之后自动初始化

实现了表示链表节点的结构体之后接下来就可以实现list类内的成员变量了

#include<iostream>
using namespace std;namespace zhz
{template<class T>struct list_node{T date;list_node<T>* prev;list_node<T>* next;list_node(const T& x = T()):date(x), prev(nullptr), next(nullptr){}};template<class T>class list{typedef list_node<T> Node;public://成员函数……private:Node* head;size_t _size;};}

在此list类的成员变量为两个;一个是list对象内的头节点head,另一个list对象内的有效节点数size 

2. list模拟实现

在以上我们实现了程序文件的创建、list类内成员变量的实现,接下来就可以一一模拟实现list内的成员函数了

2.1 无参构造函数

在list的构造函数中我们先实现无参的构造函数其他的构造函数在实现了插入函数insert之后再实现,这样的原因是使用insert来插入就不需要我我们显示的开空间而是将这些工作交给inert函数来实现,这样其他的构造函数写起来就较为简洁

在list内无参的构造函数中由于list底层要实现的是带头双向循环链表,因此在无参构造时要创建一个头节点也就是哨兵位节点

在list.cpp内实现无参构造函数,代码如下所示:

template<class T>
class list
{typedef list_node<T> Node;public://成员函数……list(){EmptyInit();}private:Node* head;size_t _size;void EmptyInit(){head = new Node();head->next = head;head->prev = head;}};

2.2 size与empty

在此size和empty的函数实现较为简单,以下就直接实现代码

代码如下所示:

size_t size()const
{return _size;
}bool empty()
{return _size == 0;
}

2.3 迭代器

在模拟实现list当中我们需要重点学习的就是list内的迭代器该如何实现,前面说过,大家可将迭代器暂时理解成类似于指针,但是在list就不能这样认为了;这是由于list底层节点物理空间不一定是连续的,所以我们就不能简单认为迭代器就是指针,那么接下来就来分析list内的迭代器该如何实现

首先要来分析的是和之前实现string和vector的迭代器不同由于无法使用原生指针来实现迭代器,在之前我们了解了list的迭代器是属于双向迭代器,那么在之后list迭代器要能实现迭代器的++与--,这就使得要实现这两个运算符的重载函数。这时你可能会简单的认为直接在list类内实习运算符重载函数不就可以实现要求的了,但是在此就会存在两个非常严重的问题:
首先是在不同的容器实现迭代器就是为了在用户使用时屏蔽底层的细节,屏蔽不同容器底层结构上的差异,通过封装底层的差异与细节来给用户实现统一的访问方式,因此如果在list内实现迭代器就会使得在之后的list迭代器++或者--时就直接通用对对象++或者--就能实现操作,这种实现不就和我们实现迭代器的初衷违背了吗?

其次就是如果是将迭代器实现在list类内,当我们对一个对象进行++或者--之后,该对象内底层的指向头节点指针不就改变了吗?这就会造成之后无法找到头节点,这就会使得之后进行的操作会出现各种问题,要解决这个问题就需要在list类内再创建多个指向头节点的指针,但是这样的话要创建多少个呢,如果是多个迭代器同时遍历list对象那么存在创建的头节点指针数不够怎么办

通过以上的分析就可以得出在list要实现迭代器就不能将迭代器实现在list类内部,那么正确的解决方式是什么呢?

在此合理的方法是载创建一个类list_iterator去封装节点的指针,将list对象内节点的指针作为该类的成员变量,之后使用这个新的类来作为迭代器。在此封装了节点的指针之后就可以重载我们想要实现的*、++、--等的运算符。并且这种实现迭代器的方式就不会出现以上的问题

那么接下来就来实现实现list_iterator类

由于在list类以及之后用户在实现list的迭代器时都会调用list_iterator的内部成员,因此list_iterator也和list_node一样不做访问限定符的限制,在此也使用struct来定义类

以下就先来实现list_iterator内的构造函数

template<class T>
struct list_iterator
{//为了简化之后的代码,将以下的两个类型重命名typedef list_node<T> Node;typedef list_iterator<T> Self;list_iterator(Node* node):_node(node){}//节点指针Node* _node;
};

我们知道在链表的节点解引用时想要得到的是对应节点内的数据data,在此接下来就在list_iterator内重载*运算符

T& operator*()
{return _node->date;
}

为了能实现对节点内的数据进行读和写,那么就下需要想以上一样将*运算符重载的函数返回值为该节点内数据的引用

接下来来实现迭代器中的++与--,在此由于list为双向迭代器因此我们要实现前置++与--、后置++与-- 

//前置++
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;
}

注:在此的后置++和后置--=要加一个形参int的原因在之前C++《类和对象》章节就讲解过,这是为了使得前置和后置函数构成函数重载,后置加一个参数便于区分

在迭代器的使用当中通常还会判断两个迭代器是否相等,那么接下来就来实现==与!=的运算符重载函数

bool operator!=(const Self& l)
{return _node != l._node;
}bool operator==(const Self& l)
{return _node == l._node;
}

接下来我们还要实现一个之前没有实现过的运算符->,要实现这个运算符是因为list对象的类型可以是自定义类型,那么当类型是自定义类型时以上实现的迭代器使用*得到的是整个自定义类型对象,那么如果我们要得到的是该自定义类型内的数据就需要再通过再一次解引用才能实现。

那么为了能一步实现以上的操作就来实现运算符->

T* operator->()
{return &_node->data;
}

注:在使用以上操作符时,当list对象为自定义类型时,要得到自定义类型对象内的数据正常应该是要迭代器->->自定义类型对象内的变量,但在此为了可读性就省略一个->,变成迭代器->自定义类型对象内的变量

以上我们就完成了list的普通迭代器,那么const迭代器该如何实现呢?

在此你会认为再创建一个const_list_iterator就可以实现const迭代器,只需要将该类内的部分函数的返回值修改就可以满足需求了。

以上这种方式也是可以满足要求的,但以上这样实现虽然能满足要求但是两个const_list_iterator和list_iterator高度的相识,这样就会使得代码很冗余,那么该如何实现呢?

其实在原本的list_iterator类的模板参数再加两个就可以解决,这就不需要实现两个类了

实现代码如下所示:

注:在此类模板的第一个参数T表示list对象内存储的数据类型,第二个参数Ref表示的是T类型的引用,第三个参数Pre表示T类型的指针

template<class T, class Ref, class Ptr>
struct list_iterator
{//为了简化之后的代码,将以下的两个类型重命名typedef list_node<T> Node;typedef list_iterator<T, Ref, Ptr> Self;list_iterator(Node* node):_node(node){}Ref operator*(){return _node->date;}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& l){return _node != l._node;}bool operator==(const Self& l){return _node == l._node;}Node* _node;};

这时list类实现的begin()和end()函数就如下所示:

template<class T>
class list
{typedef list_node<T> Node;
public:
//为了将保证的用户能使用list的迭代器需要将以上我们创建的迭代器类型进行重命名typedef list_iterator<T, T&, T*> iterator;typedef list_iterator<T, const T&, const T*> const_iterator;public:list(){EmptyInit();}iterator begin(){return iterator(head->next);}iterator end(){return iterator(head);}const_iterator begin()const{return const_iterator(head->next);}const_iterator end()const{return const_iterator(head);}private:Node* head;size_t _size;void EmptyInit(){head = new Node();head->next = head;head->prev = head;}};

注:前面说过,此处大家可将迭代器暂时理解成类似于指针,迭代器失效即迭代器所指向的节点的无效,即该节点被删除了。因为list的底层结构为带头结点的双向循环链表,因此在list中进行插入时是不会导致list的迭代器失效的,只有在删除时才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不会受到影响。

2.4 insert与erase

通过之前的学习我们知道list内实现了insert和erase来分别实现任意位置的插入和删除,并且要删除和插入的位置是通过相应的迭代器位置实现,接下来我们就来试着实现这两个函数的代码

先来实现insert函数的代码
在insert函数当中我们要实现的操作是在pos迭代器之前插入指定的值,要实现这个操作就需要先创建一个新的节点之后将指定的值存储到节点当中,之后改节点插入到pos迭代器指向的节点和pos迭代器指向的节点之前的节点中间。以上要实现操作就和之前我们数据结构中学习的双链表任意位置插入数据实现过程类型

实现代码如下所示:

iterator insert(iterator pos, const T& x)
{Node* cur = pos._node;Node* newnode = new Node(x);Node* Prev = cur->prev;//Prev newnode  cur Prev->next = newnode;cur->prev = newnode;newnode->prev = Prev;newnode->next = cur;++_size;return iterator(newnode);
}

接下来来实现erase函数的代码

在erase函数当中我们要实现的操作是将pos迭代器指向的节点删除,要实现这个操作就需要先将原来pos迭代器指向的节点之前的节点的next指针指向原来pos迭代器指向的节点之后的节点,将原来pos迭代器指向的节点之后的节点的prev指针指向原来pos迭代器指向的节点之前的节点,之后再将原pos迭代器指向的节点释放,最后返回新节点的迭代器。以上要实现操作就和之前我们数据结构中学习的双链表任意位置删除数据类似

实现代码如下所示:

iterator erase(iterator pos)
{Node* cur = pos._node;Node* Prev = cur->prev;Node* Next = cur->next;//Prev cur NextPrev->next = Next;Next->prev = Prev;delete cur;pos = Next;--_size;return iterator(pos);
}

2.5 push_back、push_front、pop_back、pop_front

实现了insert和erase之后要实现头尾插入与删除就简单了,在这些函数内部直接通过调用之前实现的insert和erase就能实现要求了

实现代码如下所示:

void push_back(const T& x)
{insert(end(), x);
}void pop_back()
{erase(--end());
}void push_front(const T& x)
{insert(begin(), x);
}void pop_front()
{erase(begin());
}

2.6 构造函数(含参)

在一开始我们实现了无参的构造函数,但仅仅这一个函数是无法满足我们的要求的,在此还要实现使用迭代器区间的构造、n个指向值构造、拷贝构造等构造函数。

由于在以上我们实现了插入函数,那么接下来实现构造函数就很简单了,在构造函数当中将数据插入到对象内就直接通过调用实现的插入函数就可以实现了

实现代码如下所示:

//拷贝构造
list(const list<T>& lt)
{EmptyInit();for (auto& i : lt){push_back(i);}
}//n个指定的值x构造
list(int n, const T& x)
{EmptyInit();for (int i = 0; i < n; i++){push_back(x);}
}//迭代器区间构造
template<class InputIterator>
list(InputIterator first, InputIterator fin)
{EmptyInit();while (first != fin){push_back(*first);++first;}}

2.7 析构函数

在list内由于成员变量是有资源的申请的,那么编译器自动生成的析构函数就无法满足要求,需要我们显示的写析构函数。在此在析构函数内要实现的是将链表的节点一一 释放(包括头节点)

在list.cpp内实现析构函数,代码如下所示:

~list(){clear();delete head;head = nullptr;}void clear(){auto s = begin();while (s != end()){s=erase(s);}}

2.8 swap

在此在list类当中实现一个函数,在list类外也要实现一个swap函数,这样就会在我们使用参数为两个list对象的swap不会调用到算法库内的swap函数,这和之前在vector章节实现两个swap的原因类型

实现代码如下所示:

//list类内的swap
void swap(list<T>& lt)
{std::swap(head, lt.head);std::swap(_size, lt._size);
}//list类外的swap函数
template<class T>
void swap(list<T>& lt1, list<T>& lt2)
{lt1.swap(lt2);
}

2.9 赋值运算符重载

在此在模拟实现的list类内赋值运算符的重载函数我们可以直接借助swap来实现

实现代码如下所示:

list<T>& operator=(list<T> tmp)
{swap(tmp);return *this;
}

2.10 front和back

在list当中front和back函数是用于分别得到list对象当中链表第一个有效节点和尾节点

实现代码如下所示:

T& front()
{return head->next->date;
}const T& front()const
{return head->next->date;
}
T& back()
{return head->prev->date;
}const T& back()const
{return head->prev->date;
}


 

3.完整代码

#include<iostream>
using namespace std;namespace zhz
{template<class T>struct list_node{T date;list_node<T>* prev;list_node<T>* next;list_node(const T& x = T()):date(x), prev(nullptr), next(nullptr){}};template<class T, class Ref, class Ptr>struct list_iterator{typedef list_node<T> Node;typedef list_iterator<T, Ref, Ptr> Self;list_iterator(Node* node):_node(node){}Ref operator*(){return _node->date;}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& l){return _node != l._node;}bool operator==(const Self& l){return _node == l._node;}Node* _node;};template<class T>class list{typedef list_node<T> Node;public:typedef list_iterator<T, T&, T*> iterator;typedef list_iterator<T, const T&, const T*> const_iterator;public:list(){EmptyInit();}//拷贝构造list(const list<T>& lt){EmptyInit();for (auto& i : lt){push_back(i);}}//n个指定的值x构造list(int n, const T& x){EmptyInit();for (int i = 0; i < n; i++){push_back(x);}}//迭代器区间构造template<class InputIterator>list(InputIterator first, InputIterator fin){EmptyInit();while (first != fin){push_back(*first);++first;}}list<T>& operator=(list<T> tmp){swap(tmp);return *this;}~list(){clear();delete head;head = nullptr;}size_t size()const{return _size;}bool empty(){return _size == 0;}void clear(){auto s = begin();while (s != end()){s=erase(s);}}iterator begin(){return iterator(head->next);}iterator end(){return iterator(head);}const_iterator begin()const{return const_iterator(head->next);}const_iterator end()const{return const_iterator(head);}void push_back(const T& x){insert(end(), x);}void pop_back(){erase(--end());}void push_front(const T& x){insert(begin(), x);}void pop_front(){erase(begin());}iterator insert(iterator pos, const T& x){Node* cur = pos._node;Node* newnode = new Node(x);Node* Prev = cur->prev;//Prev newnode  cur Prev->next = newnode;cur->prev = newnode;newnode->prev = Prev;newnode->next = cur;++_size;return iterator(newnode);}iterator erase(iterator pos){Node* cur = pos._node;Node* Prev = cur->prev;Node* Next = cur->next;//Prev cur NextPrev->next = Next;Next->prev = Prev;delete cur;pos = Next;--_size;return iterator(pos);}void swap(list<T>& lt){std::swap(head, lt.head);std::swap(_size, lt._size);}T& front(){return head->next->date;}const T& front()const{return head->next->date;}T& back(){return head->prev->date;}const T& back()const{return head->prev->date;}private:Node* head;size_t _size;void EmptyInit(){head = new Node();head->next = head;head->prev = head;}};template<class T>void swap(list<T>& lt1, list<T>& lt2){lt1.swap(lt2);}}

以上就是《list的模拟实现》章节的全部内容了,希望能得到你的点赞和收藏

相关文章:

C++《list的模拟实现》

在上一篇C《list》专题当中我们了解了STL当中list类当中的各个成员函数该如何使用&#xff0c;接下来在本篇当中我们将试着模拟实现list&#xff0c;在本篇当中我们将通过模拟实现list过程中深入理解list迭代器和之前学习的vector和string迭代器的不同&#xff0c;接下来就开始…...

Kubernetes的概述与架构

Kubernetes 的概述 Kubernetes 是一个可移植、可扩展的开源平台&#xff0c;用于管理容器化的工作负载和服务&#xff0c;方便进行声明式配置和自动化。Kubernetes 拥有一个庞大且快速增长的生态系统&#xff0c;其服务、支持和工具的使用范围广泛。 Kubernetes 这个名字源于…...

Elasticsearch实战应用:构建高效的全文搜索引擎

Elasticsearch实战应用&#xff1a;构建高效的全文搜索引擎 在当今信息爆炸的时代&#xff0c;如何快速、准确地从海量数据中检索出所需信息成为了企业和开发者面临的重要挑战。Elasticsearch作为一款开源的分布式搜索引擎&#xff0c;凭借其强大的全文搜索、实时分析和可扩展…...

达梦数据库和人大金仓数据库对数据库的运行查看情况

1、查看服务器自身资源使用情况 查看内存&#xff1a; free -g 查看整体负载: top 查看磁盘io &#xff1a; iostat -d -x 1 2、查看数据库占用服务器内存情况,登录DM管理工具&#xff0c;达梦数据库使用的内存大致等于 BUFFER MPOOL&#xff0c;对应的 SQL 语句为&#xff1a…...

Spring Boot解决 406 错误之返回对象缺少Getter/Setter方法引发的问题

目录 前言1. 问题背景2. 问题分析2.1 检查返回对象 3. 解决方案3.1 确保Controller返回Result类型3.2 测试接口响应 4. 原理探讨5. 常见问题排查与优化建议结语 前言 在Spring Boot开发中&#xff0c;接口请求返回数据是系统交互的重要环节&#xff0c;尤其在开发RESTful风格的…...

Automa入门教程详解(Automa工作流概述)

一、什么是工作流&#xff1f; 工作流其实就是一组功能模块&#xff0c;通过彼此的连接来完成一系列的自动化操作流程。你可以把它理解为一个流程图&#xff0c;系统会根据你设置的顺序&#xff0c;从触发块开始&#xff0c;一步一步地执行&#xff0c;直到最后一个模块。这让…...

Python并发编程库:Asyncio的异步编程实战

Python并发编程库&#xff1a;Asyncio的异步编程实战 在现代应用中&#xff0c;并发和高效的I/O处理是影响系统性能的关键因素之一。Python的asyncio库是专为异步编程设计的模块&#xff0c;提供了一种更加高效、易读的并发编程方式&#xff0c;适用于处理大量的I/O密集型任务…...

vueui vxe-form 分享实现表单项的联动禁用,配置式表单方式的用法

官网文档&#xff1a;https:/vxeui.com 实现表单项的联动禁用 在使用 vxe-form 时&#xff0c;有时候需要将表单项直接进行关联操作&#xff0c;比如某一项选择后&#xff0c;另外一项设置为禁用状态不可选择&#xff0c;使用插槽的话神容易实现&#xff0c;本章是分享配置式的…...

供应SW1655集成功率管的高频率、高性能同步整流

概述 SW1655 是一款集成 N 沟道 MOSFET 的高频率、高性能同步整流功率开关。针对离线式反激 变换器的副边同步整流应用&#xff0c;替代肖特基整流二极管&#xff0c;可以显著提高系统效率的同时&#xff0c;实现高集 成度与高功率密度。 SW1655 具有 VCC 自供电功能&#…...

电动机出现故障后怎么处理?

在工业生产中&#xff0c;电动机作为重要的驱动设备&#xff0c;其运行状态直接关系到生产线的效率和稳定性。然而&#xff0c;由于各种因素的影响&#xff0c;电动机在运行过程中可能会出现多种故障。那么&#xff0c;电动机出现故障后怎么处理&#xff1f; 一、电动机在工业…...

练习LabVIEW第四十题

学习目标&#xff1a; 用labvIEW做一个循环闪烁指示灯&#xff0c;要能够在前面板调节周期和占空比。 开始编写&#xff1a; 前面板 一个布尔指示灯一维数组&#xff0c;两个数值输入控件&#xff1b; 程序框图 添加一个while循环&#xff0c;循环内添加初始化数组&…...

蓝牙BLE开发——红米手机无法搜索蓝牙设备?

解决 红米手机&#xff0c;无法搜索附近蓝牙设备 具体型号当时忘记查看了&#xff0c;如果你遇到有以下选项&#xff0c;记得打开~ 设置权限...

UE5.4 PCG Layered Biomes插件

B站学习链接 官方文档 一、PCGSpawn Preset&#xff1a;负责管理PCG要用到的植被资产有哪些 二、BiomesSettings&#xff1a;设置要使用的植被资产Layer、Spawn参数 1.高度Layer参数&#xff1a; 2.地形Layer&#xff1a;我这里用地形样条线绘制了一块地形Layer 绘制点和…...

搭建你的私人云盘:使用File Browser与cpolar实现公网传输文件

文章目录 前言1.下载安装File Browser2.启动访问File Browser3.安装cpolar内网穿透3.1 注册账号3.2 下载cpolar客户端3.3 登录cpolar web ui管理界面3.4 创建公网地址 4.固定公网地址访问 前言 File Browser是一个开源的文件管理器和文件共享工具&#xff0c;它可以帮助用户轻…...

QT/QT QUICK与前端WEB开发的区别

‌ ‌开发框架与目标‌&#xff1a; ‌QT/QT QUICK‌&#xff1a;跨平台应用程序开发框架&#xff0c;用于创建图形用户界面(GUI)&#xff0c;特别适用于移动和嵌入式设备。‌前端WEB开发‌&#xff1a;主要关注Web应用的用户界面&#xff0c;使用HTML、CSS、JavaScript等技术。…...

Python+Playwright(Nuitka、Pyinstaller打包)

安装驱动 playwright install # 这个安装所有默认的浏览器 playwright install chromium # 一般只装这一个浏览器就够了&#xff0c;要是装另外两个浏览器&#xff0c;后面的参数名可以修改查看各个驱动的位置 playwright install --dry-run创建打包目录 在运行的包里面…...

2024年前三季度币安、OKX等五大交易所上币表现分析

随着加密市场竞争的加剧&#xff0c;头部交易所逐渐在上币策略、代币选择、交易活跃度等方面采取了不同的应对策略。Animoca Digital Research近期发布的一份报告&#xff0c;通过对币安、OKX、Bitget、KuCoin和Bybit五大交易所2024年前三季度的上币情况进行了详细分析。本文将…...

Go语言sync.WaitGroup与errgroup.Group用法详解

errgroup.Group 和 sync.WaitGroup 的主要区别在于它们的错误处理和协程管理方式。 errgroup.Group 专为并发操作中的错误捕获设计&#xff0c;任意goroutine返回错误时&#xff0c;会立即终止其他goroutine的执行。 而 sync.WaitGroup 主要用于等待多个 goroutine 完成&…...

【大数据学习 | kafka】kafka的ack和一致性

1. ack级别 上文中我们提到过kafka是存在确认应答机制的&#xff0c;也就是数据在发送到kafka的时候&#xff0c;kafka会回复一个确认信息&#xff0c;这个确认信息是存在等级的。 ack0 这个等级是最低的&#xff0c;这个级别中数据sender线程复制完毕数据默认kafka已经接收到…...

学习虚幻C++开发日志——定时器

官方文档&#xff1a;虚幻引擎中的Gameplay定时器 | 虚幻引擎 5.5 文档 | Epic Developer Community | Epic Developer Community 定时器 安排在经过一定延迟或一段时间结束后要执行的操作。例如&#xff0c;您可能希望玩家在获取某个能力提升道具后变得无懈可击&#xff0c;…...

问政浔川(1)—— 有了浔川社团官方联合会和社团官方,那么浔川总社部是干什么的呢?

问政浔川&#xff08;1&#xff09;—— 有了浔川社团官方联合会和社团官方&#xff0c;那么浔川总社部是干什么的呢&#xff1f; 在浔川社团组织的复杂架构中&#xff0c;浔川社团官方联合会和社团官方已广为人知&#xff0c;但对于浔川总社部&#xff0c;很多人仍心存疑惑。这…...

区块链技术应用--电子签章(模块三)

区块链技术应用–电子签章(模块三) 背景描述 电子签章可实现与纸质文件盖章操作相似的可视效果,以保障数据来源的真实性、数据完整性以及签名人行为的不可否认性。 传统的电子签章系统是基于中心化的,也就是数据是集中存储在中心数据库中,这就导致传统电子签章使用记录…...

多面体定义+多面体是凸集+多面体的重要性质

文章目录 多面体定义多面体是凸集多面体重要性质1. 有界多面体&#xff08;Convex Polytope&#xff09;2. 无界多面体&#xff08;Unbounded Polyhedron&#xff09;3. 极点表示&#xff08;顶点形式&#xff09;与极点-极射线表示定理 在数学中&#xff0c; 多面体&#xff…...

为什么 Allow 配合 meta noindex 比使用Disallow好?

为什么 Allow 配合 meta noindex 1、Disallow 的问题 当你使用 Disallow: / 时&#xff1a; 爬虫根本不会访问你的页面 因此永远看不到你的 meta noindex 标签 如果有其他网站链接到你的页面&#xff0c;Google 可能还是会将其编入索引&#xff08;因为它无法确认你是否真的…...

通讯学徒学习日记

本章内容为 长期更新模式&#xff0c;目前入职的第三天&#xff0c;学徒状态。 文章目录 前言开始接水晶头、接光纤水晶头接光纤 PON&#xff08;GPON 、EPON&#xff09;AON 和 PON 的详解AONPON 前言 编程虽然是爱好&#xff0c;但确实也想把这份爱好变成工作。但是对于目前刚…...

迪杰斯特拉算法

迪杰斯特拉算法 LeetCode 743. 网络延迟时间 https://blog.csdn.net/xiaoxi_hahaha/article/details/110257368 import sysdef dijkstra(graph, source):"""dijkstra算法:param graph: 邻接矩阵:param source: 出发点&#xff0c;源点:return:""&…...

IPsec传输模式与隧道模式的深度解析及应用实例

随着网络安全威胁的日益严峻&#xff0c;IPsec作为网络层安全协议&#xff0c;其传输模式与隧道模式的选择对确保通信安全至关重要。本文旨在深入探讨这两种模式的差异&#xff0c;并通过实际案例展示其应用。 一、传输模式和隧道模式的详细描述 传输模式&#xff1a; 应用场景…...

实现Vue3/Nuxt3 预览excel文件

安装必要的库 npm install xlsx 创建一个组件来处理文件上传和解析&#xff1a; 在src/components 目录下创建一个名为 ExcelPreview.vue 的文件 <template> <div> <input type"file" change"handleFileUpload" /> <table v-if"…...

【AI落地应用实战】HivisionIDPhotos AI证件照制作实践指南

最近在网上发现了一款轻量级的AI证件照制作的项目&#xff0c;名为HivisionIDPhotos。它利用AI模型实现对多种拍照场景的识别、抠图与证件照生成&#xff0c;支持轻量级抠图、多种标准证件照和排版照生成、纯离线或端云推理、美颜等功能。此外&#xff0c;项目还提供了Gradio D…...

php实现sl651水文规约解析

SL651-2014-《水文监测数据通信规约》 1、要素解析说明 39 23 00 00 03 45 0x39查标识符得知为:39H Z 瞬时河道水位、潮位,我们定义为水位 0x23 按照要素标识符的规定,高5bit,低3bit,00100 011 对应的转换为10进制为4与3,也就是水位数据占用4字节,小…...