[元旦小记]C++ 关于结构体拷贝的问题

本文讨论了一篇关于C++结构体复制的微信文章中提到的代码,强调在多线程环境下,当结构体包含指针尤其是数组指针时可能引发的问题。博主分析了在并发更新时可能出现的线程安全问题和潜在的内存风险,并提醒程序员在处理深拷贝和浅拷贝时应根据逻辑选择合适的方式,以避免潜在的程序错误。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近看到一篇关于结构体的复制的微信推文,里面有一段结构体赋值的代码。博主看了以后有所思考,就记录一下。

原文代码,截图留证【手动害怕】
在这里插入图片描述

对于malloc 和 free的部分,我们暂时不看,只看这么一句

test1.c = test->c;//成员直接赋值,完成拷贝

其实对于作者构建的这个结构体确实没问题,即使放在多线程环境下也没有问题,但如果子结构体里面存在数组指针呢?

假设这么一种场景:
业务程序需要从内存里面拿到某个ID’的内存块(std::map)存储,然后将用户发送过来的部分信息填入到局部结构体里面后,再回写到全局的内存块里面。
这里我们可能为了减少锁的占用,会在拿到内存块的复制以后,立即释放锁,然后需要回写的时候,再次请求锁。

场景:
A线程通过这种办法拿到了全局容器里面的一个结构体的复制,然后释放了锁。
B线程也拿到了这个ID的对应的结构体的一个复制,然后也释放了锁。
A,B都收到了用户发送过来的数据,准备回写内存了(内容不一样,并且修改的内容涉及到指针对应的内存块)
此时会有几种情况发生呢?2种对吧,这里就不写出来了,相信大家都能分析出来。

这里可能还有一种更糟糕的情形:
内存毕竟有限,有增就必须就减,如果在线程会写数据之前,程序的内存调度突然删

拷贝结构体指针,你可以通过以下步骤来完成: 1. 首先,创建一个新的结构体对象,该对象将用于存储拷贝的数据。 2. 使用适当的内存分配函数(如`malloc`)为新结构体对象分配内存空间。 3. 使用`memcpy`函数将原始结构体指针指向的数据拷贝到新的结构体指针指向的内存空间中。 下面是一个示例代码片段,演示了如何拷贝结构体指针: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> typedef struct { int id; char name[20]; } Student; void copyStructPointer(Student* src, Student** dest) { // 为目标指针分配内存空间 *dest = (Student*)malloc(sizeof(Student)); // 拷贝数据 memcpy(*dest, src, sizeof(Student)); } int main() { Student* original = (Student*)malloc(sizeof(Student)); original->id = 1; strcpy(original->name, "John"); Student* copied; copyStructPointer(original, &copied); printf("Original: id = %d, name = %s\n", original->id, original->name); printf("Copied: id = %d, name = %s\n", copied->id, copied->name); free(original); free(copied); return 0; } ``` 在上述示例中,我们定义了一个`Student`结构体,并在`main`函数中创建了一个原始的结构体指针`original`。然后,我们调用`copyStructPointer`函数来拷贝`original`指向的数据,并将拷贝结果存储在`copied`指针中。最后,我们打印出原始和拷贝结构体的数据,并释放内存空间。 注意,拷贝结构体指针实际上是拷贝指针所指向的数据,而不是重新创建一个指向相同数据的新指针。因此,在使用拷贝后的结构体指针时要小心,确保拷贝指针仍然指向有效的内存空间。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值