c++|vector使用及模拟实现

news2025/1/11 11:19:02

目录

一、vector的介绍

二、vector的使用(常用接口)

2.1string类的成员函数

2.1.1构造函数

2.1.2析构函数

2.1.3“=”运算符重载函数 

2.2 迭代器(iterator) 及 对象的遍历访问

2.2.1iterator

2.2.2 operator[] && at()

2.2.4 back() && front()

2.2.5 data

2.3vector类对象的容量操作

2.3.1 size() 

2.3.2capacity() && reserve && resize

​编辑2.3.3  empty() && clear()

2.3.4 shrink_to_fit() 

2.4vector类对象的修改操作

2.4.1 push_back  && assign && insert

2.4.2 find(算法库)

2.4.3 pop_back && erase

三、vector模拟实现

前一章学习了string,那么学习vector的过程就会轻松很多,他们的接口用法大致都是相同的,对于剩余的容器,都可以以此类推学习,轻松拿捏。学习过程中,先学会如何使用,再来剖析底层大致如何实现,要做到完全理解底层,目前水平还是不够的,最后来实现一个简易版的vector。

一、vector的介绍

跟string一样他也是一个类模板,跟string不一样的是由于没被typedef过,定义对象时需要显示实例化,例如:vector<int> v;方括号中传的是类型,表示v对象中的数据类型是int类型,vector<string> v1;表示v1对象中的数据类型是string类型

  • vector本质上可以理解为一个数组,数组是可变的,即可变大小的数组序列容器。
  • 它的空间是连续的 
  • 它的底层是使用动态分配数组来存储元素
  • 分配的空间比实际需要的存储空间更大来适应可能的空间增长。对于已经分配的空间,后续可能还要添加元素会让空间增大,所以一次性分配更大的空间。
  • vector相比其他动态序列容器,访问元素更具高效性,对于其他不在末尾修改元素的容器,vector是可以做修改的

二、vector的使用(常用接口)

跟string一样,在学习vector的过程中,一定要去查看他的文档vector - C++ Reference (cplusplus.com)icon-default.png?t=N7T8https://legacy.cplusplus.com/reference/vector/vector/,文档包含他的各种接口的介绍及用法,当然,其中可能会遇到新东西,这是很正常的,先了解,在学习使用。在学习了string类的接口之后,这里就不再过多详细介绍他们的用意了,而是直接使用,直到遇到新的接口再详说。在使用vector接口需包含头文件#include <vector>

2.1string类的成员函数

2.1.1构造函数

对于以下函数中的参数,不认识的先没必要去知道,先知道函数的作用及如何使用

default (1)
   explicit vector (const allocator_type& alloc = allocator_type());
                                                    创建空对象

fill (2)
explicit vector (size_type n, const value_type& val = value_type(),
                 const allocator_type& alloc = allocator_type());
                                                 用n个值初始化对象

range (3)
template <class InputIterator>
         vector (InputIterator first, InputIterator last,
                 const allocator_type& alloc = allocator_type());
用一段区间的值初始化对象,模板参数可以是任何类型,如迭代器类型,指针类型
copy (4)
                        vector (const vector& x);
                                                         拷贝构造
#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<int> v;//构造一个空对象
	vector<int> v1(4, 2);//用4个2初始化对象v1
	for (auto e : v1)
	{
		cout << e;
	}
	cout << endl;
	
	vector<int> v2(v1.begin(), v1.end());//用v1对象的迭代器区间的值初始化V2
	for (auto e : v2)
	{
		cout << e;
	}
	cout << endl;

	char arr[] = { 1,2,3,4,5 };
	vector<int> v3(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化对象v3
	for (auto e : v3)
	{
		cout << e;
	}
	cout << endl;

	vector<int> v4(v3);//拷贝构造
	for (auto e : v4)
	{
		cout << e;
	}
	cout << endl;
	return 0;
}

2.1.2析构函数

