当前位置: 首页 > news >正文

深入理解C++中string的深浅拷贝

目录

一、引言

二、浅拷贝与深拷贝的基本概念

2.1 浅拷贝

2.2 深拷贝

在C++ 中, string 类的深浅拷贝有着重要的区别。

浅拷贝

深拷贝

 string 类中的其他构造函数及操作

resize 构造

 = 构造(赋值构造)

 + 构造(拼接构造)

 cin 和 cin.get 的区别

三、C++中string类的默认行为

四、自定义字符串类并实现深浅拷贝

4.1 未重载拷贝构造函数和赋值运算符(默认浅拷贝)

4.2 实现深拷贝(重载拷贝构造函数和赋值运算符)

五、比较运算符的重载(> < >= <= == !=)

六、resize函数的实现

七、赋值运算符的深入探讨

7.1 自赋值情况的处理

7.2 移动赋值运算符(C++11及以后)

八、 +  运算符的重载


一、引言

在C++编程中, string  是一个常用的类,用于处理字符串。在涉及对象的复制、传递等操作时,深拷贝和浅拷贝的概念至关重要。理解  string  的深浅拷贝机制,能够帮助我们编写出更健壮、高效且无内存错误的代码。本文将深入探讨  string  的深浅拷贝原理,并结合具体代码示例进行详细分析。

二、浅拷贝与深拷贝的基本概念

2.1 浅拷贝

浅拷贝是指在对象复制时,只复制对象中成员变量的值。如果成员变量是指针类型,浅拷贝仅仅复制指针的值(即地址),这就导致源对象和拷贝对象的指针指向同一块内存区域。当其中一个对象释放该内存时,另一个对象的指针就会变成悬空指针,进而引发程序崩溃或未定义行为。

2.2 深拷贝

深拷贝则是在对象复制时,不仅复制成员变量的值,对于指针类型的成员变量,会重新分配内存空间,并将源对象所指向内存中的数据复制到新分配的内存中。这样源对象和拷贝对象就拥有各自独立的内存,互不影响,避免了悬空指针等问题 。

在C++ 中, string 类的深浅拷贝有着重要的区别。

浅拷贝

浅拷贝是指在对象复制时,只复制对象中指针的值,而不复制指针所指向的内存空间。对于 string 类,如果进行浅拷贝,那么两个 string 对象会共享同一块字符数组内存。例如,当使用默认的拷贝构造函数(如果没有自定义拷贝构造函数)时,可能会发生浅拷贝。

cpp#include <iostream>#include <string>int main() {std::string str1 = "Hello";std::string str2 = str1; // 这里可能是浅拷贝(实际string类在现代实现中通常是深拷贝)std::cout << "str1: " << str1 << ", str2: " << str2 << std::endl;return 0;}

在上述代码中,如果 string 类是浅拷贝实现,那么 str1 和 str2 的内部指针会指向同一块存储 "Hello" 的内存。但实际上,C++ 标准库中的 string 类现代实现通常采用深拷贝。

深拷贝

深拷贝是指在对象复制时,不仅复制对象中指针的值,还会为指针所指向的内存空间分配新的内存,并将原内存中的数据复制到新的内存中。对于 string 类,当进行深拷贝时,会为新的 string 对象分配独立的内存来存储字符串内容。

cpp#include <iostream>#include <string>int main() {std::string str1 = "World";// 自定义深拷贝构造函数(这里假设string类没有默认的深拷贝构造函数)std::string str2(str1.begin(), str1.end()); std::cout << "str1: " << str1 << ", str2: " << str2 << std::endl;return 0;}

在这个例子中,通过 str2 的构造函数,从 str1 的起始位置到结束位置复制字符,实现了深拷贝。这样 str1 和 str2 就拥有了各自独立的字符串内存空间。

 string 类中的其他构造函数及操作

resize 构造

 resize 函数用于改变 string 对象的大小。


 

cpp#include <iostream>#include <string>int main() {std::string str = "abc";str.resize(5, 'd'); // 将字符串大小调整为5,不足的部分用'd'填充std::cout << "Resized string: " << str << std::endl;return 0;}

 = 构造(赋值构造)

= 操作符用于给 string 对象赋值。

传统写法(带自我赋值检查)
 

cppstring& operator=(const string& s)
{if (this!= &s){string tmp(s);std::swap(_str, tmp._str);std::swap(_size, tmp._size);std::swap(_capacity, tmp._capacity);}return *this;
}


 
这段代码首先进行了自我赋值检查,也就是判断当前对象( this  所指向的对象)和要赋值的对象  s  是不是同一个。如果不是,就创建一个临时的  string  对象  tmp  ,并使用  s  来初始化它。然后通过  std::swap  函数,把当前对象和临时对象的字符串指针( _str  )、字符串长度( _size  )、字符串容量( _capacity  )进行交换。这样就巧妙地实现了把  s  的内容赋值给当前对象,同时还避免了一些潜在的内存管理问题,比如释放内存不当等。
 
