C++(14.5)——再谈拷贝构造与深浅拷贝

news2024/11/23 9:26:48

       上篇文章中,通过模拟的方式完成了string类对象中常用的函数。在本篇文章中,将通过一个例子来进一步说明拷贝构造与深浅拷贝。

目录

1.再谈深浅拷贝与拷贝构造:

2. 流插入与流提取: 

2.1 流插入:

2.2 流提取:

3. 深拷贝再优化:


(注:本文需要使用上一篇文章中模拟实现关于string类的函数

1.再谈深浅拷贝与拷贝构造:

例子如下:

string str("https://legacy.cplusplus.com/");
		string sub1, sub2, sub3;
		size_t pos1 = str.find(':');
		sub1 = str.substr(0, pos1 - 0);
		cout << sub1.c_str() << endl;

		size_t pos2 = str.find('/', pos1 + 3);
		sub2 = str.substr(pos1 + 3, pos2 - (pos1 + 3));
		cout << sub2.c_str() << endl;

		sub3 = str.substr(pos2 + 1);
		cout << sub3.c_str() << endl;

上述代码的目的是将str中存储的网址进行分隔。但是当运行代码时,编译器会显示错误:

引起错误的根源在于下面给出的代码:

sub1 = str.substr(0, pos1 - 0);

对于substr代码如下:

string substr(size_t pos = 0, size_t len = npos)
		{
			assert(pos < _size);
			size_t end = pos + len;
			if (len == npos || pos + len >= _size)
			{
				end = _size;
			}
			string str;
			str.reserve(end - pos);
			for (size_t i = pos; i < end; i++)
			{
				str += _str[i];
			}
			return str;
		}

       通过上述代码,不难得出其整体逻辑为:substr函数返回string类型的对象strstr作为右操作数用来给sub1赋值。由于string函数的返回值不是引用返回,所以,substr函数并没有返回str本身,而是返回str的一个临时拷贝,为了方便描述,这里命名为str1。并且,在模拟实现的整个过程中并没有人为编写拷贝构造函数,因此,str1对于str的拷贝是一个浅拷贝。

      对于浅拷贝也就是简单的值拷贝,者导致了str,str1中的指针char*_str指向同一块空间,即:

 当substr函数结束时,会自动调用析构函数来释放str中开辟的空间,但是,此时str1依旧指向了被清理的空间,因此是一个野指针,当str1作为返回值用于给sub1赋值时,会因为野指针问题而引起错误。

为了避免上述错误,需要将拷贝构造的方式由浅拷贝改为深拷贝,所以需要人为编写拷贝构造函数。对于深拷贝,即不但需要对某些类型的变量完成值拷贝,还需要将变量中的资源一并进行拷贝,具体代码如下:

string(const string& s)
		{
			_str = new char[s._capacity + 1];
			strcpy(_str, s._str);
			_capacity = s._capacity;
			_size = s._size;
		}

对于用str1sub1赋值,同样也有浅拷贝的问题,因此,需要按照深拷贝的思想,来人为编写一个运算符重载,具体步骤如下:

假设传递的参数为s,则先开辟一块新的空间tmp,内部内容与参数均和s相同,然后通过delete来释放s,最后,让tmp赋值给s,具体操作如下:

string& operator=(const string& s)
		{
			if (this != &s)
			{
				char* tmp = new char[s._capacity + 1];
				strcpy(tmp, s._str);
				delete[]_str;
				_str = tmp;

				_capacity = s._capacity;
				_size = s._size;
			}
			return *this;
		}

再有了上述两个函数后,对于文章一开始给出的代码就可以正常运行,运行结果如下:

2. 流插入与流提取: 

2.1 流插入:

在上一篇文章中,针对输出string类型的对象时,是通过一个函数来返回对象中存储字符串的指针_str在外部通过cout函数对返回的指针进行打印,即:

const char* c_str() const
		{
			return _str;
		}

上述方式并没有直接通过运算符重载让cout直接对对象进行打印,对于运算符重载的代码如下:

ostream& operator<<(ostream& out, const string& s)
	{
		for (auto ch : s)
		{
			out << ch;
		}
		return out;
	}

对于流插入的运算符重载,原理较为简单,因此不做解释。

通过下面的代码对于流插入的功能进行测试:
 

void test5string()
	{
		string s1("hello world");

		cout << s1 << endl;

测试结果如下:

2.2 流提取:

对于流提取的运算符重载,其代码实现方式如下:

istream& operator>>(istream& in, string& s)
	{
		char ch;
		in >> ch;
		while (ch != ' ' && ch != '\n')
		{
			s += ch;
			in >> ch;
		}

		return in;
	}

通过下面的代码对流提取功能进行测试:

cin

运行代码,此时发现,流插入的过程并不会自动结束,而是无限循环下去:

并且,即使是人为手动输入换行符或者空格,流提取的过程也不会结束。造成上述错误的原因,是因为在函数传参时,两个参数分别是操作符cin,为了区分,在函数中通过引用更名为in。对于cin或者scanf并不能识别或者说获取空格或者换行。例如对于下面的代码:

char ch1, ch2;

		cin >> ch1 >> ch2;
		cout << ch1 << ch2;

(注:这里的cin并不是运算符重载的流提取,而是库中的 cin

运行代码,向两个字符变量中依次插入下面的字符:

输出结果如下:

为了解决此问题,可以使用C语言中的getchar或者C++中的get()来解决。具体操作如下:

istream& operator>>(istream& in, string& s)
	{
		char ch;
		ch = in.get();
		while ( ch !=' '&& ch != '\n')
		{
			s += ch;
			ch = in.get();
		}

		return in;
	}

利用下面的代码来测试流提取的功能:

void test7string()
	{
		string s1;
		cin >> s1;
		cout << s1;
	}

输入内容如下:


 输出如下:

在上面给出的流提取运算符重载中,如果提取到的字符为空或者是换行符,则循环终止。从结果可以看出,当输入1234空格5,这一串字符时,结果只输出了1234,说明通过使用get()的方式成功的拿到了空格。

如果,针对一个string类型的变量连续使用两次流提取,即:

void test7string()
	{
		string s1;
		cin >> s1;
		cout << "第一次流提取:";
		cout << s1<<endl;
		
		cin >> s1;
		cout << endl;
		cout << "第二次流提取:";
		cout << s1;

	}

运行效果如下:

不难发现,针对一个string类型的对象进行两次输入时,第二次的仍然保留了第一次的输入。并不符合cin的功能,为了解决上述问题,需要额外再引进一个函数,在进行流提取之前,清理对象中的内容,即:

void clear()
		{
			_size = 0;
			_str[0] = '\0';

		}

在给定了清理函数clear后,对流提取的运算符重载再进行改进,即:
 

istream& operator>>(istream& in, string& s)
	{
		s.clear();
		char ch;
		ch = in.get();
		while ( ch !=' '&& ch != '\n')
		{
			s += ch;
			ch = in.get();
		}

		return in;
	}

再次运行代码,此时针对一个对象进行两次输入的结果为:

对于流提取,其作用过程通过前面的运算符重载+=来实现,而此运算符重载中需要调用函数push_back这个函数都需要在插入前检查空间是否足够,不足则取调用函数进行扩容。假设,针对一个对象输入长度很长的字符串,即:

针对上述一个长度很长的字符产,需要不断的取调用函数进行扩容。即使人为在流提取之前手动开辟一块空间,空间的大小也不好确定。为了解决上述问题,引入下面的代码解决:
 

istream& operator>>(istream& in, string& s)
	{
		s.clear();
		char tmp[128];
		char ch = in.get();
		int i = 0;
		while ( ch !=' '&& ch != '\n')
		{
			tmp[i++] = ch;
			if (i == 127)
			{
				tmp[i] = '\0';
				s += tmp;
				i = 0;
			}
			ch = in.get();
		}

		if (i > 0)
		{
			tmp[i] = '\0';
			s += tmp;
		}

		return in;
	}

上述代码中,创建了一个能存储128字符的字符数组,在利用get()获取字符后,不会直接向s中进行添加,而是先向tmp插入,当数组的下标等于127时,向数组中最后一个位置插入\0再一次性插入到s中。如果数组中字符的下标最终小于127,则会通过下面的代码进行判断,再一次性插入。这种方法针对于输入很长的内容的情况,减少了调用扩容函数的次数。

3. 深拷贝再优化:

       对于文章开头给出拷贝构造,假设需要利用string类型对象s2s1进行拷贝构造,即:s1(s2)则整体的动作过程为:
       让s1开辟与s2大小相同的空间,然后把s2中的内容拷贝到s1中。

       但是对于拷贝构造,还有一种更为简化的编写方法,即:

string(const string& s)
		{
			string tmp(s._str);
			swap(tmp);
		}

直接通过构造函数,构建一个临时对象tmp,其中tmp中的容量,大小,以及内容均与s相同。再通过交换函数swap来交换this指针指向的对象以及tmp二者的的资源。

而对于上面赋值的深拷贝,其简化的编写方法为:

string& operator=( string s)
		{
			swap(s);
			return *this;
		}

直接通过swap交换函数交换sthis指针指向对象的资源,然后再返回*this即可。

4. 模式实现string的代码:
 

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

namespace violent
{
	class string
	{
	public:
		string(const char* str ="")
		{
			_capacity = strlen(str);
			_size = _capacity;
			_str = new char[_size + 1];

			strcpy(_str, str);
		}

		/*string(const string& s)
		{
			_str = new char[s._capacity + 1];
			strcpy(_str, s._str);
			_capacity = s._capacity;
			_size = s._size;
		}*/

		string(const string& s)
		{
			string tmp(s._str);
			swap(tmp);
		}
		string& operator=( string s)
		{
			swap(s);
			return *this;
		}

		/*string& operator=(const string& s)
		{
			if (this != &s)
			{
				char* tmp = new char[s._capacity + 1];
				strcpy(tmp, s._str);
				delete[]_str;
				_str = tmp;

				_capacity = s._capacity;
				_size = s._size;
			}
			return *this;
		}*/

		const char* c_str() const
		{
			return _str;
		}

		size_t size() const
		{
			return _size;
		}

		typedef char* iterator;
		typedef const char* const_iterator;

		iterator begin()
		{
			return _str;
		}

		const_iterator begin() const
		{
			return _str;
		}

		iterator end()
		{
			return _str + _size;
		}

		const_iterator end() const
		{
			return _str + _size;
		}

		char& operator[](size_t pos)
		{
			assert(pos < _size);
			return _str[pos];
		}

		const char& operator[](size_t pos) const
		{
			assert(pos <= _size);

			return _str[pos];
		}

		void reserve(size_t n)
		{
			if (n > _capacity)
			{
				char* tmp = new char[n + 1];
				strcpy(tmp, _str);
				delete[]_str;
				_str = tmp;
				_capacity = n;
			}
		}

		void push_back(char ch)
		{
			if (_size == _capacity)
			{
				size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;
				reserve(newcapacity);
			}

			_str[_size] = ch;
			_size++;
			_str[_size] = '\0';
		}

		void append(const char* s)
		{
			size_t len = strlen(s);

			if (_size + len > _capacity)
			{
				reserve(_size + len);

			}
			strcpy(_str + _size, s);
			_size += len;
		}

		string& operator+=(char ch)
		{
			push_back(ch);
			return *this;
		}

		string& operator+=(const char* s)
		{
			append(s);
			return *this;
		}
		
		void insert(size_t pos, char ch)
		{
			assert(pos <  _size);
				if(_size == _capacity)
				{
					size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;
					reserve(newcapacity);
				}

				int end = _size;
				while (end >=(int)pos)
				{
					_str[end + 1] = _str[end];
					end--;
				}
				_str[pos] = ch;
				_size++;
		}

		void insert(size_t pos, const char* s)
		{
			assert(pos < _size);
			size_t len = strlen(s);
			
			if (_size + len > _capacity)
			{
				reserve(_size + len);
			}

			int end = _size;
			while (end >= pos + len)
			{
				_str[end + 1] = _str[end];
				end--;
			}
			strncpy(_str + pos, s,len);
			_size += len;

		}

		const static size_t npos = -1;

		void erase(size_t pos, size_t len = npos)
		{
			    assert(pos < _size);
				if (len >= npos || pos + len >= _size)
				{
					_str[pos] = '\0';
					_size = pos;
				}
				else
				{
					strcpy(_str + pos, _str + pos + len);
					_size -= len;
				}
		}

		void swap(string& s1)
		{
			std::swap(_str, s1._str);
			std::swap(_size, s1._size);
		    std::swap(_capacity, s1._capacity);
		}

		size_t find(char ch, size_t pos = 0)
		{
			assert(pos < _size);
			size_t end = _size;

			for (size_t i = pos; i < end; i++)
			{
				if (_str[i] == 'ch')
				{
					return i;
				}
			}
			return npos;
		}

		size_t find(const char* s, size_t pos = 0)
		{
			assert(pos < _size);

			const char* str = strstr(_str + pos, s);
			if (str == nullptr)
			{
				return npos;
			}
			else
			{
				return str - _str;
			}
		}

	

		string substr(size_t pos = 0, size_t len = npos)
		{
			assert(pos < _size);
			size_t end = pos + len;
			if (len == npos || pos + len >= _size)
			{
				end = _size;
			}
			string str;
			str.reserve(end - pos);
			for (size_t i = pos; i < end; i++)
			{
				str += _str[i];
			}
			return str;
		}

		void clear()
		{
			_size = 0;
			_str[0] = '\0';

		}
		~string()
		{
			delete[]_str;
			_str = nullptr;
			_size = 0;
			_capacity = 0;
		}
	private:
		char* _str = nullptr;
		size_t _capacity = 0;
		size_t _size = 0;
	};

	
	void print_str(const string& s)
	{
		for (size_t i = 0; i < s.size(); i++)
		{
			cout << s[i] << ' ';
		}

		string::const_iterator it2 = s.begin();

		while (it2 != s.end())
		{
			cout << *it2 << ' ';
			it2++;
		}

	}

	ostream& operator<<(ostream& out, const string& s)
	{
		for (auto ch : s)
		{
			out << ch;
		}
		return out;
	}

	istream& operator>>(istream& in, string& s)
	{
		s.clear();
		char tmp[128];
		char ch = in.get();
		int i = 0;
		while ( ch !=' '&& ch != '\n')
		{
			tmp[i++] = ch;
			if (i == 127)
			{
				tmp[i] = '\0';
				s += tmp;
				i = 0;
			}
			ch = in.get();
		}

		if (i > 0)
		{
			tmp[i] = '\0';
			s += tmp;
		}

		return in;
	}

	void test1string()
	{
		string s1;
		string s2("hello world");
		cout << s1.c_str() << endl;
		cout << s2.c_str() << endl;
		cout << endl << endl << endl;

		cout << "测试[]的访问功能;";
		for (size_t i = 0; i < s2.size(); i++)
		{
			cout << s2[i] << ' ';
			i++;
		}
		cout << endl;
		cout << "测试[]对于返回值的修改功能:";
		for (size_t i = 0; i < s2.size(); i++)
		{
			s2[i]++;
			cout << s2[i] << ' ';
		}

		cout << endl;
		cout << "测试迭代器访问:";
		string::iterator it1 = s2.begin();
		while (it1 != s2.end())
		{
			cout << *it1 << ' ';
			it1++;
		}

		cout << endl << endl << endl;
		cout << "print_str测试:";
		print_str(s2);
	}

	void test2string()
	{
		string s2("hello world");
		cout << s2.c_str() << endl;

		s2 += 'x';
		cout << s2.c_str() << endl;

		s2 += "yyyyyy";
		cout << s2.c_str() << endl;

		cout << endl << endl;
		s2.insert(0, 'a');
		cout << s2.c_str() << endl;

		s2.insert(3, "wwwww");
		cout << s2.c_str() << endl;
	}

	void test3string()
	{
		string s3("hello world");

		s3.erase(5, 3);
		cout << s3.c_str() << endl;

		string s4("hello world");
		s4.erase(5, 100);
		cout << s4.c_str() << endl;	
	}

	void test4string()
	{
		string s1("hello world");
		cout << s1.c_str() << endl;
		string s2("hahaha");

		string s3(s1);
		cout << s3.c_str() << endl;

		s1 = s2;
		cout << s1.c_str() << endl;
	}

	void test5string()
	{
		string s1("hello world");

		cout << s1 << endl;
		string s2;
		cin >> s1 ;
		cout << s1;
	}

	void test6string()
	{
		string str("https://legacy.cplusplus.com/");
		string sub1, sub2, sub3;
		size_t pos1 = str.find(':');
		sub1 = str.substr(0, pos1 - 0);
		cout << sub1.c_str() << endl;

		size_t pos2 = str.find('/', pos1 + 3);
		sub2 = str.substr(pos1 + 3, pos2 - (pos1 + 3));
		cout << sub2.c_str() << endl;

		sub3 = str.substr(pos2 + 1);
		cout << sub3.c_str() << endl;
	}

	void test7string()
	{
		string s1;
		cin >> s1;
		cout << "第一次流提取:";
		cout << s1<<endl;
		
		cin >> s1;
		cout << endl;
		cout << "第二次流提取:";
		cout << s1;

	}

}

 

#include"string.h"

int main()
{
	violent::test7string();

	return 0;
}

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

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

相关文章

性能优化-OpenCL运行时API介绍

「发表于知乎专栏《移动端算法优化》」 本文首先给出 OpenCL 运行时 API 的整体编程流程图&#xff0c;然后针对每一步介绍使用的运行时 API&#xff0c;讲解 API 参数&#xff0c;并给出编程运行实例。总结运行时 API 使用的注意事项。最后展示基于 OpenCL 的图像转置代码。在…

matlab模型变量一般说明,标定和显示量,以及产生a2l文件,自动填充a2l地址,并使用标定工具ati进行标定(推荐重要)

注意我是用的是matlab2019b 1&#xff0c;输入标定量&#xff0c;使用constant&#xff0c;用cal函数包裹 2&#xff0c;输出显示量&#xff0c;在划线上标注&#xff0c;然后用display函数包裹&#xff0c; 第一步和第二步完成以后&#xff0c;生产标定量a2l 3&#xff0c;输入…

分类预测 | Matlab实现LSTM-Attention-Adaboost基于长短期记忆网络融合注意力机制的Adaboost数据分类预测/故障识别

分类预测 | Matlab实现LSTM-Attention-Adaboost基于长短期记忆网络融合注意力机制的Adaboost数据分类预测/故障识别 目录 分类预测 | Matlab实现LSTM-Attention-Adaboost基于长短期记忆网络融合注意力机制的Adaboost数据分类预测/故障识别分类效果基本描述程序设计参考资料 分类…

2024年【焊工(初级)】考试内容及焊工(初级)新版试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 焊工&#xff08;初级&#xff09;考试内容参考答案及焊工&#xff08;初级&#xff09;考试试题解析是安全生产模拟考试一点通题库老师及焊工&#xff08;初级&#xff09;操作证已考过的学员汇总&#xff0c;相对有…

Centos7 安装redis 详细步骤访问不了github和windows系统下载

windows系统下载 https://hellowindows.cn/ VMware虚拟机安装Windows Server 2016 VL https://blog.csdn.net/qq_37545849/article/details/134828341 VMware全屏时不显示上方命令栏的边缘 此时如果要返回&#xff0c;可以把鼠标移动至屏幕上方边缘短暂停留以呼出命令栏。或使…

MNIST 数据集详析:使用残差网络RESNET识别手写数字(文末送书)

MNIST 数据集已经是一个几乎每个初学者都会接触的数据集, 很多实验、很多模型都会以MNIST 数据集作为训练对象, 不过有些人可能对它还不是很了解, 那么今天我们一起来学习一下MNIST 数据集&#xff0c;同时构建残差网络来识别手写数字。 1.MNIST 介绍 MNIST手写数字数据库具有…

RabbitMQ数据隔离

1、新建用户 2、登录用户&#xff0c;设置虚拟主机 登录用户只能操作自己的虚拟主机&#xff0c;交换机等&#xff0c;不能操作其他人的&#xff01;&#xff01;&#xff01;

【书生·浦语大模型实战营06】《OpenCompass 大模型评测》学习笔记

《OpenCompass 大模型评测》 文档&#xff1a;OpenCompass大模型评测教程 1、主观评测 2、提示词工程 李华每周给2个不同的朋友写一封3页的信&#xff0c;一周写两次。他一年总共写了多少页的信? 李华每周给2个不同的朋友写一封3页的信&#xff0c; 一周写两次。他一年总共…

【数据结构和算法】--- 二叉树(3)--二叉树链式结构的实现(1)

目录 一、二叉树的创建(伪)二、二叉树的遍历2.1 前序遍历2.2 中序遍历2.3 后序遍历 三、二叉树节点个数及高度3.1 二叉树节点个数3.2 二叉树叶子节点个数3.3二叉树第k层节点个数3.4 二叉树查找值为x的节点 四、二叉树的创建(真) 一、二叉树的创建(伪) 在学习二叉树的基本操作前…

OTA 升级软件推荐,附带MD5,CRC16,CRC32,AES算法工具

说明&#xff1a;推荐 OTA 工具软件&#xff0c;可以通过串口按 OTA 协议发送 bin 文件给 MCU,完成 bootloader 升级app 功能 , 这个软件 附带提供 MD5,CRC16,CRC32,AES 算法工具。 文档持续完善中... 1. OTA界面 2.AES.MD5.CRC界面 3.下载链接&#xff1a; 链接: https://p…

逆序对数量(归并排序做法)

先给出上一篇文章归并排序模板-CSDN博客里的归并排序模板&#xff1a; #include<iostream> using namespace std;const int N 100010;int n; int q[N], temp[N];void merge_sort(int q[], int l, int r) {if(l > r) return;int mid (lr) >> 1;merge_sort(q, …

打造专业级ChatGPT风格聊天界面:SpringBoot与Vue实现动态打字机效果,附完整前后端源码

大家好&#xff0c;今天用SpringBoot、vue写了一个仿ChatGPT官网聊天的打字机效果。 所有代码地址:gitee代码地址 &#xff0c;包含前端和后端&#xff0c;可以直接运行 使用本技术实现的项目&#xff1a;aicnn.cn&#xff0c;欢迎大家体验 如果文章知识点有错误的地方&#xf…

HTTPS基本概念

HTTP 与 HTTPS 有哪些区别&#xff1f; HTTP 是超文本传输协议&#xff0c;信息是明文传输&#xff0c;存在安全风险的问题。HTTPS 则解决 HTTP 不安全的缺陷&#xff0c;在 TCP 和 HTTP 网络层之间加入了 SSL/TLS 安全协议&#xff0c;使得报文能够加密传输。HTTP 连接建立相…

VScode 好用的插件合集

VS Code是一个轻量级但功能强大的源代码编辑器&#xff0c;轻量级指的是下载下来的VS Code其实就是一个简单的编辑器&#xff0c;强大指的是支持多种语言的环境插件拓展&#xff0c;也正是因为这种支持插件式安装环境开发让VS Code成为了开发语言工具中的霸主&#xff0c;让其同…

如何保持工业产线业务安全稳定运行?IoT设备敏捷准入方案有诀窍

在工业数字化转型过程中&#xff0c;企业在工业产线和数字化场景&#xff0c;除了面临大量互联网终端&#xff0c;还有传统的工控终端和新加入的物联网&#xff08;IoT&#xff09;设备。在日益严峻的网络攻击面前&#xff0c;工控终端和IoT设备几乎“裸奔”&#xff0c;传统安…

Text Workflow 1.8.2 mac文本转换处理

Text Workflow for mac是一款易于使用但功能强大的应用程序&#xff0c;可将任何文本转换成您需要的格式&#xff0c;以满足您的需求。Text Workflow具有广泛&#xff08;并不断增长&#xff09;的文本转换操作列表&#xff0c;可以实现各种功能。 软件下载&#xff1a;Text Wo…

HarmonyOS 发送http网络请求

好 本文 我们来说 http请求 首先 我们要操作网络内容 需要申请权限 项目中找到 main目录下的module.json5 最下面加上 "requestPermissions": [{"name": "ohos.permission.INTERNET"} ]这里 我在本地写了一个get接口 大家可以想办法 弄一个后…

python Seq2Seq模型源码实战,超详细Encoder-Decoder模型解析实战;早期机器翻译模型源码demo

1.Seq2Seq&#xff08;Encoder-Decoder&#xff09;模型简介 Seq2Seq&#xff08;Encoder-Decoder&#xff09;模型是一种常用于序列到序列&#xff08;sequence-to-sequence&#xff09;任务的深度学习模型。它由两个主要的组件组成&#xff1a;编码器&#xff08;Encoder&am…

H6201L 150V降压芯片 48V 60V 72V 90V 100V 120V降33V 12V 5V 控制器

电动车控制器中的降压芯片也是一种电源管理芯片&#xff0c;用于调整和稳定电动车的电源输出电压。其工作原理与一般降压恒压芯片类似&#xff0c;但在电动车控制器中有一些特定的应用和考虑因素。 降压模式&#xff08;Buck Mode&#xff09;:输入电压通常由电动车的电池提供&…

Qt采集本地摄像头推流成rtsp/rtmp(可网页播放/支持嵌入式linux)

一、功能特点 支持各种本地视频文件和网络视频文件。支持各种网络视频流&#xff0c;网络摄像头&#xff0c;协议包括rtsp、rtmp、http。支持将本地摄像头设备推流&#xff0c;可指定分辨率和帧率等。支持将本地桌面推流&#xff0c;可指定屏幕区域和帧率等。自动启动流媒体服…