~vector(); 这个没啥好说的,对象销毁前编译器会自动调用

2.1.3“=”运算符重载函数 

vector& operator=(const vector& x);

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<int> v;//构造一个空对象
	vector<int> v1(4, 2);//用4个2初始化对象v1
	v = v1;//v1赋值给v
	for (auto e : v)
	{
		cout << e;
	}
	cout << endl;
	return 0;
}

 

2.2 迭代器(iterator) 及 对象的遍历访问

2.2.1iterator

begin + end

#include <iostream>
#include <vector>
using namespace std;

int main()
{

	vector<int> v1(4, 2);//用4个2初始化对象v1
	vector<int>::iterator it = v1.begin();
	
	while (it != v1.end())
	{
		cout << *it;
		it++;
	}
	cout << endl;
	return 0;
}

 

rbegin + rend 反向迭代

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr)/sizeof(arr[0]));//用指针区间的值初始化v1
	vector<int>::reverse_iterator it = v1.rbegin();
	
	while (it != v1.rend())
	{
		cout << *it;
		it++;
	}
	cout << endl;
	return 0;
}

 

2.2.2 operator[] && at()

重载方括号,像数组一样使用下标进行访问    at()跟重载方括号差不多,返回对应位置的元素

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		cout << v1[i];
	}
	cout << endl;

	for (int i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
	{
		cout << v1.at(i);
	}
	cout << endl;
	return 0;
}

 

2.2.4 back() && front()

分别获取对象的最后一个元素和第一个元素

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	cout << v1.front() << endl;
	cout << v1.back() << endl;
	return 0;
}

2.2.5 data

value_type* data() noexcept;返回指向第一个元素的指针,指向元素可修改
const value_type* data() const noexcept;返回指向第一个元素的指针,指向元素不可修改

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	int* p = v1.data();
	*p = 10;
	p++;
	*p = 20;
	p[2] = 100;
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl ;
	return 0;
}

2.3vector类对象的容量操作

2.3.1 size() 

注意:vector类中没有提供length的操作

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	for (int i = 0; i < v1.size(); i++)
	{
		cout << v1[i] << " ";
	}
	cout << endl;

	return 0;
}

2.3.2capacity() && reserve && resize

capacity():获取当前对象的容量大小,reserve():扩容,但不会影响数据,resize():既影响容量,又影响数据 

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	cout << v1.capacity() << endl;
	
	v1.reserve(20);
	cout << v1.capacity() << endl;

	v1.resize(30, 1);//扩容,在原先基础上,多余的空间初始化为1,默认初始化为0
	cout << v1.capacity() << endl;
	for (int i = 0; i < v1.size(); i++)
	{
		cout << v1[i] << " ";
	}
	cout << endl;

	return 0;
}

2.3.3  empty() && clear()

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	while (!v1.empty())
	{
		for (int i = 0; i < v1.size(); i++)
		{
			cout << v1[i] << " ";
		}
		cout << endl;
		v1.clear();//清空所有元素
	}
	

	return 0;
}

 

2.3.4 shrink_to_fit() 

 void shrink_to_fit();缩容到与_size一样大,不会影响数据

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	int arr[] = { 1,2,3,4,5 };
	vector<int> v1(arr, arr + sizeof(arr) / sizeof(arr[0]));//用指针区间的值初始化v1
	cout << v1.size() << endl;
	cout << v1.capacity() << endl;

	v1.reserve(100);
	cout << v1.capacity() << endl;

	v1.resize(3);//没有缩容,但改变了size大小
	cout << v1.size() << endl;
	cout << v1.capacity() << endl;

	v1.shrink_to_fit();//缩容到与size一样大小
	cout << v1.capacity() << endl;


	return 0;
}

 

2.4vector类对象的修改操作