现代写法(利用移动语义)
 

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


这种写法更加简洁高效,它利用了C++ 的移动语义。这里直接把传入的  string  对象  tmp  作为参数(注意这里没有  const  修饰),在函数内部调用  swap  函数(这里的  swap  函数应该是类内部自定义的交换函数,用于交换当前对象和  tmp  的内部成员)。这种写法的好处在于,当  tmp  是一个临时对象时,能避免不必要的深拷贝操作,提高赋值操作的效率,在处理大量字符串数据时,能显著提升程序性能。

cpp#include <iostream>#include <string>int main() {std::string str1 = "apple";std::string str2;str2 = str1; // 使用赋值构造std::cout << "str1: " << str1 << ", str2: " << str2 << std::endl;return 0;}

 + 构造(拼接构造)

+ 操作符用于拼接两个 string 对象。

cpp#include <iostream>#include <string>int main() {std::string str1 = "Hello";std::string str2 = " World";std::string str3 = str1 + str2; // 拼接两个字符串std::cout << "str3: " << str3 << std::endl;return 0;}

 cin 和 cin.get 的区别

 cin 和 cin.get 都是用于从标准输入读取数据,但它们有一些区别。

-  cin : cin 使用空白字符(空格、制表符、换行符)作为输入数据的分隔符。当使用 cin 读取 string 时,它会在遇到空白字符时停止读取。

cpp#include <iostream>#include <string>int main() {std::string str;std::cout << "Enter a string: ";cin >> str;std::cout << "You entered: " << str << std::endl;return 0;}

如果输入 "Hello World" , cin 只会读取到 "Hello" ,因为它遇到空格就停止了。

-  cin.get : cin.get 可以读取包括空白字符在内的所有字符。它有多种重载形式,常用的一种是读取单个字符,另一种是读取一行字符串。

cpp#include <iostream>#include <string>
using namespace std;int main() {std::string str;std::cout << "Enter a string: ";std::getline(cin, str); // 读取一行字符串,最多读取99个字符(第100个字符留给'\0')std::cout << "You entered: " << str << std::endl;return 0;}
}

在这个例子中, cin.getline 会读取整行输入,包括空格等空白字符,直到遇到换行符或者达到指定的读取字符数上限。

总的来说, cin 适合读取以空白字符分隔的单个数据,而 cin.get 更适合读取包含空白字符的完整字符串或行数据。

三、C++中string类的默认行为

在C++标准库中, string  类已经对拷贝构造函数和赋值运算符进行了重载,实现了深拷贝。下面通过简单的代码示例来验证:

cpp#include <iostream>#include <string>int main() {std::string original = "Hello, World!";// 调用拷贝构造函数std::string copy1(original);// 调用赋值运算符std::string copy2 = original;std::cout << "Original: " << original << std::endl;std::cout << "Copy 1: " << copy1 << std::endl;std::cout << "Copy 2: " << copy2 << std::endl;// 修改其中一个字符串copy1[0] = 'h';std::cout << "After modification:" << std::endl;std::cout << "Original: " << original << std::endl;std::cout << "Copy 1: " << copy1 << std::endl;std::cout << "Copy 2: " << copy2 << std::endl;return 0;}

在上述代码中:

1.  std::string copy1(original);  调用了  string  的拷贝构造函数。

2.  std::string copy2 = original;  调用了  string  的赋值运算符。

运行结果显示,修改  copy1  后, original  和  copy2  不受影响,说明  string  的默认拷贝行为是深拷贝,每个  string  对象都有独立的内存存储字符串内容。

四、自定义字符串类并实现深浅拷贝

为了更深入理解深浅拷贝的原理,我们手动实现一个简单的字符串类  MyString ,并分别展示浅拷贝和深拷贝的实现方式。

4.1 未重载拷贝构造函数和赋值运算符(默认浅拷贝)

cpp#include <iostream>#include <cstring>class MyString {private:char* data;public:MyString(const char* str) {data = new char[strlen(str) + 1];strcpy(data, str);}~MyString() {delete[] data;}// 未重载拷贝构造函数和赋值运算符,默认浅拷贝};int main() {MyString s1("Hello");// 默认浅拷贝,s1和s2的data指针指向同一块内存MyString s2 = s1;return 0;}

在上述代码中, MyString  类没有重载拷贝构造函数和赋值运算符,会使用编译器默认生成的浅拷贝版本。当程序结束时, s1  和  s2  的析构函数会先后调用,导致同一块内存被释放两次,从而引发程序崩溃。

