<C++>STL->list

news2025/1/23 15:07:17

list的介绍

listimage-20240120113126755

  • list是一个类模板,第一个模板参数为存储数据类型;第二个模板参数为空间适配器
  • list是一个可以在常数时间内完成任意位置的插入和删除的顺序容器。
  • list容器是以双链表的形式实现的;双链表可以将其包含的每个元素存储在不同且不相关的存储位置。每个元素都有一个指向其前面元素的指针和一个指向其后面元素的指针,从而在内部保持排序。
  • 与其他基本标准序列容器(arrayvectordeque)相比,list在插入、提取和移动容器内任何位置的元素(已获得迭代器)时通常表现更好,因此在密集使用这些元素的算法(如排序算法)中也是如此。
  • 与其他序列容器相比,listforward_lists 的主要缺点是它们无法通过元素的位置直接访问元素;例如,要访问列表中的第六个元素,必须从已知位置(如开始或结束位置)迭代到该位置,而这需要花费这些位置之间距离的线性时间。此外,它们还需要消耗一些额外的内存来保存与每个元素相关的链接信息(这可能是由小尺寸元素组成的大列表的一个重要因素)。

list的使用

list的使用和vector类似,主要介绍listvector之间的不同点:

  • 时间复杂度:vector获取元素的时间复杂度为 O ( 1 ) O(1) O(1),尾插和尾删的时间复杂度为 O ( n ) O(n) O(n),其他位置插入和删除是 O ( n ) O(n) O(n);list获取元素的时间复杂度为 O ( n ) O(n) O(n),任意位置插入删除时间复杂度为 O ( 1 ) O(1) O(1)

  • 迭代器失效:vector在insert,erase之后,原本指向插入、删除后面所有元素的迭代器都会失效;list在insert,erase之后,只有指向插入或删除位置的迭代器才会失效

  • clear成员函数:vectorclear函数会将size清零,capacity保持不变;listclear函数会将size清零,只留下头节点

  • 迭代器的实现不同:

    • vector的迭代器底层就是原生指针。即typedef T* std::vector::iterator ,此后对迭代器进行解引用和++就相当于对指针进行解引用和++,可以访问容器的数据和遍历容器。
    • list的迭代器底层是指针,但不是原生指针,而是封装后的指针。在使用上我们需要解引用迭代器可以获得节点的数据,迭代器++可以遍历容器。list的数据结构是链表,链表各个节点的存储是不连续的,所以直接对指针进行解引用和++是不能实现获取链表节点的数据和遍历链表的。我们必须对指针进行运算符重载,但是内置类型不支持运算符重载,需要将指针封装成一个以该指针为成员变量的类,对类进行operator+operator*重载。
    • vector没有的成员函数:
    1. spliceimage-20240120213123046

      • 功能:将元素从 x 传输到容器中,并将它们插入到位置

      • 参数:
        position:容器内插入x元素的位置
        x:与*this具有相同类型元素的list
        i:x中的迭代器,只有单个数据会被传输
        first,last:指向容器x要被传输区间的头和尾部的下一个位置,及区间[first,last)会被传输到*thisposition位置。

      • 返回值:空

      • 注意:再调用函数后,原本指向元素的迭代器仍然指向该元素,但是迭代器可能指向的容器发生改变

      demo:

      // splicing lists
      #include <iostream>
      #include <list>
      
      int main ()
      {
        std::list<int> mylist1, mylist2;
        std::list<int>::iterator it;
      
        // set some initial values:
        for (int i=1; i<=4; ++i)
           mylist1.push_back(i);      // mylist1: 1 2 3 4
      
        for (int i=1; i<=3; ++i)
           mylist2.push_back(i*10);   // mylist2: 10 20 30
      
        it = mylist1.begin();
        ++it;                         // points to 2
      
        mylist1.splice (it, mylist2); // mylist1: 1 10 20 30 2 3 4
                                      // mylist2 (empty)
                                      // "it" still points to 2 (the 5th element)
                                                
        mylist2.splice (mylist2.begin(),mylist1, it);
                                      // mylist1: 1 10 20 30 3 4
                                      // mylist2: 2
                                      // "it" is now invalid.
        it = mylist1.begin();
        std::advance(it,3);           // "it" points now to 30
      
        mylist1.splice ( mylist1.begin(), mylist1, it, mylist1.end());
                                      // mylist1: 30 3 4 1 10 20
      
        std::cout << "mylist1 contains:";
        for (it=mylist1.begin(); it!=mylist1.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        std::cout << "mylist2 contains:";
        for (it=mylist2.begin(); it!=mylist2.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        return 0;
      }// splicing lists
      #include <iostream>
      #include <list>
      
      int main ()
      {
        std::list<int> mylist1, mylist2;
        std::list<int>::iterator it;
      
        // set some initial values:
        for (int i=1; i<=4; ++i)
           mylist1.push_back(i);      // mylist1: 1 2 3 4
      
        for (int i=1; i<=3; ++i)
           mylist2.push_back(i*10);   // mylist2: 10 20 30
      
        it = mylist1.begin();
        ++it;                         // points to 2
      
        mylist1.splice (it, mylist2); // mylist1: 1 10 20 30 2 3 4
                                      // mylist2 (empty)
                                      // "it" still points to 2 (the 5th element)
                                                
        mylist2.splice (mylist2.begin(),mylist1, it);
                                      // mylist1: 1 10 20 30 3 4
                                      // mylist2: 2
                                      // "it" is now invalid.
        it = mylist1.begin();
        std::advance(it,3);           // "it" points now to 30
      
        mylist1.splice ( mylist1.begin(), mylist1, it, mylist1.end());
                                      // mylist1: 30 3 4 1 10 20
      
        std::cout << "mylist1 contains:";
        for (it=mylist1.begin(); it!=mylist1.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        std::cout << "mylist2 contains:";
        for (it=mylist2.begin(); it!=mylist2.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        return 0;
      }
      

      Output:

      mylist1 contains: 30 3 4 1 10 20
      mylist2 contains: 2
      
    2. removeimage-20240120220101850

      • 功能:移除容器中所有值为val的数据,会调用移除数据对应的析构函数,同时会减少容器的size。
      • 参数:需要移除的元素,类型value_type是类模板T的重定义
      • 返回值:无
      • 注意:
        • remove函数是删除所有值为val的数据,erase删除迭代器pos处的位置。
        • 指向val数据处的迭代器会失效,其余的不会失效。

      demo:

      // remove from list
      #include <iostream>
      #include <list>
      
      int main ()
      {
        int myints[]= {17,89,7,14};
        std::list<int> mylist (myints,myints+4);
      
        mylist.remove(89);
      
        std::cout << "mylist contains:";
        for (std::list<int>::iterator it=mylist.begin(); it!=mylist.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        return 0;
      }
      

      output:

      mylist contains: 17 7 14
      
    3. remove_if:删除满足条件的数据,条件需要通过仿函数传递, 等我们讲解仿函数后再来讲解。

    4. uniqueimage-20240120221434347

      • 功能:删除连续的重复元素,所有重复元素只保留第一个
      • 参数:
        第一个版本是严格删除所有值相等的连续元素。第二个版本我们可以根据传参自己定义“相同性”,通过仿函数实现。
      • 返回值:空
      • 注意:指向删除元素的迭代器会失效。

      demo:

      output:

    5. mergeimage-20240120222422941

      • 功能:通过将 x 的所有元素在各自有序位置传输到容器中(两个容器都应已排序),将 x 合并到列表中。
      • 参数:
        • x:与*this具有相同类型的list,函数调用结束后x会被修改
        • comp:仿函数参数,决定具体是如何比较的
      • 返回值:空
      • 注意:
        • 执行merge的前提是两个list都是有序的,list无序请使用std::list::splicex中的元素按照operator<或者comp定义的比较方法找到合适的位置进行插入,由此产生的等价元素顺序是稳定的。
        • 执行完后,xsize会变为0。
        • 执行该操作时,既不构造也不销毁任何元素。
        • 如果this==&x,则函数不会执行任何操作。
        • 调用函数之后,迭代器仍然指向之前的元素,但是指向的可能从一个容器变成另一个容器。

      demo:

      // list::merge
      #include <iostream>
      #include <list>
      
      // compare only integral part:
      bool mycomparison (double first, double second)
      { return ( int(first)<int(second) ); }
      
      int main ()
      {
        std::list<double> first, second;
      
        first.push_back (3.1);
        first.push_back (2.2);
        first.push_back (2.9);
      
        second.push_back (3.7);
        second.push_back (7.1);
        second.push_back (1.4);
      
        first.sort();
        second.sort();
      
        first.merge(second);
      
        // (second is now empty)
      
        second.push_back (2.1);
      
        first.merge(second,mycomparison);
      
        std::cout << "first contains:";
        for (std::list<double>::iterator it=first.begin(); it!=first.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        return 0;
      }
      

      output:

      first contains: 1.4 2.2 2.9 2.1 3.1 3.7 7.1
      

      请注意,在第二次合并中,函数 mycomparison(仅比较整数部分)没有考虑 2.1 低于 2.2 或 2.9,因此它被插入到它们之后、3.1 之前。

    6. sortimage-20240120224158440

      • 功能:对列表中的元素进行排序,改变它们在容器中的位置。排序是通过使用 operator<(版本(1))或 comp(版本(2))来比较元素的算法进行的。这种比较应产生严格的元素弱排序(即一致的反式比较,不考虑其反射性)。
      • 参数:二元谓词,在取两个与列表中的值类型相同的值时,如果第一个参数在其定义的严格弱排序中位于第二个参数之前,则返回 true,否则返回 false。该谓词应是函数指针或函数对象。
      • 返回值:空
      • 注意:
        • 整个操作不涉及任何元素对象的构造、销毁或复制。元素在容器内移动。
        • 迭代器不会失效,还是只想之前的元素。

      demo:

      // list::sort
      #include <iostream>
      #include <list>
      #include <string>
      #include <cctype>
      
      // comparison, not case sensitive.
      bool compare_nocase (const std::string& first, const std::string& second)
      {
        unsigned int i=0;
        while ( (i<first.length()) && (i<second.length()) )
        {
          if (tolower(first[i])<tolower(second[i])) return true;
          else if (tolower(first[i])>tolower(second[i])) return false;
          ++i;
        }
        return ( first.length() < second.length() );
      }
      
      int main ()
      {
        std::list<std::string> mylist;
        std::list<std::string>::iterator it;
        mylist.push_back ("one");
        mylist.push_back ("two");
        mylist.push_back ("Three");
      
        mylist.sort();
      
        std::cout << "mylist contains:";
        for (it=mylist.begin(); it!=mylist.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        mylist.sort(compare_nocase);
      
        std::cout << "mylist contains:";
        for (it=mylist.begin(); it!=mylist.end(); ++it)
          std::cout << ' ' << *it;
        std::cout << '\n';
      
        return 0;
      }
      

      output:

      mylist contains: Three one two
      mylist contains: one Three two
      

      对于默认字符串,比较是严格的字符代码比较,其中所有大写字母都比所有小写字母小,在第一次排序操作中将所有以大写字母开头的字符串放在前面。

    7. reverse image-20240120224901029

      • 功能:逆置list中的数据。
      • 参数:空
      • 返回值:空
      • 注意:迭代器不会失效,让然指向之前的元素,但是位置可能发生改变

list的实现

  • 有关list的拷贝时,注意深拷贝问题!

  • 实现list的重头主要是迭代器的实现,其余的操作和双链表没有什么区别。下面直接直接看代码~

    /*list.h*/
    #define _CRT_SECURE_ 1
    #pragma once
    #include <iostream>
    #include <assert.h>
    #include <algorithm>
    #include <string>
    #include <list>
    using std::cout;
    using std::cin;
    using std::endl;
    using std::string;
    namespace myList
    {
    	/*定义节点类型*/
    	template<class T>
    	struct Node
    	{
    		T _val;
    		Node* _next;
    		Node* _prev;
    
    		Node(const T& val = T())
    			:_val(val),
    			_next(nullptr),
    			_prev(nullptr)
    		{}
    	};
    
    	/*定义list迭代器,我们想让list迭代器的++和*与普通指针操作不同
    	所以要将迭代器封装为一个以指向节点的指针为成员变量的类,对类进行运算符重载*/
    	template <class T, class Ref, class Ptr>
    	struct __list_iterator	//封装的迭代器类
    	{
    		typedef Node<T> Node;
    		typedef __list_iterator<T, Ref, Ptr> self;//简化迭代器的名称
    		/*成员变量---指向节点的指针*/
    		Node* _node;
    		/*成员函数*/
    		__list_iterator( Node* node)			  //单参数构造函数
    			:_node(node)				
    		{}
    		__list_iterator(const self& it)			  //拷贝构造函数
    		{
    			_node = it._node;
    		}
    		//重载后缀++								  //迭代器++相当于指向节点的指针指向链表中的下一个节点---_node=_node->next
    		self operator++(int)
    		{
    			self tmp(*this);
    			_node = _node->_next;
    			return tmp;
    		}
    		//重载前缀++
    		self& operator++()
    		{
    			_node = _node->_next;
    			return *this;
    		}
    		//重载后缀--
    		self operator--(int)
    		{
    			self tmp(*this);
    			_node = _node->_prev;
    			return tmp;
    		}
    		//重载前缀--
    		self& operator--()
    		{
    			_node = _node->_prev;
    			return *this;
    		}
    		//重载*
    		Ref operator*()							   //迭代器解引用相当于访问链表节点的数据与---_node->_val
    		{
    			return _node->_val;
    		}
    		//重载->
    		Ptr operator->()						   //list存储自定义类型时,重载->使得通过it->自定义成员可以直接访问自定义类型
    		{
    			return &_node->_val;
    		}
    		//重载==
    		bool operator==(const self& it)			   //2个迭代器指向链表的节点是否一样
    		{
    			return _node == it._node;
    		}
    		//重载!=
    		bool operator!=(const self& it)
    		{
    			return _node != it._node;
    		}
    
    	};
    	template<class T>
    	class list
    	{
    	private:
    		Node<T>* _head;							  //指向头节点
    	public:
    		typedef Node<T> Node;
    		//泛型编程
    		typedef __list_iterator<T, T&, T*> iterator;//普通迭代器的operator* operator->返回值所引用/指向的对象是可以修改的,返回普通引用/指针
    		typedef __list_iterator<T, const T&, const T*> const_iterator;//const迭代器的operator* operator->的返回值所引用/指向的对象不可以修改
    																	  //无法对*it或it->__进行写入,返回常引用/指向常量的指针
    
    		//下面直接返回指针,通过单参数构造函数的隐式类型转换,可以根据返回的指针构造一个对象,传值时在进行拷贝构造-->会优化为直接构造
    		iterator begin()
    		{
    			return _head->_next;//单参数构造函数的隐式类型转换
    		}
    		const_iterator begin() const
    		{
    			return _head->_next;//单参数构造函数的隐式类型转换
    		}
    		iterator end()
    		{
    			return _head;		//单参数构造函数的隐式类型转换
    		}
    		const_iterator end() const
    		{
    			return _head;		//单参数构造函数的隐式类型转换
    		}
    
    		//初始化一个带头双向循环链表
    		void empty_init()
    		{
    			_head = new Node;
    			_head->_prev = _head->_next = _head;
    		}
    		list()
    		{
    			empty_init();
    		}
    		list(int n, const T& val = T())					//fill构造函数
    		{
    			assert(n >= 0);
    			empty_init();
    			while (n--)
    			{
    				push_back(val);
    			}
    		}
    		template<class InputIterator>
    		list(InputIterator first, InputIterator last)  //range构造函数
    		{
    			empty_init();
    			while (first != last)
    			{
    				push_back(*first++);
    			}
    		}
    		//lt1(lt2)拷贝时注意深拷贝问题,使用push_back可以进行深拷贝
    		list(const list<T>& lt)
    		{
    			empty_init();
    			for (auto e : lt)
    			{
    				push_back(e);
    			}
    		}
    		/*修改类函数*/
    		void push_back(const T& val)
    		{
    			insert(end(), val);
    		}
    		void push_front(const T& val)
    		{
    			insert(begin(), val);
    		}
    		void pop_back()
    		{
    			erase(--end());
    		}
    		void pop_front()
    		{
    			erase(begin());
    		}
    		iterator insert(iterator pos, const T& val)//pos位置前插入元素
    		{
    			Node* newnode = new Node(val);
    			Node* cur = pos._node;
    			Node* prev = pos._node->_prev;
    			//连接节点
    			prev->_next = newnode;
    			newnode->_prev = prev;
    			newnode->_next = cur;
    			cur->_prev = newnode;
    			return pos;
    		}
    		iterator erase(iterator pos)
    		{
    			assert(pos != _head);//不能删除头节点
    			Node* prev = pos._node->_prev;
    			Node* next = pos._node->_next;
    			prev->_next = next;
    			next->_prev = prev;
    			delete pos._node;
    			return next;
    		}
    
    		void clear()//除头节点外所有节点全部删除
    		{
    			iterator it = begin();
    			while (it != end())
    			{
    				it = erase(it);
    			}
    		}
    		void swap(list<T>& lt)//交换两个list的值
    		{
    			std::swap(_head, lt._head);
    		}
    		//lt1=lt3
    		list<T>& operator=( list<T> lt)
    		{
    			swap(lt);
    			return *this;
    		}
    
    		size_t size()
    		{
    			size_t sz = 0;
    			iterator it = begin();
    			while (it != end())
    			{
    				sz++;
    			}
    			return sz;
    		}
    		~list()
    		{
    			clear();
    			delete _head;
    		}
    	};
    
    		//test1现在不是成员函数
    	void test1()
    	{
    		int arr[5] = { 1, 3 ,5, 7, 9 };
    		list<int> lt1(arr, arr + 5);
    		for (auto e : lt1)
    		{
    			cout << e << " ";
    		}
    		cout << endl;
    		list<int> lt2(3, 1);
    		for (auto e : lt2)
    		{
    			cout << e << " ";
    		}
    		cout << endl;
    		cout << endl;
    	}
    	void test2()
    	{
    		list<string> lt_str;
    		lt_str.push_back("hello");
    		lt_str.push_back("list");
    		lt_str.push_back("!!!");
    		list<string> lt_copy_str(lt_str);
    		for (auto e : lt_copy_str)
    		{
    			cout << e << endl;
    		}
    		lt_copy_str.pop_back();
    		lt_copy_str.pop_front();
    		for (auto e : lt_copy_str)
    		{
    			cout << e << endl;
    		}
    		lt_str = lt_copy_str;
    		for (auto e : lt_str)
    		{
    			cout << e << endl;
    		}
    	}
    }
    
    /*main.cpp*/
    #define _CRT_SECURE_NO_WARNINGS 1
    #include "List.h"
    //using namespace std;
    int main()
    {
            myList::test1();
    	myList::test2();
    	return 0;
    }
    

    output:image-20240121094627973

说明:list的模板参数有三个

  • 第一个参数class T决定list存储的数据类型。
  • 第二个参数class Ref决定operator*的返回值类型
    1. iterator的operator*返回值类型为可以修改的引用T&
    2. const_iterator的operator*返回值类型为不可修改的引用const T&
  • 第三个参数clss Ptr决定operator->的返回值类型
    1. iterator的operator*返回值类型为指向对象可修改的指针T*
    2. const_iterator的operator->返回值类型为指向不可修改的对象的指针const T*
struct AA
{
    int a;
    int b;
};

重载operator->后,假若list中存放的是自定义类型AA,可以通过it->a1访问自定义类型中的成员image-20240121100158968

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

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

相关文章

性能进阶:使用JMeter进行websocket测试【建议收藏】

本次测试案例主要是分享如何使用JMeter进行websocket协议下的聊天接口性能测试。 包含websocket插件的下载安装、线程组及sampler的设置、csv参数化和组建分布式测试的方法、如何通过调整参数来获得发压机的最大并发数以及对测试过程的总结。 整篇文章只侧重介绍进行websocke…

西圣H1头戴式耳机,百元性能小钢炮,舒适音质性能绝不妥协

近日&#xff0c;深耕智能声学领域多年的 xisem西圣品牌‏‏发布——‏西圣H1头戴式主动降噪蓝牙耳机正式迎来首销。从各个方面来看&#xff0c;这款头戴式耳机展现出了极具颠覆性的创新&#xff0c;不仅在音质和降噪性能上表现出色&#xff0c;更重要的是采用了更加轻巧的佩戴…

外包干了5个月,技术退步明显...

先说一下自己的情况&#xff0c;大专生&#xff0c;18年通过校招进入武汉某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了四年的功能测…

顺时针打印矩阵:偏移量法与边界控制法比较---剑指offer-JZ29 顺时针打印矩阵

在编程中&#xff0c;处理二维数组的问题可以有多种解法。今天&#xff0c;我们将探讨两种解决“顺时针打印矩阵”问题的方法&#xff1a;偏移量法和边界控制法&#xff0c;并进行比较。 题目 题目连接&#xff1a;顺时针打印矩阵_牛客题霸_牛客网 (nowcoder.com) 偏移量法 偏…

枚举类型有着一篇足以

✅作者简介&#xff1a;大家好&#xff0c;我是橘橙黄又青&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;橘橙黄又青-CSDN博客 1.关键字enum的定义 enum是C语言中的一个关键字&#xff0c;enum叫枚举数据类型&#…

Unity 编辑器篇|(十三)自定义属性绘制器(PropertyDrawer ,PropertyAttribute) (全面总结 | 建议收藏)

目录 1. 前言2. PropertyDrawer2.1 参数总览2.2 两种用途2.3 注意事项2.4 代码样例 3. PropertyDrawer与PropertyAttribute结合使用 1. 前言 在Unity中&#xff0c;PropertyDrawer和PropertyAttribute是两个重要的工具&#xff0c;它们主要用于自定义属性的显示和行为。Proper…

ipv6-ipv4隧道和bgp全网通小实验

效果图:r1能ping通r11和r8,r8能ping通r11 个人理解:ipv6-ipv4隧道只认前缀,例如:r2隧道的source是其接口g0/0/1,它的ipv4 address是29.1.1.1/24,那么转化过来的ipv6就是2002:1d01:0101::/48,对端的r10只能ping通2002:1d01:0101::/48网段内的ipv6。 同理,r10隧道的source是其接…

代理设计模式JDK动态代理CGLIB动态代理原理

代理设计模式 代理模式&#xff08;Proxy&#xff09;&#xff0c;为其它对象提供一种代理以控制对这个对象的访问。如下图 从上面的类图可以看出&#xff0c;通过代理模式&#xff0c;客户端访问接口时的实例实际上是Proxy对象&#xff0c;Proxy对象持有RealSubject的引用&am…

qml定位器:Row、Column、Grid、Flow

03.qml import QtQuickWindow {width: 640height: 480visible: truetitle: qsTr("2.7 定位器")Column {RedSquare {}RedSquare {}RedSquare {}spacing: 10 //间隔}Row {RedSquare {}RedSquare {}RedSquare {}spacing: 10x: 58}Flow { //中文是“流”的意思&#xff…

特征融合篇 | YOLOv8 引入长颈特征融合网络 Giraffe FPN

在本报告中,我们介绍了一种名为DAMO-YOLO的快速而准确的目标检测方法,其性能优于现有的YOLO系列。DAMO-YOLO是在YOLO的基础上通过引入一些新技术而扩展的,这些技术包括神经架构搜索(NAS)、高效的重参数化广义FPN(RepGFPN)、带有AlignedOTA标签分配的轻量级头部以及蒸馏增…

【大数据】流处理基础概念(一):Dataflow 编程基础、并行流处理

流处理基础概念&#xff08;一&#xff09;&#xff1a;Dataflow 编程基础、并行流处理 1.Dataflow 编程基础1.1 Dataflow 图1.2 数据并行和任务并行1.3 数据交换策略 2.并行流处理2.1 延迟与吞吐2.1.1 延迟2.1.2 吞吐2.1.3 延迟与吞吐 2.2 数据流上的操作2.2.1 数据接入和数据…

企业网架构

企业网架构 局域网通信不同网段 局域网通信 MAC地址&#xff1a;硬件地址&#xff0c;固定在网卡上的地址(唯一标识一个网卡)&#xff0c;确定网络设备位置的,数据链路层。一个设备可以有多个网卡&#xff0c;每一个网卡都需要一个唯一MAC。ARP协议&#xff1a;通过目的IP&…

回归预测 | Matlab实现GA-APSO-MBP、GA-MBP、MBP、BP多输入单输出回归预测

回归预测 | Matlab实现GA-APSO-MBP、GA-MBP、MBP、BP多输入单输出回归预测 目录 回归预测 | Matlab实现GA-APSO-MBP、GA-MBP、MBP、BP多输入单输出回归预测预测效果基本描述程序设计参考资料 预测效果 基本描述 1.Matlab实现GA-APSO-MBP、GA-MBP、MBP、BP多输入单输出回归预测&…

图片如何转文字?手把手教你图片转文字

面对日益增长的数字化信息&#xff0c;图片转文字成为了一种必不可少的技术&#xff0c;通过简单的步骤&#xff0c;将图片中的文字转换为可编辑和可搜索的文本&#xff0c;提高工作效率和准确性。 这是一款通过光学字符识别&#xff08;OCR&#xff09;实现从图像中提取文字的…

细数2023测试测量及通信行业的十大年度热点动态

2023年&#xff0c;全球测试测量及通信行业在多重技术革新和市场需求的驱动下呈现出强劲的增长态势。这一年里&#xff0c;5G通信网络的大规模部署与商用、人工智能技术的广泛应用、物联网&#xff08;IoT&#xff09;设备数量的爆炸性增长以及电动汽车行业的迅猛发展等因素&am…

在CentOS 7 中配置NFS服务器

目录 1、克隆两个虚拟机 2、安装 NFS 服务 3、NFS 服务使用 1、克隆两个虚拟机 nfs-servernfs-client&#xff08;修改ip地址&#xff09;[rootxnode1 ~]# cd /etc/sysconfig/network-scripts/[rootxnode1 network-scripts]# vi ifcfg-eno16777736 #修改内容如下 BOOTPROT…

浅析Java虚拟机中的ZGC

引言 为什么需要垃圾回收&#xff08;Garbage Collection&#xff09; 垃圾回收是Java开发中的关键机制&#xff0c;负责自动管理内存&#xff0c;防止内存泄漏&#xff0c;提高开发效率和应用程序的稳定性。 Java中主要的垃圾回收方法 标记-清除算法&#xff08;Mark and …

YZ系列工具之YZ05:代码运行中调用“计算器”

我给VBA下的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。我的教程一共九套一部VBA手册&#xff0c;教程分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的…

CentOS搭建DNS服务器

服务器规划 DNS服务器IP为&#xff1a;172.16.32.253 需要自定义域名解析 172.16.32.253 dns.zhangsan.com 172.16.32.128 test1.zhangsan.com 172.16.32.129 test2.zhangsan.com 172.16.32.130 www.zhangsan.com 1. 服务器初始化 [rootlocalhost ~]# hostnamectl set-hostnam…

python|写一个简单的web应用框架

写应用框架需要写底层服务器么? 这个要区分2种情况&#xff0c;如果应用框架&#xff0c;你没有参考WSGI标准&#xff0c;那么在写应用框架之前&#xff0c;你就必须要定义一套属于自己的服务器&#xff0c;当然本文不采取这种方式&#xff0c;专业的事情应该专业的人来做。我…