2.4.1 push_back  && assign && insert

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<double> v;
	v.push_back(1.1);
	v.push_back(2.2);
	v.push_back(3.3);
	v.push_back(4.4);
	v.push_back(5.5);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	v.assign(4, 10.1);//4个10.1赋值给对象,原先的内容会被替代,size也会改变
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	vector<double> v1;
	v1.assign(v.begin() + 1, v.end() - 1);//用迭代器区间之间的值赋值给v1
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;


	v1.insert(v1.end(), 20.2);//末尾插入20.2
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;
	v1.insert(v1.end(), 2, 30.3);//末尾插入2个30.3
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;
	v1.insert(v1.begin(), v.begin(), v.begin() + 2);//头部插入迭代器区间的值
	for (auto e : v1)
	{
		cout << e << " ";
	}
	cout << endl;
	return 0;
}

2.4.2 find(算法库)

注意:在vector中并没有提供find接口,但算法库中有find,所以可以使用算法库中的find,也要包含一下算法库的头文件。 

template <class InputIterator, class T>
   InputIterator find (InputIterator first, InputIterator last, const T& val);左闭右开
在迭代器区间查找值,找到,返回指向该值的迭代器。没找到,返回指向迭代器区间的last位置的迭代器

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<double> v;
	v.push_back(1.1);
	v.push_back(2.2);
	v.push_back(3.3);
	v.push_back(4.4);
	v.push_back(5.5);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	vector<double>::iterator it = find(v.begin(), v.end(), 2.2);
	cout << *it << endl;
	return 0;
}

 

2.4.3 pop_back && erase

函数原型:

void pop_back();删除最后一个位置的元素,size减1

iterator erase (iterator position);删除迭代器指向位置的元素
iterator erase (iterator first, iterator last);删除迭代器指向区间的值

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<double> v;
	v.push_back(1.1);
	v.push_back(2.2);
	v.push_back(3.3);
	v.push_back(4.4);
	v.push_back(5.5);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	v.pop_back();
	v.pop_back();
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;

	v.erase(v.begin());
	v.erase(v.begin(), v.begin() + 1);
	for (auto e : v)
	{
		cout << e << " ";
	}
	cout << endl;
	return 0;
}

三、vector模拟实现

//vector.h
#include <iostream>
#include <assert.h>
using namespace std;

namespace bit
{
	
	template<class T>
	class vector
	{
	public:
		typedef T* iterator;
		typedef const T* const_iterator;
		iterator begin()
		{
			return _start;
		}
		iterator end()
		{
			return _finish;
		}
		const_iterator begin() const
		{
			return _start;
		}
		const_iterator end() const
		{
			return _finish;
		}
		vector()
		{}
		//vector(const vector<T>& x)
		//{
		//	_start = new T[x.capacity()];
		//	memcpy(_start, x._start, x.size() * sizeof(T));
		//	_finish = _start + x.size();
		//	_end_of_storage = _start + x.capacity();
		//}

		vector(const vector<T>& v)
		{
			reserve(v.capacity());
			for (auto& e : v)
			{
				push_back(e);
			}
		}
		//vector(size_t n, const T& x = T())//为了兼容模板参数,内置类型也有默认构造函数
		//{
		//	if (_start)
		//	{
		//		delete[] _start;
		//		_start = new T[n];
		//		for (int i = 0; i < n; i++)
		//		{
		//			_start[i] = x;
		//		}
		//		_finish += n;
		//		_end_of_storage = _finish;
		//	}

		//}

		vector(size_t n, const T& x = T())
		{
			resize(n, x);
		}
	/*	template<class InputIterator>
		vector(InputIterator first, InputIterator last)
		{
			if (_start)
			{
				delete[] _start;
				_start = new T[last - first];
				memcpy(_start, first, last - first);
				_finish += last - first;
				_end_of_storage = _finish;
			}

		}*/
		template<class InputIterator>
		vector(InputIterator first, InputIterator last)
		{
			while (first != last)
			{
				push_back(*first);
				first++;
			}
		}
		vector<T>& operator=(vector<T> x)
		{
			swap(x);
			return *this;
		}
		~vector()
		{
			delete[] _start;
			_start = _finish = _end_of_storage = nullptr;
		}