4.2 实现深拷贝(重载拷贝构造函数和赋值运算符)


 

cpp#include <iostream>#include <cstring>class MyString {private:char* data;public:MyString(const char* str) {data = new char[strlen(str) + 1];strcpy(data, str);}// 拷贝构造函数MyString(const MyString& other) {data = new char[strlen(other.data) + 1];strcpy(data, other.data);}// 赋值运算符MyString& operator=(const MyString& other) {if (this != &other) {delete[] data;data = new char[strlen(other.data) + 1];strcpy(data, other.data);}return *this;}~MyString() {delete[] data;}const char* get_data() const {return data;}};int main() {MyString s1("Hello");MyString s2(s1); // 调用拷贝构造函数MyString s3 = s1; // 调用赋值运算符std::cout << "s1: " << s1.get_data() << std::endl;std::cout << "s2: " << s2.get_data() << std::endl;std::cout << "s3: " << s3.get_data() << std::endl;s2 = MyString("World"); // 再次调用赋值运算符std::cout << "After modification:" << std::endl;std::cout << "s1: " << s1.get_data() << std::endl;std::cout << "s2: " << s2.get_data() << std::endl;std::cout << "s3: " << s3.get_data() << std::endl;return 0;}

在上述代码中:

- 拷贝构造函数:为新对象分配独立的内存空间,并将源对象的字符串内容复制到新内存中。

- 赋值运算符:首先检查自赋值情况( if (this != &other) ),避免释放自身内存后无法访问  other  的数据;然后释放当前对象的内存,再分配新内存并复制数据。

运行结果表明,每个  MyString  对象都有独立的字符串数据,修改一个对象不会影响其他对象,实现了深拷贝。

五、比较运算符的重载(> < >= <= == != >>)

为了使自定义的  MyString  类能像标准库  string  类一样进行比较操作,我们需要重载比较运算符。以下是具体的实现代码:

cpp#include <iostream>#include <cstring>class MyString {private:char* data;public:MyString(const char* str) {data = new char[strlen(str) + 1];strcpy(data, str);}MyString(const MyString& other) {data = new char[strlen(other.data) + 1];strcpy(data, other.data);}MyString& operator=(const MyString& other) {if (this != &other) {delete[] data;data = new char[strlen(other.data) + 1];strcpy(data, other.data);}return *this;}~MyString() {delete[] data;}const char* get_data() const {return data;}// 重载大于运算符bool operator>(const MyString& other) const {return strcmp(data, other.data) > 0;}// 重载小于运算符bool operator<(const MyString& other) const {return strcmp(data, other.data) < 0;}// 重载大于等于运算符bool operator>=(const MyString& other) const {return strcmp(data, other.data) >= 0;}// 重载小于等于运算符bool operator<=(const MyString& other) const {return strcmp(data, other.data) <= 0;}// 重载等于运算符bool operator==(const MyString& other) const {return strcmp(data, other.data) == 0;}// 重载不等于运算符bool operator!=(const MyString& other) const {return strcmp(data, other.data) != 0;}};int main() {MyString s1("apple");MyString s2("banana");std::cout << "s1 > s2: " << (s1 > s2 ? "true" : "false") << std::endl;std::cout << "s1 < s2: " << (s1 < s2 ? "true" : "false") << std::endl;std::cout << "s1 >= s2: " << (s1 >= s2 ? "true" : "false") << std::endl;std::cout << "s1 <= s2: " << (s1 <= s2 ? "true" : "false") << std::endl;std::cout << "s1 == s2: " << (s1 == s2 ? "true" : "false") << std::endl;std::cout << "s1 != s2: " << (s1 != s2 ? "true" : "false") << std::endl;return 0;}

在上述代码中,我们通过调用  strcmp  函数来实现各个比较运算符的逻辑。 strcmp  函数会按照字典序比较两个字符串,并返回相应的值,根据这个返回值来确定比较结果。

but strcmp遇到'\0'会终止,需要大家结合实际用例判断是否修改他

六、resize函数的实现

 resize  函数用于改变字符串的长度,以下是  MyString  类中  resize  函数的实现:

cpp#include <iostream>#include <cstring>class MyString {private:char* data;public:MyString(const char* str) {data = new char[strlen(str) + 1];strcpy(data, str);}MyString(const MyString& other) {data = new char[strlen(other.data) + 1];strcpy(data, other.data);}MyString& operator=(const MyString& other) {if (this != &other) {delete[] data;data = new char[strlen(other.data) + 1];strcpy(data, other.data);}return *this;}~MyString() {delete[] data;}const char* get_data() const {return data;}void resize(size_t new_size, char fill_char = '') {char* new_data = new char[new_size + 1];size_t len = strlen(data);if (new_size < len) {strncpy(new_data, data, new_size);new_data[new_size] = '\0';} else {strcpy(new_data, data);for (size_t i = len; i < new_size; ++i) {new_data[i] = fill_char;}new_data[new_size] = '\0';}delete[] data;data = new_data;}};int main() {MyString s("Hello");std::cout << "Original: " << s.get_data() << std::endl;s.resize(3);std::cout << "After resize to 3: " << s.get_data() << std::endl;s.resize(8, 'x');std::cout << "After resize to 8 with fill char 'x': " << s.get_data() << std::endl;return 0;}

