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的模拟实现》章节的全部内容了,希望能得到你的点赞和收藏

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/59228.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

本篇万字,博客最细,oled多级菜单代码解析,与实现教程,指针实现(含源码)!!!

目录 教程前言 多级菜单基本知识 驱动文件创建 ​编辑 ​编辑 ​编辑 定义菜单数据类型代码解析 按键代码解析 菜单数据赋值代码解析 菜单按键切换显示代码解析 项目工程移植地址 教程前言 前言&#xff1a;编写不易&#xf…

华为HarmonyOS打造开放、合规的广告生态 - 贴片广告

场景介绍 贴片广告是一种在视频播放前、视频播放中或视频播放结束后插入的视频或图片广告。 接口说明 接口名 描述 loadAd(adParam: AdRequestParams, adOptions: AdOptions, listener: AdLoadListener): void 请求单广告位广告&#xff0c;通过AdRequestParams、AdOptions…

Leetcode137只出现一次的数字|| 及其拓展

简述&#xff1a; 虽然标题是这么描述的&#xff0c;但是我们不是一上来就解这道题&#xff0c;先看一下他的子题和扩展 子题&#xff1a;136. 只出现一次的数字 - 力扣&#xff08;LeetCode&#xff09; 扩展题&#xff1a; 所以我们由易到难&#xff0c;先来看第一道&#x…

【解决办法】无法使用右键“通过VSCode打开文件夹”

个人博客&#xff1a;苏三有春的博客 前言 作者的编程环境为VScode&#xff0c;工作时常使用VScode打开整个工程文件夹。如果先打开VScode再从VScode中选择文件夹打开效率太慢&#xff0c;作者一般使用的方式是右键文件夹&#xff0c;直接选择"通过code打开文件夹"…

数据揭秘:掌握K-means聚类算法的精髓与实践

数据揭秘&#xff1a;掌握K-means聚类算法的精髓与实践 在机器学习领域&#xff0c;聚类是一种探索性的数据挖掘技术&#xff0c;用于将数据集中的样本划分成若干个簇&#xff0c;使得同一簇内的样本相似度高&#xff0c;而不同簇之间的样本相似度低。本文将深入探讨聚类分析的…

ADNI蛋白质数据集下载

&#xff08;我发现这个网站最近又更新了界面&#xff0c;现在变得很好看很简洁&#xff0c;但是有一些入口变了&#xff09; 1.官网链接 https://ida.loni.usc.edu/home/projectPage.jsp?projectADNI 2.登录 选择ADNI&#xff08;其实PPMI数据也是这样下的&#xff09;&a…

【数据分享】2024年我国省市县三级的生活服务设施数量(46类设施/Excel/Shp格式)

人才市场、售票处、旅行社等生活服务设施的配置情况是一个城市公共基础设施完善程度的重要体现&#xff0c;一个城市生活服务设施种类越丰富&#xff0c;数量越多&#xff0c;通常能表示这个城市的公共服务水平越高&#xff01; 本次我们为大家带来的是我国各省份、各地级市、…

彻底解决idea不识别java项目

需求背景 下载了一个java swing的项目,通过idea导入后,项目无法识别。打开java文件,也不会报错,也不编译。 无法识别效果图 可以看到左侧的菜单,项目是没有被识别。 打开java文件,可以看到没有识别,java的图标也没有出现。 解决方法 1、打开Project Structure 2、修改…

HTMLCSS:打造酷炫下载安装模拟按钮

效果演示 这段代码通过HTML和CSS创建了一个具有交互效果的下载按钮&#xff0c;当复选框被选中时&#xff0c;会触发一系列动画和样式变化&#xff0c;模拟了一个下载和安装的过程&#xff0c;包括圆形的动画、文本的显示和隐藏等。 HTML <div class"container&quo…

Multi Agents协作机制设计及实践

01 多智能体协作机制的背景概述 在前述博客中&#xff0c;我们利用LangChain、AutoGen等开发框架构建了一个数据多智能体的平台&#xff0c;并使用了LangChain的Multi-Agents框架。然而&#xff0c;在实施过程中&#xff0c;我们发现现有的框架存在一些局限性&#xff0c;这些…