		size_t size() const
		{
			return _finish - _start;
		}
		void reserve(size_t n)
		{

			if (n > capacity())
			{
				size_t len = size();
				iterator tmp = new T[n];
				
				if (_start)
				{
					//memcpy(tmp, _start, sizeof(T) * (_finish - _start));//浅拷贝
					for (int i = 0; i < len; i++)
					{
						tmp[i] = _start[i];
					}

					delete[] _start;
				}
				
				_start = tmp;
				_finish = _start + len;
				_end_of_storage = _start + n;
			}
			
		}
		void resize(size_t n, T x = T())
		{
			if (n > size())
			{
				reserve(n);
				while (_finish < _end_of_storage)
				{
					*_finish = x;
					_finish++;
				}
			}
			else
			{
				_finish = _start + n;
			}
		}

		size_t capacity() const
		{
			return _end_of_storage - _start;
		}
		void push_back(const T& x)
		{
			if (_finish == _end_of_storage)
			{
				size_t newstorage = capacity() == 0 ? 4 : capacity() * 2;
				reserve(newstorage);
			}
			//_start[_finish - _start] = x;
			*_finish = x;
			_finish++;
		}

		void pop_back()
		{
			if (size())
			{
				--_finish;
			}
		}

		iterator insert(iterator position, const T& x)
		{
			assert(position >= _start && position <= _finish);
			if (_finish == _end_of_storage)
			{
				size_t newstorage = capacity() == 0 ? 4 : capacity() * 2;
				reserve(newstorage);
			}
			iterator end = _finish - 1;
			while (end >= position)
			{
				*(end + 1) = *end;
			}
			*position = x;
			_finish++;
			return position;
		}

		iterator erase(iterator position)
		{
			assert(position < size());
			if (size())
			{
				size_t len = _finish - position - 1;
				for (int i = 0; i < len; i++)
				{
					*(position + i) = *(position + i + 1);
				}
				--_finish;
			}
			return position;
		}
		void swap(vector<T>& x)
		{
			std::swap(_start, x._start);
			std::swap(_finish, x._finish);
			std::swap(_end_of_storage, x._end_of_storage);
		}
	
		T& operator[](size_t pos)
		{
			assert(pos < size());
			return _start[pos];
		}
		const T& operator[](size_t pos) const
		{
			return _start[pos];
		}

		void vectortest()
		{
			vector<int> v;
			v.push_back(1);
			v.push_back(2);
			v.push_back(3);
			v.push_back(4);
			v.push_back(5);

			for (auto e : v)
			{
				cout << e;
			}
			cout << endl;
			cout << v.capacity() << endl;

			v.reserve(100);
			cout << v.capacity() << endl;

			vector<string> s;
			s.push_back("a");
			vector<string> s1 = s;
			for (auto& e : s1)
			{
				cout << e;
			}
			cout << endl;
			s.resize(10,"a");
			for (auto& e : s)
			{
				cout << e;
			}
			cout << endl;
			cout << s.capacity() << endl;
		}
	private:
		iterator _start;
		iterator _finish;
		iterator _end_of_storage;
	};
}
//test.cpp
#include "vector.h"

int main()
{
	bit::vector<int> s;
	s.vectortest();

	
	return 0;
}

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

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

相关文章

认识什么是Git

目录 1. 认识Git 1.1. 问题引入 1.2. 概念 1.3. 作用 1.4. 如何学 1.5. Git 安装 1.6. Git配置用户信息 2. Git仓库 2.1. Git 仓库&#xff08;repository&#xff09; 2.2. 创建 2.3. 需求 3. Git的三个区域 3.1. Git 使用时的三个区域 3.2. 工作区的内容&#…

5.动态规划