在上述  resize  函数中:

1. 首先分配新的内存空间来存储调整长度后的字符串。

2. 根据新的长度  new_size  与原字符串长度  len  的关系进行处理:

- 如果  new_size  小于  len ,则将原字符串的前  new_size  个字符复制到新字符串中,并添加字符串结束符  '\0' 。

- 如果  new_size  大于  len ,先复制原字符串,然后用填充字符  fill_char  填充剩余的位置,最后添加字符串结束符  '\0' 。

3. 释放原字符串的内存,并将  data  指针指向新分配的内存。

七、赋值运算符的深入探讨

前面已经实现了基本的赋值运算符  operator=(const MyString& other) ,它用于处理对象之间的赋值操作,实现深拷贝。但在实际应用中,还有一些特殊情况和性能优化的考虑。

7.1 自赋值情况的处理

在之前的赋值运算符实现中,我们通过  if (this != &other)  来检查自赋值情况。这是非常必要的,因为如果不检查自赋值,在释放  this  对象的  data  内存后,就无法再访问  other  对象的  data  了(因为它们已经是同一个对象),从而导致错误。

例如,以下代码展示了如果不处理自赋值情况会发生什么:

cppMyString s("Test");s = s; // 自赋值

如果赋值运算符没有自赋值检查,就会先释放  s  的  data  内存,然后尝试从已经释放的内存中复制数据,这会导致未定义行为。

7.2 移动赋值运算符(C++11及以后)

C++11引入了移动语义,为了提高性能,我们可以实现移动赋值运算符  operator=(MyString&& other) 。移动赋值运算符用于处理右值(临时对象)的赋值操作,它可以直接转移资源(如  data  指针)的所有权,而不需要进行深拷贝,从而避免不必要的内存分配和复制操作。

以下是移动赋值运算符的实现代码:


 

cppMyString& operator=(MyString&& other) noexcept {if (this != &other) {delete[] data;data = other.data;other.data = nullptr;}return *this;}

在上述代码中:

1. 同样先检查自赋值情况。

2. 释放  this  对象原来的  data  内存。

3. 将  other  对象的  data  指针赋值给  this  对象的  data ,并将  other  对象的  data  指针置为  nullptr ,这样就完成了资源的转移。

4. 最后返回  *this ,以支持链式赋值。

通过实现移动赋值运算符,当我们对右值进行赋值操作时,就可以提高程序的性能。例如:

cppMyString s1("Hello");MyString s2("World");s2 = std::move(s1); // 调用移动赋值运算符

这里  std::move  函数将  s1  转换为右值,从而触发移动赋值运算符,直接转移  s1  的资源给  s2 ,而不是进行深拷贝。

八、 +  运算符的重载

为了让自定义的  MyString  类能够像标准库中的  string  类一样进行字符串连接操作,我们需要重载  +  运算符。 +  运算符的重载有两种常见形式:一种是将两个  MyString  对象连接,另一种是将  MyString  对象和一个 C 风格字符串( const char* )连接。

以下是实现代码:

cpp#include <iostream>#include <cstring>class MyString {private:char* data;public:MyString(const char* str = "") {data = new char[strlen(str) + 1];strcpy(data, str);}MyString(const MyString& other) {data = new char[strlen(other.data) + 1];strcpy(data, other.data);}MyString& operator=(const MyString& other) {if (this != &other) {delete[] data;data = new char[strlen(other.data) + 1];strcpy(data, other.data);}return *this;}MyString& operator=(MyString&& other) noexcept {if (this != &other) {delete[] data;data = other.data;other.data = nullptr;}return *this;}~MyString() {delete[] data;}const char* get_data() const {return data;}// 重载大于运算符bool operator>(const MyString& other) const {return strcmp(data, other.data) > 0;}// 重载小于运算符bool operator<(const MyString& other) const {return strcmp(data, other.data) < 0;}// 重载大于等于运算符

相关文章:

深入理解C++中string的深浅拷贝

目录 一、引言 二、浅拷贝与深拷贝的基本概念 2.1 浅拷贝 2.2 深拷贝 在C 中&#xff0c; string 类的深浅拷贝有着重要的区别。 浅拷贝 深拷贝 string 类中的其他构造函数及操作 resize 构造 构造&#xff08;赋值构造&#xff09; 构造&#xff08;拼接构造&#xf…...

