【STL】模拟实现简易 list

目录

1. 读源码

2. 框架搭建 

3. list 的迭代器

4. list 的拷贝构造与赋值重载

拷贝构造

赋值重载

5. list 的常见重要接口实现

operator--() 

insert 接口

erase 接口

push_back 接口

push_front 接口

pop_back 接口

pop_front 接口

size 接口

clear 接口

别忘了析构函数

源码分享

写在最后:


1. 读源码

读源码千万不能一行一行读啊,不然你就看晕在那里了,

我们先从核心框架开始抓取,比如说先找到 list 在哪:

 然后老规矩,我们先找他的成员变量:

那我们就来找找这个 link_type 是什么:

link_type 是 list_node*,list_node 是一个类类型,那我就知道了,

成员变量 node 就是链表的一个节点指针。

那问题又来了,有单链表,双链表,带头的链表等等,库里实现的是什么链表呢?

我们需要确定他的结构,还是老样子,我们先从构造函数和插入(核心)接口开始看:

我们先看看这个无参的构造是怎么实现的:

他先 get_node() 获取一个节点,然后再两个指针指向自己,

那我们基本就能确定这是一个带头双向循环的链表了。 

那我们奖励自己再看一眼他的 get_node() 吧

我们就可以看到他是通过空间配置器的接口开空间了,

再往下看其实就是定位 new 的那一套操作了。

我们继续接着来看 push_back() 接口是怎么样的:

 

我们可以看到,他这里就是复用的 insert,在 end() 位置插入,

 

他这里调用的就是这个 insert 的重载,就是普通的插入操作。

最后我们再来瞅一眼 node 这个节点类库里是怎么定义的:

他这里用了 void* 作为他的类型,我比较菜,不太懂这样做有什么妙用,

我就不这么麻烦去用 void* 作为我的指针类型了,不然之后每次用都得强转,我用 T* 就好了。

那么源码看到这里就差不多了,框架看的差不多了,到时候有问题再来看细节。

2. 框架搭建 

框架搭建主要就是把 list 的核心框架搭建出来,让代码快速跑起来:

#pragma once

#include <iostream>
#include <list>

#include <assert.h>

using namespace std;

namespace xl {
	template<class T>
	struct list_node {
		list_node<T>* _next;
		list_node<T>* _prev;
		T _val;

		list_node(const T& val = T())
			: _next(nullptr)
			, _prev(nullptr)
			, _val(val)
		{}
	};

	template<class T>
	class list {
	public:
		typedef list_node<T> Node;

	private:
		Node* _head;

	public:
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
		}

	public:
		void push_back(const T& x) {
			Node* tail = _head->_prev;
			Node* newnode = new Node;
			
			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

	};
}

这里我们实现了 list 的节点,以及 list 的构造函数和尾插接口,

来看看测试:

#include "list.h"

void test1() {
	xl::list<int> lt1;
	lt1.push_back(1);
	lt1.push_back(2);
	lt1.push_back(3);
	lt1.push_back(4);

}

int main()
{
	test1();

	return 0;
}

通过调试来看结果:

我们确实是插入了 4 个节点,

你有没有觉得少了点什么,之前我们搭框架的时候都会实现一个基本的迭代器,

但是这次没有,问题来了, list 的迭代器很明显是不能用原生指针实现的,

毕竟你有见过链表能用指针或者说下标直接访问吗,那肯定是没见过,

那 list 的迭代器该怎么实现呢?

3. list 的迭代器

当我们不明白一件事情的时候,就去看源码是怎么做的:

找到了,但是更迷惑了,怎会有三个模板参数啊,

先就此打住,我们一点一点慢慢看,他的类型是一个类模板,那我们先去找到这个类:

我们看到这里,发现他是用一个叫 __list_iterator 的类来封装他的迭代器, 

而这个类的成员变量就是链表的节点:

那我们再来看看他的迭代器是怎么跑起来的(也就是++是怎么实现的)

我们发现这不就是让 node = node->next 的操作吗。

再来看看解引用的操作:

不出所料确实就是返回该节点的值,

但是他这个返回值的类型 reference 是啥东东呢?

这个是他的一个模板参数,看来看去,这源码很复杂,又有许多意义不明的操作,

我们还是先根据大思路上手试一下,遇到问题了再来细看源码的实现。

在搭完基本的架子之后,我们遇到了第一个问题,

begin 和 end 该指向什么位置?我们来看看库:

库里的 begin 返回的是第一个节点,end 返回的是哨兵位的头结点,

所以我们就这样实现即可:

iterator begin() {
	return _head->_next;
}

iterator end() {
	return _head;
}

这个时候你可能又有疑问了,迭代器不是自定义类型吗?他怎么能返回节点的指针呢?

这就又用到我们前面学的知识了:单参数的构造函数支持隐式类型转换:

__list_iterator(Node* node)
	: _node(node)
{}

是的,我们在迭代器的类里实现了这样一个东西。

然后我们再把解引用实现了:

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

最后还剩 ++ 和 != 需要实现:

iterator operator++() {
	_node = _node->_next;
	return *this;
}

bool operator!=(iterator& it) {
	return _node != it._node;
}

这样我们的迭代器就跑通了,来看看测试:

void test2() {
	xl::list<int> lt;
	lt.push_back(1);
	lt.push_back(2);
	lt.push_back(3);
	lt.push_back(4);

	xl::list<int>::iterator it = lt.begin();
	while (it != lt.end()) {
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

输出:

 这样我们的基本框架算是搭建完了:

#pragma once

#include <iostream>
#include <list>

#include <assert.h>

using namespace std;

namespace xl {
	template<class T>
	struct list_node {
		list_node<T>* _next;
		list_node<T>* _prev;
		T _val;

		list_node(const T& val = T())
			: _next(nullptr)
			, _prev(nullptr)
			, _val(val)
		{}
	};

	template<class T>
	struct __list_iterator {
		typedef __list_iterator<T> iterator;
		
		typedef list_node<T> Node;

		Node* _node;

		__list_iterator(Node* node) 
			: _node(node)
		{}

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

		iterator& operator++() {
			_node = _node->_next;
			return *this;
		}

		bool operator!=(const iterator& it) {
			return _node != it._node;
		}
	};

	template<class T>
	class list {
	public:
		typedef list_node<T> Node;

		typedef __list_iterator<T> iterator;

		iterator begin() {
			return _head->_next;
		}

		iterator end() {
			return _head;
		}

	private:
		Node* _head;

	public:
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
		}

	public:
		void push_back(const T& x) {
			Node* tail = _head->_prev;
			Node* newnode = new Node(x);
			
			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

	};
}

所以这里我们可以得出一个小结论,

list 的迭代器是什么?他是通过对自定义类型的封装,改变了他的行为。

那我们继续,现在来设计实现一个 const 迭代器,

我们可以通过加 const 来完成这件事情:

如果我们想要重载一整份迭代器,那岂不是得重新写一份自定义的 const 迭代器?

那这样设计也太冗余了,凭空又多出一大坨代码,有没有什么更好的方法实现呢?

还记得我们一开始看库的时候,那两个意义不明的模板参数吗?

他们还是同一个类,但是传了不同的模板参数。

然后就增加了迭代器的模板参数:

然后他这里就把迭代器重命名成了 self,我们就跟着库里的来:

首先是传模板参数这里,因为我们暂时只需要传 T* 给解引用的重载,

所以暂时先设置这两个模板参数:

实际上,这种做法和我们一开始否决的冗余写法没有本质上的区别,

因为模板的实例化就是再生成一段代码,只不过这个工作原本是由我们做,

使用模板之后变成让编译器帮我做了。

这里我把这个阶段的代码也放出来:

#pragma once

#include <iostream>
#include <list>

#include <assert.h>

using namespace std;

namespace xl {
	template<class T>
	struct list_node {
		list_node<T>* _next;
		list_node<T>* _prev;
		T _val;

		list_node(const T& val = T())
			: _next(nullptr)
			, _prev(nullptr)
			, _val(val)
		{}
	};

	template<class T, class Ref>
	struct __list_iterator {
		typedef __list_iterator<T, Ref> self;
		
		typedef list_node<T> Node;

		Node* _node;

		__list_iterator(Node* node) 
			: _node(node)
		{}

		Ref operator*() {
			return _node->_val;
		}

		self& operator++() {
			_node = _node->_next;
			return *this;
		}

		self operator++(int) {
			self tmp(*this);
			_node = _node->_next;
			return tmp;
		}

		bool operator!=(const self& it) {
			return _node != it._node;
		}

		bool operator==(const self& it) {
			return _node != it._node;
		}
	};

	template<class T>
	class list {
	public:
		typedef list_node<T> Node;

		typedef __list_iterator<T, T&> iterator;
		typedef __list_iterator<T, const T&> const_iterator;

		iterator begin() {
			return _head->_next;
		}

		iterator end() {
			return _head;
		}

	private:
		Node* _head;

	public:
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
		}

	public:
		void push_back(const T& x) {
			Node* tail = _head->_prev;
			Node* newnode = new Node(x);
			
			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

	};
}

这里新的问题又来了,为什么库里是有三个模板参数呢?

我们来看看:

  

我们可以看到,还需要这个模板参数的是 -> 操作符的重载,

那事不宜迟,我们也来实现一下:

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

现在我们是正常的实现了这个操作符重载,

那这个操作符有什么应用场景吗?我们为什么要重载他?

来看这样一个场景:

struct A {
	A(int a1 = 0, int a2 = 0) 
		: _a1(a1)
		, _a2(a2)
	{}

	int _a1;
	int _a2;
};

void test3() {
	xl::list<A> lt;
	lt.push_back({ 1, 1 });
	lt.push_back({ 2, 2 });
	lt.push_back({ 3, 3 });
	lt.push_back({ 4, 4 });

	xl::list<A>::iterator it = lt.begin();
	while (it != lt.end()) {
		cout << (*it)._a1 << (*it)._a2 << endl;
		it++;
	}
}

如果我们想取结构体内的成员,可以通过 (*it). 来取,

但是我们一般更喜欢使用 -> 直接取结构体成员:

struct A {
	A(int a1 = 0, int a2 = 0) 
		: _a1(a1)
		, _a2(a2)
	{}

	int _a1;
	int _a2;
};

void test3() {
	xl::list<A> lt;
	lt.push_back({ 1, 1 });
	lt.push_back({ 2, 2 });
	lt.push_back({ 3, 3 });
	lt.push_back({ 4, 4 });

	xl::list<A>::iterator it = lt.begin();
	while (it != lt.end()) {
		cout << it->_a1 << it->_a2 << endl;
		it++;
	}
}

这个就是重载 -> 的意义。

但是,你有没有发现有一些不太对劲的地方?

这个函数返回的只是一个指针,而调用这个操作符重载需要一个 -> ,

然后,使用这个指针去调用结构体成员还需要一个 -> ,那为什么这里只有一个 -> 呢?

实际上是为了代码的可读性,编译器特殊处理让我们可以省略一个 -> 。

明白了这个之后,我们就再来添加一个模板参数给他用。

那这样我们的大框架总算是搭好了:

#pragma once

#include <iostream>
#include <list>

#include <assert.h>

using namespace std;

namespace xl {
	template<class T>
	struct list_node {
		list_node<T>* _next;
		list_node<T>* _prev;
		T _val;

		list_node(const T& val = T())
			: _next(nullptr)
			, _prev(nullptr)
			, _val(val)
		{}
	};

	template<class T, class Ref, class Ptr>
	struct __list_iterator {
		typedef __list_iterator<T, Ref, Ptr> self;
		
		typedef list_node<T> Node;

		Node* _node;

		__list_iterator(Node* node) 
			: _node(node)
		{}

		Ref operator*() {
			return _node->_val;
		}

		Ptr operator->() {
			return &_node->_val;
		}

		self& operator++() {
			_node = _node->_next;
			return *this;
		}

		self operator++(int) {
			self tmp(*this);
			_node = _node->_next;
			return tmp;
		}

		bool operator!=(const self& it) {
			return _node != it._node;
		}

		bool operator==(const self& it) {
			return _node != it._node;
		}
	};

	template<class T>
	class list {
	public:
		typedef list_node<T> Node;

		typedef __list_iterator<T, T&, T*> iterator;
		typedef __list_iterator<T, const T&, const 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;
		}

	private:
		Node* _head;

	public:
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
		}

	public:
		void push_back(const T& x) {
			Node* tail = _head->_prev;
			Node* newnode = new Node(x);
			
			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

	};
}

4. list 的拷贝构造与赋值重载

拷贝构造

直接通过复用 push_back 来完成拷贝构造。

list(const list<T>& lt)
{
	_head = new Node;
	_head->_next = _head;
	_head->_prev = _head;

	for (auto& e : lt) {
		push_back(e);
	}
}

赋值重载

我们就直接用现代写法实现,还能顺便把 swap 函数提供了:

void swap(list<T>& lt) { ::swap(_head, lt._head); }

list<T>& opeartor = (list<T> lt)
{
	swap(lt);
	return *this;
}

我们可以来集中测试一下:

void test4() {
	xl::list<int> lt;
	lt.push_back(1);
	lt.push_back(2);
	lt.push_back(3);
	lt.push_back(4);
	
	xl::list<int> lt3;
	xl::list<int> lt2(lt);
	lt3 = lt2;

	xl::list<int>::iterator it = lt3.begin();
	while (it != lt3.end()) {
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

输出:

5. list 的常见重要接口实现

operator--() 

self& operator--() {
	_node = _node->_prev;
	return *this;
}

self operator--(int) {
	self tmp(*this);
	_node = _node->_prev;
	return tmp;
}

insert 接口

我们先实现 insert 和 erase 接口,之后直接复用就好了:

// pos 位置之前插入
iterator insert(iterator pos, const T& x) {
	Node* cur = pos._node;
	Node* prev = cur->_prev;
	Node* newnode = new Node(x);

	prev->_next = newnode;
	newnode->_next = cur;

	cur->_prev = newnode;
	newnode->_prev = cur;

	return newnode;
}

erase 接口

iterator erase(iterator pos) {
	assert(pos != end);

	Node* cur = pos._node;
	Node* prev = cur->_prev;
	Node* next = cur->_next;

	prev->_next = next;
	next->_prev = prev;

	delete cur;

	return next;
}

剩下的通通复用~        

push_back 接口

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

push_front 接口

void push_front(const T& x) {
	insert(begin(), x);
}

pop_back 接口

void pop_back() {
	erase(--end());
}

pop_front 接口

void pop_front() {
	erase(begin());
}

不知道你爽了没,反正我爽了,现在就再来一些常见接口:

size 接口

我们直接用迭代器来计数:

size_t size() {
	size_t sz = 0;
	iterator it = begin();
	while (it != end()) {
		sz++;
		it++;
	}
	return sz;
}

clear 接口

清理所有的数据,直接复用 erase:

void clear() {
	iterator it = begin();
	while (it != end()) {
		it = erase(it);
	}
}

对了,差点忘了析构函数还没定义:

别忘了析构函数

直接复用 clear:

~list()
{
	clear();
	delete _head;
	_head = nullptr;
}

源码分享

Gitee链接:模拟实现简易STL: 模拟实现简易STL (gitee.com)

写在最后:

以上就是本篇文章的内容了,感谢你的阅读。

如果感到有所收获的话可以给博主点一个哦。

如果文章内容有遗漏或者错误的地方欢迎私信博主或者在评论区指出~

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Window环境RabbitMq搭建部署

Erlang下载安装及配置环境变量 下载erlang&#xff0c;原因在于RabbitMQ服务端代码是使用并发式语言Erlang编写的 Erlang下载 Erlang官网下载&#xff1a; http://www.erlang.org/downloads Erlang国内镜像下载&#xff08;推荐&#xff09;&#xff1a; http://erlang.org/d…

旧版Xcode文件较大导致下载总是失败但又不能断点续传重新开始的解决方法

问题&#xff1a; 旧版mac下载旧版Xcode时需要进入https://developer.apple.com/download/all/?qxcode下载&#xff0c;但是下载这些文件需要登录。登录后下载中途很容易失败&#xff0c;失败后又必须重新下载。 解决方案&#xff1a; 下载这里面的内容都需要登录&#xff0…

Appium+python自动化(十九)- Monkey(猴子)参数(超详解)

前边几篇介绍了Monkey以及Monkey的事件&#xff0c;今天就给小伙伴们介绍和分享一下Monkey的参数。 首先我们看一下这幅图来大致了解一下&#xff1a; 1、Monkey 命令 基本参数介绍 -p <允许的包名列表> 用此参数指定一个或多个包。指定包之后&#xff0c;mon…

用html+javascript打造公文一键排版系统7:落款排版

一、公文落款的格式 公文落款包括单位署名和成文日期两个部分&#xff0c;其中成文日期中的数字 用阿拉伯数字将年、月、日标全&#xff0c;年份应标全称&#xff0c;月、日不编虚位&#xff08;即 1 不编为 01&#xff09;。 在实际应用工作中分为三种情况&#xff1a; &am…

【Selenium+Pytest+allure报告生成自动化测试框架】附带项目源码和项目部署文档

目录 前言 【文章末尾给大家留下了大量的福利】 测试框架简介 首先管理时间 添加配置文件 conf.py config.ini 读取配置文件 记录操作日志 简单理解POM模型 简单学习元素定位 管理页面元素 封装Selenium基类 创建页面对象 简单了解Pytest pytest.ini 编写测试…

保护数字世界的壁垒

随着科技的不断发展和互联网的普及&#xff0c;我们的生活日益依赖于数字化的世界。然而&#xff0c;随之而来的是网络安全威胁的不断增加。网络攻击、数据泄露和身份盗窃等问题已经成为我们所面临的现实。因此&#xff0c;网络安全变得尤为重要&#xff0c;我们需要采取措施来…

什么是分布式操作系统?我们为什么需要分布式操作系统?

分布式操作系统是一种特殊的操作系统&#xff0c;本质上属于多机操作系统&#xff0c;是传统单机操作系统的发展和延伸。它是将一个计算机系统划分为多个独立的计算单元(或者也可称为节点)&#xff0c;这些节点被部署到每台计算机上&#xff0c;然后被网络连接起来&#xff0c;…

Win10环境下Android Studio中运行Flutter HelloWorld项目

一、引言 Android Studio是Android的官方IDE(Integrated Development Environment)。它专为Android而打造&#xff0c;可以加快开发速度&#xff0c;为Android设备构建最高品质的应用。 Flutter是Google推出并开源的移动应用开发框架&#xff0c;主打跨平台、高保真、高性能。开…

【STL】list用法试做_底层实现

目录 一&#xff0c;list 使用 1. list 文档介绍 2. 常见接口 1. list中的sort 2. list sort 与 vector sort效率对比 3. 关于迭代器失效 4. clear 二&#xff0c;list 实现 1.框架搭建 2. 迭代器类——核心框架 3. operator-> 实现 4. const——迭代…

【计算机网络 01】说在前面 信息服务 因特网 ISP RFC技术文档 边缘与核心 交换方式 定义与分类 网络性能指标 计算机网络体系结构 章节小结

第一章--概述 说在前面1.1 计算机网络 信息时代作用1.2 因特网概述1.3 三种交换方式1.4 计算机网络 定义与分类1.5 计算机网络的性能指标1.6 计算机网络体系结构1 常见的计算机网络体系结构2 计算机网络体系结构分层的必要性3 计算机网络体系结构分层思想举例4 计算机网络体系结…

RuntimeError: DataLoader worker (pid 2105929) is killed by signal: Killed.

PyTorch DataLoader num_workers Test - 加快速度 可以利用PyTorch DataLoader类的多进程功能来加快神经网络训练过程。 加快训练进程 为了加快训练过程&#xff0c;我们将利用DataLoader类的num_workers可选属性。 num_workers属性告诉DataLoader实例要使用多少个子进程进…

23.多项式与非多项式曲线拟合对比(matlab程序)

1.简述 拟合标准&#xff1a; (1)原始数据向量与拟合向量之间的距离最小&#xff0c;该距离的度量一般使用误差平方和表示&#xff0c;即均方误差&#xff1a;R||Q-Y||22 (2)当均方误差最小时&#xff0c;说明构造的拟合向量与原始向量最为接近&#xff0c;这种曲线拟合的方法…

git commit -m时候没有保存package.json等文件

项目场景&#xff1a; 提示&#xff1a;git add . 和 git commit -m "保存" 操作&#xff0c;没有保存package.json等文件。 解决方案&#xff1a; 1.确保 package.json 文件没有被列在 .gitignore 文件中。打开 .gitignore 文件&#xff0c;检查是否有类似于 packa…

论文工具——ChatGPT结合PlotNeuralNet快速出神经网络深度学习模型图

文章目录 引言正文PlotNeuralNet安装使用使用python进行编辑使用latex进行编辑 样例利用chatGPT使用chatGPT生成Latex代码利用chatGPT生成对应的python代码 总结引用 引言 介绍如何安装PlotNeuralNet工具&#xff0c;并结合chatGPT减少学习成本&#xff0c;快速出图。将按照软…

4.2 Bootstrap HTML编码规范

文章目录 Bootstrap HTML编码规范语法HTML5 doctype语言属性IE 兼容模式字符编码引入 CSS 和 JavaScript 文件HTML5 spec links 实用为王属性顺序布尔&#xff08;boolean&#xff09;型属性减少标签的数量JavaScript 生成的标签 Bootstrap HTML编码规范 语法 用两个空格来代替…

通过 EXPLAIN 分析 SQL 的执行计划

通过 EXPLAIN 分析 SQL 的执行计划 EXPLAIN SELECTleave_station_area_id,ROUND( ( SUM( station_dist ) / 1000 ) / ( SUM( station_travel_time ) / 60 ), 2 ) evnPeakAvgSpeedFROMV3_SHIFT_ANALYSISWHERESTAT_DATE DATE_SUB( CURRENT_DATE, INTERVAL 1 DAY )AND LEAVE_STA…

NetSuite财务报表General Ledger Report的缺陷及改造案例

本周有用户提到一个特殊的业务场景&#xff0c;比较有代表性&#xff0c;在此分享。 问题 “如果在一张JE中&#xff0c;某个科目既有借又有贷&#xff0c;金额相同。那么在General Ledger Report中此JE的借贷都显示为0。这与事实不符&#xff0c;所以是不对的。” JE 155&a…

vue3-element-plus,控制表格多选的数量

1. 需求描述 控制表格的多选&#xff0c;最多只能选择5条数据&#xff0c;并且其他项禁用 2. 需求描述 <!-- selection-change 当选择项发生变化时会触发该事件--><template><el-tableref"multipleTableRef"v-loading"loading":data"…

[Linux] CentOS7 中 pip3 install 可能出现的 ssl 问题

由于解决问题之后, 才写的博客, 所以没有图片记录. 尽量描述清楚一些 今天写代码的时候, 突然发现 文件里用了#define定义宏之后, coc.nvim的coc-clangd补全就用不了 :checkhealth了一下, 发现nvim忘记支持python3了 尝试pip3 install neovim的时候, 发现会警告然后安装失败.…

网络安全(黑客)自学路线笔记

一、什么是黑客&#xff1f; 黑客泛指IT技术主攻渗透窃取攻击技术的电脑高手&#xff0c;现阶段黑客所需要掌握的远远不止这些。 二、为什么要学习黑客技术&#xff1f; 其实&#xff0c;网络信息空间安全已经成为海陆空之外的第四大战场&#xff0c;除了国与国之间的博弈&am…