1.背包问题 (1)0/1背包问题 01背包问题即每个物品只能选1个 考虑第i件物品&#xff0c;当j<w[i]时&#xff0c;f[i][j]f[i-1][j]&#xff0c;当j>w[i]时&#xff0c;此时有两种选择&#xff0c;选择第i件物品和不选第i件物品。此时f[i][j]max(f[i-1][j],f[i-1][j-w[i]]v…

c++20协程详解(一)

前言 本文是c协程第一篇&#xff0c;主要是让大家对协程的定义&#xff0c;以及协程的执行流有一个初步的认识&#xff0c;后面还会出两篇对协程的高阶封装。 在开始正式开始协程之前&#xff0c;请务必记住&#xff0c;c协程 不是挂起当前协程&#xff0c;转而执行其他协程&a…

789. 数的范围 (二分学习)左端大右,右端小左

题目链接https://www.acwing.com/file_system/file/content/whole/index/content/4317/ 当求左端点时&#xff0c;条件是a【mid】大于等于x&#xff0c;并把右端点缩小。 当求右端点时&#xff0c;条件是a【mid】小于等于x&#xff0c;并把左端点扩大。 1.确定一个区间&…

面试复盘1 - 测试相关(实习)

写在前&#xff1a;hello&#xff0c;大家早中晚上好~这里是西西&#xff0c;最近有在准备测试相关的面试&#xff0c;特此开设了新的篇章&#xff0c;针对于面试中的问题来做一下复盘&#xff0c;会把我自己遇到的问题进行整理&#xff0c;除此之外还会进行对一些常见面试题的…

CentOSPython使用openpyxl、pandas读取excel报错

一、openpyxl报错 由于本人在centos7下使用python的openpyxl报错,所以决定使用强大的pandas。 pandas是一个快速、强大、灵活且易于使用的开源数据操作和分析工具,建立在Python语言之上。 pandas是一个广泛使用的数据分析库,它提供了高效的数据结构和数据分析工具。pandas…