C++ 常用的智能指针

C 智能指针 一、智能指针类型概览 C 标准库提供以下智能指针&#xff08;需包含头文件 <memory>&#xff09;&#xff1a; unique_ptr&#xff1a;独占所有权&#xff0c;不可复制&#xff0c; 可移动shared_ptr&#xff1a;共享所有权&#xff0c;用于引用计数weak_pt…...

【AI部署】腾讯云GPU-常见故障—SadTalker的AI数字人视频—未来之窗超算中心 tb-lightly

ERROR: Could not find a version that satisfies the requirement tb-nightly (from torchreid) (from versions: none) ERROR: No matching distribution found for tb-nightly 解决 阿里云 python -m pip install tb-nightly -i https://mirrors.aliyun.com/pypi/simple …...

三大等待和三大切换

三大等待 1、三大等待&#xff1a;等待的方式有三种&#xff1a;强制等待&#xff0c;隐性等待&#xff0c;显性等待。 1、强制等待&#xff1a;time.sleep(2)&#xff0c;秒 优点&#xff1a;使用简单缺点&#xff1a;等待时间把握不准&#xff0c;容易造成时间浪费或者等待时…...

工程化实践:Flutter项目结构与规范

工程化实践&#xff1a;Flutter项目结构与规范 在Flutter项目开发中&#xff0c;良好的工程化实践对于提高开发效率、保证代码质量和团队协作至关重要。本文将从项目结构、代码规范、CI/CD流程搭建以及包管理等方面&#xff0c;详细介绍Flutter项目的工程化最佳实践。 项目结…...

数据结构-Map和Set

文章目录 1. 搜索树2. Map3. Set4. 哈希表4.1 哈希表的基本概念4.2 哈希表的实现方法4.3 Java中的哈希表实现 5. 哈希桶哈希桶的实现方式哈希桶的作用哈希桶的应用模拟实现 1. 搜索树 二叉搜索树&#xff08;Binary Search Tree, BST&#xff09;是一种特殊的二叉树&#xff0…...

cpolar 内网穿透 实现公网可以访问本机

1、登录网站&#xff0c;升级成专业版&#xff0c;测试的话建议选一个月付费&#xff0c;选择预留 2、保留的TCP地址增加一条记录&#xff0c;描述可以自己取 3、验证&#xff0c;生成一个Authtocken码 4、在安装目录下&#xff0c;打开CMD命令&#xff0c;复制上面的码运行aut…...

QT调用ffmpeg库实现视频录制

可以通过QProcess调用ffmpeg命令行,也可以直接调用ffmpeg库,方便。 调用库 安装ffmpeg ffmpeg -version 没装就装 sudo apt-get update sudo apt-get install ffmpeg sudo apt-get install ffmpeg libavdevice-dev .pro引入库路径,引入库 LIBS += -L/usr/lib/aarch64-l…...

AI专题(一)----NLP2SQL探索以及解决方案

前面写了很多编码、算法、底层计算机原理等相关的技术专题&#xff0c;由于工作方向调整的缘故&#xff0c;今天开始切入AI人工智能相关介绍。本来按照规划&#xff0c;应该先从大模型的原理开始介绍会比较合适&#xff0c;但是计划赶不上变化&#xff0c;前面通用大模型的工作…...

Redis 的指令执行方式:Pipeline、事务与 Lua 脚本的对比

Pipeline 客户端将多条命令打包发送&#xff0c;服务器顺序执行并一次性返回所有结果。可以减少网络往返延迟&#xff08;RTT&#xff09;以提升吞吐量。 需要注意的是&#xff0c;Pipeline 中的命令按顺序执行&#xff0c;但中间可能被其他客户端的命令打断。 典型场景&…...

群辉默认docker数据存储路径

做一下笔记 今天不小心路径规划错误&#xff0c;好不容易找到了数据&#xff0c;特此做个路径记录。 /var/packages/ContainerManager/var/docker/...

【C++】入门基础【上】

目录 一、C的发展历史二、C学习书籍推荐三、C的第一个程序1、命名空间namespace2、命名空间的使用3、头文件<iostream>是干什么的&#xff1f; 个人主页<—请点击 C专栏<—请点击 一、C的发展历史 C的起源可以追溯到1979年&#xff0c;当时Bjarne Stroustrup(本…...

Git LFS 学习笔记:原理、配置、实践与心路历程

最近在学习 Git LFS&#xff0c;把一些零散的笔记整理成一篇博文&#xff0c;记录我的学习思路与心路历程。以下内容均为个人理解总结&#xff0c;部分尚未在生产项目中验证&#xff0c;仅供回顾与参考。 &#x1f50d; Git LFS 是什么&#xff1f;原理是什么&#xff1f; 刚接…...