ML2001-1 机器学习/深度学习 Introduction of Machine / Deep Learning

图片说明来自李宏毅老师视频的学习笔记&#xff0c;如有侵权&#xff0c;请通知下架 影片参考 【李宏毅】3.第一节 - (上) - 机器学习基本概念简介_哔哩哔哩_bilibili 1. 机器学习的概念与任务类型 概念&#xff1a;机器学习近似于寻找函数&#xff0c;用于处理不同类型的任…

90%会展主办方都会用的6款数字化工具

在会展行业&#xff0c;数字化转型已成为提升竞争力的关键。面对日益增长的运营成本和收入增长的瓶颈&#xff0c;主办方需要借助数字化工具来实现效率提升和成本控制。 今天介绍几种常见的数字化工具和应用方式。 一、线上展览平台 构建线上展览平台是会展主办方拓展线上销…

JMeter快速造数之数据导入导出

导入数据 输入表格格式如下 创建CSV Data Set Config 在Body Data中调用 { "username": "${email}", "password": "123456", "client_id": "00bb9dbfc67439a5d42e0e19f448c7de310df4c7fcde6feb5bd95c6fac5a5afc"…

渗透测试-快速获取目标中存在的漏洞(小白版)

《Java代码审计》http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247484219&idx1&sn73564e316a4c9794019f15dd6b3ba9f6&chksmc0e47a67f793f371e9f6a4fbc06e7929cb1480b7320fae34c32563307df3a28aca49d1a4addd&scene21#wechat_redirect 《Web安全》h…

[免费]基于Python的Django+Vue3在线考试系统【论文+源码+SQL脚本】

大家好&#xff0c;我是java1234_小锋老师&#xff0c;看到一个不错的基于Python的DjangoVue3在线考试系统&#xff0c;分享下哈。 项目视频演示 【免费】基于Python的DjangoVue3在线考试系统 Python毕业设计_哔哩哔哩_bilibili 项目介绍 本论文提出并实现了一种基于Python…

Unity3D学习FPS游戏(9)武器音效添加、创建敌人模型和血条

前言&#xff1a;虽然已经实现了基本玩家操作&#xff0c;但是游戏运行起来并没有音效。既然是FPS游戏有了玩家和武器&#xff0c;肯定还得有敌人。本篇演示如何给武器添加音效和创建敌人。 武器音效添加和创建敌人 武器音效添加Audio Source代码控制 创建敌人目标敌人模型敌人…

雷池社区版新版本功能防绕过人机验证解析

前两天&#xff0c;2024.10.31&#xff0c;雷池社区版更新7.1版本&#xff0c;其中有一个功能&#xff0c;新增请求防重放 更新记录&#xff1a;hhttps://docs.waf-ce.chaitin.cn/zh/%E7%89%88%E6%9C%AC%E6%9B%B4%E6%96%B0%E8%AE%B0%E5%BD%95 仔细研究了这个需求&#xff0c;…

PyCharm中pylint安装与使用

目录 1. 安装插件2. pycharm中使用该功能3. 命令行使用 1. 安装插件 然后重启 2. pycharm中使用该功能 3. 命令行使用 前提是先 pip install pylint pylint demo01.py下面红框内容的意思是&#xff0c;得到10分/ 满分10分&#xff0c;上次运行获得8.33分&#xff0c;经调整…

IDEA加载通义灵码插件及使用指南

安装通义灵码插件 登录通义灵码IDE插件 下载登录参考教程 https://help.aliyun.com/zh/lingma/user-guide/download-the-installation-guide 本地工程和企业知识库准备 请下载本地工程和知识库压缩包&#xff0c;并在本地解压缩&#xff0c;其中包含demoProject和知识库文件…

只因把 https 改成 http,带宽减少了 70%!

起因 是一个高并发的采集服务上线后&#xff0c;100m的上行很快就被打满了。因为这是一条专线&#xff0c;并且只有这一个服务在使用&#xff0c;所以可以确定就是它导致的。 但是&#xff01;这个请求只是一个 GET 请求&#xff0c;同时并没有很大的请求体&#xff0c;这是为…