new mars3d.layer.HeatLayer({实现动态修改热力图半径

1.使用热力图插件的时候&#xff0c;实现动态修改热力图效果半径 2.直接修改是不可以的&#xff0c;因为这个是热力图本身的参数。 因此我们需要拿到这个热力图对象之后&#xff0c;参考api文档&#xff0c;对整个 heatLayer.heatStyle进行传参修改。 heatStyle地址&#x…

SSL通配符证书怎么选?看这里

通配符证书&#xff0c;作为一种特殊的数字证书类型&#xff0c;以其独特的优势在网络安全领域扮演着重要角色。相较于传统的单一域名证书&#xff0c;通配符证书能够为同一主域名下的所有子域名提供安全保护&#xff0c;显著提升管理效率&#xff0c;简化证书部署流程&#xf…

【深度学习】sdwebui的token_counter,update_token_counter,如何超出77个token的限制?对提示词加权的底层实现

文章目录 前言关于token_counter关于class StableDiffusionProcessingTxt2Img(StableDiffusionProcessing)如何超出77个token的限制&#xff1f;对提示词加权的底层实现Overcoming the 77 token limit in diffusers方法1 手动拼方法2 compel 问询、帮助请看&#xff1a; 前言 …

[已解决] slam_gmapping: undefined symbol: _ZN8GMapping14sampleGaussianEdm问题

之前用的好好的gampping建图功能包&#xff0c;今天突然不能用了&#xff0c;运行报错如下&#xff1a; /opt/ros/noetic/lib/gmapping/slam_gmapping: symbol lookup error: /opt/ros/noetic/lib/gmapping/slam_gmapping: undefined symbol: _ZN8GMapping14sampleGaussianEdm …

首场直播,就在4月11日!

2024年的第一场直播&#xff0c;我们把目光聚焦到“大会员”。 这一次我们想聊聊&#xff0c;当大会员遇上泛零售企业&#xff0c;会产生怎样的“火花”。泛零售企业突破增长压力的机会在哪里&#xff1f;又有哪些挑战必须直面&#xff1f; 本次直播将结合泛零售企业“多业态、…

双机 Cartogtapher 建图文件配置

双机cartogtapher建图 最近在做硕士毕设的最后一个实验&#xff0c;其中涉及到多机建图&#xff0c;经过调研最终采用cartographer建图算法&#xff0c;其中配置多机建图的文件有些麻烦&#xff0c;特此博客以记录 非常感谢我的同门 ”叶少“ 山上的稻草人-CSDN博客的帮助&am…

Chrome 设置在新窗口中打开链接(已登录google账号版)

Chrome的链接默认是在原标签页中打开的&#xff0c;如果要在新窗口中打开&#xff0c;需要自己自行设置&#xff0c;在此&#xff0c;针对已经登录google账号的chrome浏览器怎么进行设置进行说明。 一、点击登录图标->更多设置 二、选择其他设置->在新窗口中打开搜索结果…

Linux上管理文件系统

Linux上管理文件系统 机械硬盘 机械硬盘由多块盘片组成&#xff0c;它们都绕着主轴旋转。每块盘片上下方都有读写磁头悬浮在盘片上下方&#xff0c;它们与盘片的距离极小。在每次读写数据时盘片旋转&#xff0c;读写磁头被磁臂控制着不断的移动来读取其中的数据。 所有的盘片…

一套C#自主版权+应用案例的手麻系统源码

手术麻醉信息管理系统源码&#xff0c;自主版权应用案例的手麻系统源码 手术麻醉信息管理系统包含了患者从预约申请手术到术前、术中、术后的流程控制。手术麻醉信息管理系统主要是由监护设备数据采集子系统和麻醉临床系统两个子部分组成。包括从手术申请到手术分配&#xff0c…

SSM框架学习——JSP语法入门

JSP语法入门 前提 在前一节中我们已经写过JSP的代码了&#xff0c;这一节将单独介绍JSP一些基础语法。当然&#xff0c;你可以跳过这一节&#xff0c;当后面有代码不太理解的时候再回来阅读。 中文编码问题 如果中文乱码&#xff0c;看看JSP是否是以UTF8的方式编码&#xf…

mysql建表必须知道的18个重点(荣耀典藏版)

大家好&#xff0c;我是月夜枫&#xff0c;又和大家见面了&#xff01;&#xff01;&#xff01;&#xff01; 目录 前言 1.名字 1.1 见名知意 1.2 大小写 1.3 分隔符 1.4 表名 1.5 字段名称 1.6 索引名 2.字段类型 3.字段长度 4.字段个数 5. 主键 6.存储引擎 7.…

计算机网络:数据链路层 - 点对点协议PPP

计算机网络&#xff1a;数据链路层 - 点对点协议PPP PPP协议的帧格式透明传输字节填充法零比特填充法 差错检测循环冗余校验 对于点对点链路&#xff0c;PPP协议是目前使用最广泛的数据链路层协议。比如说&#xff0c;当用户想要接入互联网&#xff0c;就需要通过因特网服务提供…

vulnhub靶机: DC-9

dc-9靶机下载 将靶机设置为NAT模式&#xff0c;本次实验使用的内网网段为192.168.198.0/24&#xff0c;kali的ip为192.168.198.172 信息搜集 ip主机扫描&#xff1a; nmap -sP 192.168.198.0/24 确定靶机ip为192.168.198.171 主机端口扫描&#xff1a; nmap -T4 -A -v 192…

【JVM】如何定位、解决内存泄漏和溢出

目录 1.概述 2.堆溢出、内存泄定位及解决办法 2.1.示例代码 2.2.抓堆快照 2.3.分析堆快照 1.概述 常见的几种JVM内存溢出的场景如下&#xff1a; Java堆溢出&#xff1a; 错误信息: java.lang.OutOfMemoryError: Java heap space 原因&#xff1a;Java对象实例在运行时持…