SpringBoot集成oshi 查询系统数据

实现功能&#xff1a; ​​​​​​​ <!-- 获取系统信息 --><dependency><groupId>com.github.oshi</groupId><artifactId>oshi-core</artifactId><version>6.6.1</version></dependency><dependency><groupI…...

iOS Facebook 登录

iOS Facebook 登录 官方文档 SDK下载链接...

uniapp打包IOS私钥证书过期了,如何在非mac系统操作

在非Mac系统下解决uniapp打包iOS私钥证书过期的问题&#xff0c;需通过以下步骤实现&#xff1a; --- ### **一、重新生成iOS证书&#xff08;非Mac环境操作&#xff09;** 1. **生成私钥和CSR文件** 使用OpenSSL工具&#xff08;需提前安装&#xff09;生成私钥和证书签…...

Axios的使用

Axios 是一个基于 Promise 的现代化 HTTP 客户端库&#xff0c;专为浏览器和 Node.js 设计。在企业级应用中&#xff0c;它凭借以下核心优势成为首选方案&#xff1a; 一、Axios 的核心优势 特性说明Promise 支持天然支持异步编程&#xff0c;避免回调地狱拦截器机制可全局拦截…...

第八篇:系统分析师第三遍——3、4章

目录 一、目标二、计划三、完成情况四、意外之喜(最少2点)1.计划内的明确认知和思想的提升标志2.计划外的具体事情提升内容和标志 五、总结 一、目标 通过参加考试&#xff0c;训练学习能力&#xff0c;而非单纯以拿证为目的。 1.在复习过程中&#xff0c;训练快速阅读能力、掌…...

【2025-泛计算机类-保研/考研经验帖征集】

【2025-泛计算机类-保研/考研经验帖征集】 打扰您1分钟时间看下这里&#xff1a; 这是一个无偿为爱发电的项目&#xff0c;旨在收集湖南大学2025届毕业的计算机类学科同学的经验帖&#xff0c; 我将定期汇总链接&#xff0c;在校内推免群中宣传&#xff0c;为校内的学弟学妹们…...

Flink介绍——实时计算核心论文之Kafka论文详解

引入 我们通过S4和Storm论文的以下文章&#xff0c;已经对S4和Storm有了不错的认识&#xff1a; S4论文详解S4论文总结Storm论文详解Storm论文总结 不过&#xff0c;在讲解这两篇论文的时候&#xff0c;我们其实没有去搞清楚对应的流式数据是从哪里来的。虽然S4里有Keyless …...

细节:如何制作高质量的VR全景图

细节&#xff1a;如何制作高质量的VR全景图 VR全景图是通过虚拟现实和3D技术实现的全景展示方式&#xff0c;能够将实景以1:1的比例等比复刻&#xff0c;并还原到互联网上&#xff0c;使用户能够在线上游览世界&#xff0c;获得沉浸式的体验。制作高质量的VR全景图是一个复杂而…...

深度学习中的概念——元素积(哈达玛积)

元素积操作&#xff08;哈达玛积&#xff09; &#x1f522; 基本定义 矩阵的哈达玛积 对于两个同维度的矩阵&#xff1a; A [ a i j ] , B [ b i j ] A [a_{ij}], \quad B [b_{ij}] A[aij​],B[bij​] 它们的哈达玛积定义为&#xff1a; C A ∘ B 其中 c i j a i j…...

探索 Flowable 后端表达式:简化流程自动化

什么是后端表达式&#xff1f; 在 Flowable 中&#xff0c;后端表达式是一种强大的工具&#xff0c;用于在流程、案例或决策表执行期间动态获取或设置变量。它还能实现自定义逻辑&#xff0c;或将复杂逻辑委托…… 后端表达式在 Flowable 的后端运行&#xff0c;无法访问前端…...

AI语音助手 React 组件使用js-audio-recorder实现,将获取到的语音转成base64发送给后端,后端接口返回文本内容

页面效果&#xff1a; js代码&#xff1a; import React, { useState, useRef, useEffect } from react; import { Layout, List, Input, Button, Avatar, Space, Typography, message } from antd; import { SendOutlined, UserOutlined, RobotOutlined, AudioOutlined, Stop…...

《软件设计师》复习笔记(11.6)——系统转换、系统维护、系统评价

目录 一、遗留系统&#xff08;Legacy System&#xff09; 定义&#xff1a; 特点&#xff1a; 演化策略&#xff08;基于价值与技术评估&#xff09;&#xff1a; 高水平 - 低价值&#xff1a; 高水平 - 高价值&#xff1a; 低水平 - 低价值&#xff1a; 低水平 - 高价…...

