【STL】模拟实现vector
目录
1、基本成员变量
2、默认成员函数
构造函数
析构函数
拷贝构造函数
赋值运算符重载函数
3、容器访问相关函数接口
operator [ ]运算符重载
迭代器
范围for
4、vector容量和大小相关函数
size和capacity
reserve扩容
resize
swap交换数据
empty
5、修改容器内容相关函数
push_back尾插
insert
pop_back尾删
erase
clear清空数据
1、基本成员变量
namespace Fan {template <class T>class vector{public:typedef T* iterator;typedef const T* const_iterator;private:iterator _start; //指向容器的头iterator _finish; //指向有效数据的尾iterator _endofstoage; //指向容器的尾}; }
2、默认成员函数
构造函数
- 1、无参构造函数
我们只需要把每个成员变量初始化为nullptr即可。
//无参构造函数 vector():_start(nullptr),_finish(nullptr),_endofstoage(nullptr) {}
- 2、带参构造函数
vector的带参构造函数首先在初始化列表对基本成员变量进行初始化,然后将迭代器区间在[first, last)的数据一个个尾插到容器当中即可。
//带参构造函数 template <class InputIterator> vector(InputIterator first, InputIterator last):_start(nullptr), _finish(nullptr), _endofstoage(nullptr) {//将迭代器区间在[first,last)的数据一个个尾插到容器当中while (first != last){push_back(*first);first++;} }
- 3、用n个val去初始化vector
vector的构造函数还支持用n个val去进行初始化,只需要先调用reserve函数开辟n个大小的空间,然后用for循环把val的值依次push_back尾插进去即可。
//用n个val来构造vector vector(size_t n, const T& val = T()): _start(nullptr), _finish(nullptr), _endofstoage(nullptr) {reserve(n);for (size_t i = 0; i < n; i++){push_back(val);} }⚠:注意
此时会出现一个问题:内存寻址错误。当我们实现下面的语句时:
Fan::vector<int> v(5, 4);对于构造函数3而言其两个参数类型为size_t和相对应模板数据类型。而对于构造函数2,其两个参数类型都为迭代器。我们上述代码调用的地方两个参数都是int,此时调用构造函数时匹配的是第二个传迭代器区间的构造函数,导致这样的原因在于编译器会优先寻找最匹配的那个函数。为了解决这一问题,我们还需要重载一个第一个参数类型为int类型的构造函数。
vector(int n, const T& val = T()): _start(nullptr), _finish(nullptr), _endofstoage(nullptr) { reserve(n);for (int i = 0; i < n; i++){push_back(val);} }
析构函数
首先我们判断该容器_start是否为空,不为空就释放空间+置空即可。
//析构函数 ~vector() {if (_start) //避免释放空指针{delete[] _start; //释放容量所指向的空间_start = _finish = _endofstoage = nullptr; //置空} }
拷贝构造函数
- 传统写法
拷贝构造传统写法的思想是我们最容易想到的:先开辟一块与该容器大小相同的空间,然后将容器当中的数据一个个拷贝过来即可,最后更新_finish和_endofstorage即可。
//传统写法 vector(const vector<T>& v):_start(nullptr),_finish(nullptr),_endofstoage(nullptr) {_start = new T[v.capacity()]; //开辟一块和容器V大小相同的空间for (size_t i = 0; i < v.size(); i++) //将容器v中的数据一个个拷贝过来{_start[i] = v[i];}_finish = _start + v.size(); _endofstoage = _start + v.capacity(); }注意:将容器中的数据一个个拷贝过来时不能够用memcpy函数,当vector存储的数据是内置类型或者是无需深拷贝的自定义类型时,使用memcpy函数并没有什么问题,但是当vector存储的数据是需要进行深拷贝的自定义类型,那么使用memcpy就会存在弊端。例如,当vector存储的数据是string类的时候。
vector当中存储的每一个string都指向自己所存储的字符串。
如果此时我们使用的是memcpy函数进行拷贝构造的话,那么拷贝构造出来的vector当中存储的每个string的成员变量值将与被拷贝的vector当中存储的每个string的成员变量值相同,即两个vector当中的每个对应的string成员都指向同一个字符串空间。
这显然并不是我们想要的结果,那么我们所给的代码是如何解决了这个问题呢?
总结: 如果vector当中存储的数据类型是内置类型(int)或深拷贝的自定义类型(Date),使用memcpy函数进行拷贝构造是没有问题的,但如果vector当中存储的数据类型是深拷贝的自定义类型(string),则使用memcpy函数将不能达到我们想要的结果。
- 现代写法
拷贝构造我们可以仿照string的现代方法拷贝构造思路。首先对基本成员变量进行初始化,然后创建一个tmp的模板将要拷贝的数据利用构造函数传递过去,然后再将这个tmp模板与自己交换即可。
//拷贝构造函数 vector(const vector<T>& v):_start(nullptr),_finish(nullptr),_endofstoage(nullptr) {vector<T> tmp(v.begin(), v.end()); //调用构造函数swap(tmp); }
赋值运算符重载函数
- 传统写法
首先判断是否是给自己赋值,若是给自己赋值则无需进行操作。若不是给自己赋值,则先开辟一块和容器V大小相同的空间,然后将容器V当中的数据一个个拷贝过来,最后更新_finish和_endofstorage的值即可。
//传统写法 vector<T>& operator=(const vector<T>& v) {if (this != &v) //防止自己给自己赋值{delete[] _start; //释放原来的空间_start = new T[v.capacity()]; //开辟一块和容器v大小相同的空间for (size_t i = 0; i < v.size(); i++) //将容器v当中的数据一个个拷贝过来{_start[i] = v[i];}_finish = _start + v.size(); //容器有效数据的尾_endofstorage = _start + v.capacity(); //整个容器的尾}return *this; //支持连续赋值 }注意:和拷贝构造的传统写法类似,这里也不能使用memcpy函数进行拷贝。
- 现代写法
我们这里直接用传值传参,不用引用传参。利用vector调用构造函数返回的值与左值swap交换。
//赋值运算符重载 vector<T>& operator=(vector<T> v) //调用构造 {this->swap(v); //交换两个对象return *this; }
3、容器访问相关函数接口
operator [ ]运算符重载
直接返回pos位置的数据即可进行下标+[ ]的方式进行访问
//operator[]运算符重载 T& operator[](size_t pos) {assert(pos < size()); //检测pos的合法性return _start[pos]; }为了方便const对象也可以调用[ ]运算符重载,我们还需要写一个const版本的[ ]运算符重载。
//const版本的[]运算符重载 const T& operator[](size_t pos)const {assert(pos < size()); //检测pos的合法性return _start[pos]; }
迭代器
vector的begin直接返回容器的_start,end返回容器的_finish。
//begin iterator begin() {return _start; //返回容器的起始位置 } //end iterator end() {return _finish; //返回有效数据下一个的地址 }const版本:
//const版本迭代器 const_iterator begin()const {return _start; } //end const_iterator end()const {return _finish; }
范围for
现在我们实现了迭代器,实际上也就可以使用范围for遍历容器,因为编译器在编译时会自动将范围for替换成迭代器的形式。
vector<int> v(5, 3); //范围for进行遍历 for (auto e : v) {cout << e << " "; } cout << endl;
4、vector容量和大小相关函数
size和capacity
因为指针相减的结果就是这两个指针之间对应类型的数据个数,所以获取size只需_finish-_start。获取capacity只需_endofstoage-_start。
- size函数:
size_t size() const //最好加上const,普通对象和const对象均可调用 {return _finish - _start; //指针相减就能得到size的个数 }
- capacity函数:
size_t capacity() const {return _endofstoage - _start; }
reserve扩容
reserve扩容思路在模拟实现string讲过,这里不再多赘述。
//reserve扩容 void reserve(size_t n) {size_t sz = size();//提前算出size()的大小,方便后续更新_finishif (n > capacity()){T* tmp = new T[n];if (_start)//判断旧空间是否有数据{//不能用memcpy,因为memcpy是浅拷贝for (size_t i = 0; i < size(); i++){tmp[i] = _start[i];//将容器当中的数据一个个拷贝到tmp当中}delete[] _start;//释放旧空间}_start = tmp;//指向新空间}//更新_finish和_endofstoage_finish = _start + sz;_endofstoage = _start + n; }这里有两个需要注意的地方:
- 在进行操作之前需要提前记录当前容器当中有效数据的个数。
因为我们最后需要更新_finish指针的指向,而_finish指针的指向就等于_start指针加上容器当中有效数据的个数,当_start指针的指向改变后我们再调用size函数通过_finish-_start计算出的有效数据的个数就是一个随机值了。此时就会出bug。
- 拷贝容器当中的数据时,不能够使用memcpy函数进行拷贝。
memcpy是浅拷贝,当我们vector当中存储的是string类的时候,使用memcpy函数reserve出来的容器与原容器当中每个对应的string成员都指向同一个字符串空间。当我们释放原容器空间的时候,原容器当中存储的每个string在释放时会调用string的析构函数,将其指向的字符串也进行释放。所以使用memcpy函数reserve出来的容器当中的每一个string所指向的字符串实际上是一块已经被释放的空间,访问该容器时就是对内存空间进行非法访问。
所以我们还是需要用for循环将容器当中的string一个个赋值过来,因为这样能够间接调用string的赋值运算符重载,实现string的深拷贝。
resize
- 如果 n 小于当前容器的size(),则内容将减少到其前 n 个元素,删除超出(并销毁)的元素。
- 如果 n 大于当前容器 size(),则通过在末尾插入所需数量的元素以达到 n 的大小来扩展内容。若指定了 val,则新元素将初始化为 val 的副本,否则,它们将进行值初始化。
- 如果 n 也大于当前容器容量capacity(),则会自动重新分配分配的存储空间。
//resize //void resize(size_t n, T val = T()) void resize(size_t n, const T& val = T()) //利用T()调用默认构造函数的值进行初始化,这样写说明C++的内置类型也有自己的构造函数 {//如果 n > capacity()容量,就需要扩容if (n > capacity()){reserve(n);}//如果 n > size(),就需要把有效数据_finish到_start + n之间的数据置为缺省值valif (n > size()){while (_finish < _start + n){*_finish = val;_finish++;}}//如果 n < size(),更新有效数据到_start + nelse{ _finish = _start + n;} }
- 补充:在C++当中内置类型也可以看作是一个类,它们也有自己的默认构造函数,数据类型默认值为0,指针为空。这样也能更好的支持模板,所以我们在给resize函数的参数val设置缺省值时,设置为T()即可。
void test() {int i = 0;int j = int();int k = int(1);cout << i << endl;//0cout << j << endl;//0cout << k << endl;//1 }
swap交换数据
我们直接调用库函数里面的swap去进行成员变量的交换即可。
//交换函数 void swap(vector<T>& v) {std::swap(_start, v._start);std::swap(_finish, v._finish);std::swap(_endofstoage, v._endofstoage); }
empty
empty函数可以直接通过比较容器当中的_start和_finish指针的指向来判断容器是否为空,若指向的位置相同,则该容器为空。
bool empty()const {return _start == _finish; }
5、修改容器内容相关函数
push_back尾插
要尾插首先要判断是否需要扩容,把尾插的值赋过去,再更新有效数据地址_finish即可。
void push_back(const T& x) {//检测是否需要扩容if (_finish == _endofstoage){size_t newcapcacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapcacity);}*_finish = x;_finish++; }这里的push_back还可以复用下文实现好的insert进行尾插,当insert中的pos为_finish时,insert实现的就是push_back尾插。而_finish可以通过调用迭代器end函数来解决。
void push_back(const T& x) {//法二:复用insertinsert(end(), x); //当insert中的参数pos为end()时,就是尾插 }
insert
insert函数可以在所给迭代器pos位置插入数据,在插入数据前先判断是否需要增容,然后将pos位置及其之后的数据统一向后挪动一位,以留出pos位置进行插入,最后将数据插入到pos位置即可。
⚠注意:注意扩容以后,pos就失效了,要记得更新pos,否则就会发生迭代器失效。(迭代器失效问题我们在下一篇文章讲)我们可以通过设定变量n来计算扩容前pos指针位置和_start指针位置的相对距离,最后在扩容后,让_start再加上先前算好的相对距离n就是更新后的pos指针的位置了。
//insert iterator insert(iterator pos, const T& x) {//检测参数合法性assert(pos >= _start && pos <= _finish);//检测是否需要扩容/*扩容以后pos就失效了,需要更新一下*/if (_finish == _endofstoage){size_t n = pos - _start;//计算pos和start的相对距离size_t newcapcacity = capacity() == 0 ? 4 : capacity() * 2;reserve(newcapcacity);pos = _start + n;//防止迭代器失效,要让pos始终指向与_start间距n的位置}//挪动数据iterator end = _finish - 1;while (end >= pos){*(end + 1) = *(end);end--;}//把值插进去*pos = x;_finish++;return pos; }
pop_back尾删
我们首先判断_finish是否大于_start,如果大于,直接_finish--即可,否则什么也不需要操作。
void pop_back() {if (_finish > _start)//判断是否可以进行删除{_finish--;} }pop_back也可以复用下文的erase实现,当erase的参数为_finish时,实现的就是尾删,而_finish可以通过调用迭代器end()函数来解决。
void pop_back() {//法二:复用eraseerase(end() - 1);//不能用end()--,因为end()是传值返回,返回的是临时对象,临时对象具有常性,不能自身++或--,因此要用end() - 1 }
erase
首先要检查删除位置pos的合法性,其次从pos+1位置开始往前覆盖即可删除pos位置,最后返回的值为删除位置的下一个位置,其实返回的就是pos,因为在pos删除后,下一个值会覆盖到pos的位置上。
//erase iterator erase(iterator pos) {//检查合法性assert(pos >= _start && pos < _finish);//从pos + 1的位置开始往前覆盖,即可完成删除pos位置的值iterator it = pos + 1;while (it < _finish){*(it - 1) = *it; it++;}_finish--;return pos; }
- 补充1:
一般vector删除数据,都不考虑缩容的方案,当size() < capacity() / 2 时,可以考虑开一个size()大小的新空间,拷贝数据,释放旧空间。缩容的本质是时间换空间。一般设计不会考虑缩容,因为实际比较关注时间效率,不是太关注空间效率,因为现在硬件设备空间都比较大,空间存储也比较便宜。
- 补充2:
- erase也会存在迭代器失效,erase的失效是意义变了,或者不存在有效访问数据范围。
- 一般不会使用缩容的方案。那么erase的失效一般也就不存在野指针导致的失效。
- erase(pos)以后pos失效了,pos的意义变了,但是在不同平台下面对于访问pos的反应是不一样的,我们用的时候要以失效的角度去看待此问题。
- 对于insert和erase造成迭代器失效问题,linux的g++平台检查很佛系,基本靠操作系统本身野指针越界检查机制。windows下VS系列检查更严格一些,使用一些强制检查机制,意义变了可能会检查出来。
- 虽然g++对于迭代器失效检查时是非常佛系的,但是套在实际场景中,迭代器意义变了,也会出现各种问题。
clear清空数据
只需要把起始位置的指针_start赋给有效数据指针_finish即可完成数据的清空。
//clear清空数据 void clear() {_finish = _start; }
相关文章:
【STL】模拟实现vector
目录 1、基本成员变量 2、默认成员函数 构造函数 析构函数 拷贝构造函数 赋值运算符重载函数 3、容器访问相关函数接口 operator [ ]运算符重载 迭代器 范围for 4、vector容量和大小相关函数 size和capacity reserve扩容 resize swap交换数据 empty 5、修…...
Window 的 PHP XAMPP 安装 mongodb 的扩展
需要安装的扩展为:extensionphp_mongodb.dll根据官方的指引:PHP: Installing the MongoDB PHP Driver on Windows - Manual 1需要到 GitHub 上下载扩展,然后进行安装。这里的版本选择有些讲究。首先1.51 是 mongoDB 的驱动版本号,…...
Codeforces Round #849 (Div. 4)(E~G)
A~D比较简单就不写了,哎嘿E. Negatives and Positives给出一个数组a,可以对数组进行若干次操作,每次操作可以将相邻的两个数换为它们的相反数,求进行若干次操作之后能得到数组和的最大值是多少。思路:最大的肯定是把负…...
网易云音乐财报解读:收入大增亏损收窄,“云村”草长莺飞
独家版权时代结束后,在线音乐产业进入了新的发展阶段,各家音乐平台经营状况备受关注。 2月23日,网易云音乐公布了2022年全年财务业绩。财报显示,网易云音乐2022年全年收入为90亿元,较2021年同比增长28.5%。 值得一提的…...
MariaDB-10.8.6安装+主从搭建
【系统版本】CentOS 7.x Linux version 3.10.0-1062.18.1.el7.x86_64【检查系统是否安装过Mysql|mariadb】【查看是否安装Mysql|mariadb】#搜索mysql rpm -qa|grep mysql #搜索mariadb rpm -qa|grep mariadb #搜索MariaDB rpm -qa|grep MariaDB #如果安装过Mysql|mariadb&#…...
Win11系统user profile service服务登录失败解决方法
Win11系统user profile service服务登录失败解决方法分享。有用户在使用电脑的时候遇到了一些问题,系统的user profile service服务无法登录了。出现这个问题可能是系统文件损坏,或者中了病毒。接下来我们一起来看看如何解决这个问题的操作方法分享吧。 …...
Solon2 之基础:四、应用启动过程与完整生命周期
串行的处理过程(含六个事件扩展点 两个函数扩展点),代码直接、没有什么模式。易明 提醒: 启动过程完成后,项目才能正常运行(启动过程中,不能把线程卡死了)AppBeanLoadEndEvent 之前…...
Java性能分析
0、问题代码: 代码问题其实很明显,但是这里主要是为了练习如何使用工具进行分析 所以最好先不要看代码,假装不知道程序逻辑,而是先通过工具去分析,再结合分析数据去看代码,从而推出问题点在哪 import jav…...
2023年阿里云ECS服务器S6/C6/G6/N4/R6/sn2ne/sn1ne/se1ne处理器CPU性能详解
阿里云ECS服务器S6/C6/G6/N4/R6/sn2ne/sn1ne/se1ne处理器CPU性能怎么样?阿里云服务器优惠活动机型有云服务器S6、计算型C6、通用型G6、内存型R6、云服务器N4、云服务器sn2ne、云服务器sn1ne、云服务器se1ne处理器CPU性能详解及使用场景说明。 1、阿里云服务器活动机…...
数据分析与SAS学习笔记8
过程步:一个典型的SAS完整程序: 代码说明: 1)reg:回归分析; 2)model:因变量和自变量。 proc开头部分叫过程步。 常用过程: SORT过程: PRINT过程与FORTMAT…...
切割多个conf文件Nginx和Apache配置多版本PHP
有时候我们的项目不可能都是同一个PHP版本,需要每个项目都配置不同版本的PHP,宝塔和PHPStudy就是通过以下配置实现的:Nginx切割conf(非选)在nginx.conf添加include vhosts/*.conf;这样Nginx会自动引入当前目录->vho…...
使用Navicat进行SSH加密方式连接MySQL数据库
前言近年来网络安全形式日趋严峻,为保障企业信息安全和业务连续性,越来越多的要求业务系统上线前需要满足等保要求。其中数据库作为存储数据的载体,安全更是重中之重。部分等保要求,mysql数据库不能通过直连方式连接,需…...
大数据Hadoop教程-学习笔记04【数据仓库基础与Apache Hive入门】
视频教程:哔哩哔哩网站:黑马大数据Hadoop入门视频教程 总时长:14:22:04教程资源: https://pan.baidu.com/s/1WYgyI3KgbzKzFD639lA-_g 提取码: 6666【P001-P017】大数据Hadoop教程-学习笔记01【大数据导论与Linux基础】【17p】【P018-P037】大…...
20230223 刚体上的两个点速度之间的关系
刚体上的两个点速度之间的关系 注意:这里所讨论的都是投影在惯性坐标系上的。 dMAdMOdOAdMOdCA−dCOd_{_{MA}}d_{_{MO}}d_{_{OA}}d_{_{MO}}d_{_{CA}}-d_{_{CO}}dMAdMOdOAdMOdCA−dCO 求导 d˙MAd˙MOd˙CA−d˙CO\dot d_{_{MA}}\dot d_{_…...
17.1 Display system tasks
系统任务的显示组分为三类:显示和写入任务、选通监视任务和连续监视任务。17.1.1 The display and write tasks $display和$write系统任务的语法如语法17-1所示。 display_tasks ::display_task_name [ ( list_of_arguments ) ] ; display_task_name ::$display | …...
【4】linux命令每日分享——cd切换路径
大家好,这里是sdust-vrlab,Linux是一种免费使用和自由传播的 类UNIX操作系统,Linux的基本思想有两点:一切都是文件;每个文件都有确定的用途;linux涉及到IT行业的方方面面,在我们日常的学习中&am…...
诚邀您体验人工智能AI
近期,人工智能(AI)领域动作频频,OPENAI公司Chat GPT的出现,标志着人工智能的研究与应用已经进入了一个崭新的发展阶段,国内腾讯、阿里巴巴、百度、易网、国外微软、谷歌、苹果、IBM、Amazon,等互…...
【蓝桥杯集训·每日一题】AcWing 2058. 笨拙的手指
文章目录一、题目1、原题链接2、题目描述二、解题报告1、思路分析2、时间复杂度3、代码详解三、知识风暴哈希表秦九韶算法一、题目 1、原题链接 2058. 笨拙的手指 2、题目描述 奶牛贝茜正在学习如何在不同进制之间转换数字。 但是她总是犯错误,因为她无法轻易的用两…...
运维排查篇 | Linux 连接跟踪表满了怎么处理
nf_conntrack (在老版本的 Linux 内核中叫 ip_conntrack )是一个内核模块,用于跟踪一个网络连接的状态 一旦内核 netfilter 模块 conntrack 相关参数配置不合理,导致 nf_conntrack table full ,就会出现丢包、连接无法建立的问题 这个问题其…...
docker网络基
本文简单介绍下,容器之间的网络访问、容器与宿主机之间的网络访问、宿主机上有哪些网络接口。lolocal的简写,本地回环地址,127.0.0.1,它代表本地虚拟设备接口,默认被看作是永远不会宕掉的接口eth0ethernet的简写&#…...
React 第五十五节 Router 中 useAsyncError的使用详解
前言 useAsyncError 是 React Router v6.4 引入的一个钩子,用于处理异步操作(如数据加载)中的错误。下面我将详细解释其用途并提供代码示例。 一、useAsyncError 用途 处理异步错误:捕获在 loader 或 action 中发生的异步错误替…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...
Xshell远程连接Kali(默认 | 私钥)Note版
前言:xshell远程连接,私钥连接和常规默认连接 任务一 开启ssh服务 service ssh status //查看ssh服务状态 service ssh start //开启ssh服务 update-rc.d ssh enable //开启自启动ssh服务 任务二 修改配置文件 vi /etc/ssh/ssh_config //第一…...
大语言模型如何处理长文本?常用文本分割技术详解
为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...
Python实现prophet 理论及参数优化
文章目录 Prophet理论及模型参数介绍Python代码完整实现prophet 添加外部数据进行模型优化 之前初步学习prophet的时候,写过一篇简单实现,后期随着对该模型的深入研究,本次记录涉及到prophet 的公式以及参数调优,从公式可以更直观…...
Nginx server_name 配置说明
Nginx 是一个高性能的反向代理和负载均衡服务器,其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机(Virtual Host)。 1. 简介 Nginx 使用 server_name 指令来确定…...
什么是EULA和DPA
文章目录 EULA(End User License Agreement)DPA(Data Protection Agreement)一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA(End User License Agreement) 定义: EULA即…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...
【JavaWeb】Docker项目部署
引言 之前学习了Linux操作系统的常见命令,在Linux上安装软件,以及如何在Linux上部署一个单体项目,大多数同学都会有相同的感受,那就是麻烦。 核心体现在三点: 命令太多了,记不住 软件安装包名字复杂&…...