学习threejs,使用EffectComposer后期处理组合器(采用RenderPass、GlitchPass渲染通道)

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言1.1 ☘️THREE.EffectComposer 后期…...

Yarn的定义?

YARN&#xff08;Yet Another Resource Negotiator&#xff09; 是 Apache Hadoop 的核心组件之一&#xff0c;负责集群的资源管理和任务调度。它的主要作用是将 Hadoop 的资源管理和作业调度/监控功能分离&#xff0c;形成一个通用的资源管理平台&#xff0c;可以支持多种计算…...

职坐标IT培训热门技术实战精讲

在数字化转型浪潮中&#xff0c;人工智能、大数据与云原生已成为驱动产业升级的核心引擎。职坐标IT培训课程以实战导向为基石&#xff0c;聚焦高薪岗位核心技术栈&#xff0c;通过拆解企业级项目案例&#xff0c;将复杂的技术理论转化为可落地的工程实践。课程模块涵盖从架构设…...

前端:uniapp框架中<scroll-view>r如何控制元素进行局部滚动

以下是使用 <scroll-view> 实现局部滚动的完整示例&#xff0c;包含动态内容、滚动控制和滚动位置监听&#xff1a; 一、基础局部滚动示例 <template><view class"container"><!-- 固定高度的滚动容器 --><scroll-view scroll-y :scroll…...

【KWDB 创作者计划】_算法篇---Stockwell变换

文章目录 前言一、Stockwell变换原理详解1.1 连续S变换定义1.2 离散S变换1.3简介 二、S变换的核心特点2.1频率自适应的时频分辨率2.1.1高频区域2.1.2低频区域 2.2无交叉项干扰2.3完全可逆2.4相位保持2.5与傅里叶谱的直接关系 三、应用领域3.1地震信号分析3.2生物医学信号处理3.…...

桌面级OTA测试解决方案:赋能智能网联汽车高效升级

一、前言 随着智能网联汽车的快速发展&#xff0c;OTA&#xff08;Over-The-Air&#xff09;技术已成为汽车软件更新和功能迭代的关键手段。为确保OTA升级的可靠性、安全性和效率&#xff0c;构建一套高效、便捷的桌面级OTA测试解决方案至关重要。 本方案基于Vector先进的软硬…...

京东物流基于Flink StarRocks的湖仓建设实践

摘要&#xff1a;本文整理自京东物流高级数据开发工程师梁宝彬先生在 Flink Forward Asia 2024 论坛中的分享。内容主要为以下四个部分&#xff1a; 1、实时湖仓探索与建设 2、实时湖仓应用 3、问题与思考 4、未来展望 今天&#xff0c;将分享的主题大纲包括&#xff1a;首先&a…...

【Test Test】灰度化和二值化处理图像

文章目录 1. 图像灰度化处理对比2. 代码示例3. 二值化处理 1. 图像灰度化处理对比 2. 代码示例 #include <opencv2/opencv.hpp> using namespace cv;int main() {Mat currentImage imread("path_to_image.jpg"); // 读取彩色图像Mat grayImage;// 将彩色图像…...

Docker快速入门

1 基本信息 1.1 基础概览 &#xff08;1&#xff09;容器与镜像 容器‌&#xff1a;轻量级的独立运行时环境&#xff0c;基于Linux的命名空间&#xff08;namespace&#xff09;和控制组&#xff08;cgroup&#xff09;技术实现资源隔离‌。容器通过镜像创建&#xff0c;每个容…...

AI 编程工具—如何在 Cursor 中集成使用 MCP工具

AI 编程工具—如何在 Cursor 中集成使用 MCP工具 这里我们给出了常用的MCP 聚合工具,也就是我们可以在这些网站找MCP服务 这是一个MCP Server共享平台,用户可以在上面发布和下载MCP Server配置。在这里可以选择你需要的MCP 服务。 如果你不知道你的mcp 对应的server 名称也不…...

航电系统通信与数据链技术分析

一、技术要点 1. 高带宽与低时延通信技术 航电系统需支持实时数据传输&#xff0c;如民航VDL2数据链采用D8PSK调制和Reed Solomon编码技术&#xff0c;传输速率达31.5Kb/s&#xff0c;并通过物理层优化减少码间串扰。新一代LDACS系统则利用L频段频谱&#xff0c;提供比传统VD…...

windwos脚本 | 基于scrcpy,只投声音、只投画面

安装scrcpy&#xff0c;scrcpy自带adb 写脚本命名为 .bat 结尾 注意这里的set "PATHD:\tools\scrcpy-win64-v3.2;%PATH%" 替换成scrcpy的安装目录 echo off :: 设置UTF-8编码 chcp 65001 > nul :: 设置标题 title 手机投屏工具:: 添加 scrcpy 路径到 PATH set &q…...

产品经理面试题与参考答案资料(2025年版)

一、技术背景与工具能力 问题1: 软件开发生命周期(SDLC)包括哪些阶段?作为产品经理,你如何参与每个阶段? 参考答案: 软件开发生命周期通常划分为需求分析、设计、开发、测试、部署和维护六个阶段。作为产品经理,你需要在...

HTML 初识

段落标签 <p><!-- 段落标签 -->Lorem ipsum dolor sit amet consectetur adipisicing elit. Fugiat, voluptate iure. Obcaecati explicabo sint ipsum impedit! Dolorum omnis voluptas sint unde sed, ipsa molestiae quo sapiente quos et ad reprehenderit.&l…...

CSS3笔记

文章目录 1.CSS的概念2.CSS的引入方式3.选择器3.1基础选择器3.1.1标签选择器3.1.2类选择器3.1.3id选择器3.1.4通配符选择器 4.画盒子5.文字控制属性5.1 字体大小-font-size5.2 字体粗细-font-weight5.3 字体倾斜-font-style5.4 行高-line-height5.5 字体族-font-family5.6 字体…...

yarn的定义

YARN是一个分布式资源管理平台&#xff0c;它通过分离资源管理和作业调度的功能&#xff0c;为Hadoop集群提供了一种高效、灵活且可扩展的方式来管理和分配计算资源。YARN的核心目标是将资源管理与作业调度/监控的功能分离&#xff0c;从而让Hadoop能够支持更多种类的计算框架&…...

CSS 美化页面(五)

一、position属性 属性值‌‌描述‌‌应用场景‌static默认定位方式&#xff0c;元素遵循文档流正常排列&#xff0c;top/right/bottom/left 属性无效‌。普通文档流布局&#xff0c;默认布局&#xff0c;无需特殊定位。relative相对定位&#xff0c;相对于元素原本位置进行偏…...

互联网大厂Java面试:微服务与分布式系统挑战

互联网大厂Java面试&#xff1a;微服务与分布式系统挑战 在互联网的大潮中&#xff0c;无数程序员怀揣着梦想&#xff0c;希望能在一线大厂找到自己的位置。今天的故事主角是马飞机&#xff0c;一位充满幽默感但技术略显水货的程序员。他来到了一家知名互联网公司参加Java开发…...

数据结构——顺序表(C语言实现)

1.顺序表的概述 1.1 顺序表的概念及结构 在了解顺序表之前&#xff0c;我们要先知道线性表的概念&#xff0c;线性表&#xff0c;顾名思义&#xff0c;就是一个线性的且具有n个相同类型的数据元素的有限序列&#xff0c;常见的线性表有顺序表、链表、栈、队列、字符串等等。线…...

FastGPT安装前,系统环境准备工作?

1.启用适用于 Linux 的 Windows 子系统 方法一&#xff1a;打开控制面板 -> 程序 -> 启用或关闭Windows功能->勾选 “适用于Linux的Vindows子系统” 方法二&#xff1a;以管理员身份打开 PowerShell&#xff08;“开始”菜单 >“PowerShell” >单击右键 >“…...

【2】CICD持续集成-k8s集群中安装Jenkins

一、背景&#xff1a; Jenkins是一款开源 CI&CD 系统&#xff0c;用于自动化各种任务&#xff0c;包括构建、测试和部署。 Jenkins官方提供了镜像&#xff1a;https://hub.docker.com/r/jenkins/jenkins 使用Deployment来部署这个镜像&#xff0c;会暴露两个端口&#xff…...

相比其他缓存/内存数据库(如 Memcached, Ehcache 等),Redis 在微服务环境中的优势和劣势是什么?

我们来比较一下 Redis 与 Memcached、Hazelcast、Ehcache 等在微服务环境下的优势和劣势。 Redis 的优势 : 丰富的数据结构 (Rich Data Structures): 优势: 这是 Redis 最显著的优势之一。除了简单的 Key-Value (字符串) 外&#xff0c;Redis 还原生支持 Lists, Sets, Sorted …...

Day53 二叉树的层序遍历

给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历 。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* T…...

mac上安装VMWare Fusion安装ubuntu系统问题

mac不能复制粘贴到虚拟机的ubuntu系统里&#xff0c;没有下载vmtools 在ubuntu系统执行命令 sudo apt update sudo apt install open-vm-tools open-vm-tools-desktop -y ubuntu 下载地址 https://cdimage.ubuntu.com/ubuntu/releases/20.04/release/...

JAVA Web_定义Servlet_处理POST请求【练习】

题目 有一个登录页面&#xff08;login.html&#xff09;&#xff0c;其登录表单的HTML代码如下&#xff1a; </form action"doLogin" method "post"> 用户名&#xff1a;<input type"text" name"userName"><br>…...