成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C語言結(jié)構(gòu)體成員賦值的深拷貝和淺拷貝

開發(fā) 后端
最近在做通信協(xié)議的解析處理、傳遞分析等問題,總是遇到通信幀中的結(jié)構(gòu)體拷貝等問題,遇到了一些坑,也是比較基礎(chǔ)但是易錯的C語言知識,一起來探究一下結(jié)構(gòu)體的深拷貝和淺拷貝。

淺拷貝

C語言中的淺拷貝是指在拷貝過程中,對于指針型成員變量只拷貝指針本身,而不拷貝指針?biāo)赶虻哪繕?biāo),它按字節(jié)復(fù)制的。我們分幾種情況舉例子來看一下。

結(jié)構(gòu)體中不存在指針成員變量時

代碼如下:

//在win10_64位+vs2017
//來源:技術(shù)讓夢想更偉大
//作者:李肖遙
#include <stdio.h>
typedef struct {
char name[64];
int age;
}Member;
int main(){
Member stu1 = { "LiMing", 18 };
Member stu2;
stu2 = stu1;
printf("%s,%d\n", stu2.name, stu2.age);
system("pause");
return 0;
}

運(yùn)行如下:

結(jié)構(gòu)體中存在指針成員變量時

代碼如下:

//在win10_64位+vs2017
//來源:技術(shù)讓夢想更偉大
//作者:李肖遙
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct {
char *name;
int age;
}Member;
int main() {
Member Member1, Member2;
Member1.name = malloc(sizeof(char) * 64);
if (NULL == Member1.name)
{
printf("malloc failed\n");
}
memset(Member1.name, 0, 64);
//strcpy(Member1.name, "LiMing");
snprintf(Member1.name, 64, "LiMing");
Member1.age = 18;
Member2 = Member1;/*拷貝*/
snprintf(Member2.name, 64, "LiXiaoYao");
Member2.age = 29;
printf("%s, %d\n", Member1.name, Member1.age);
if (NULL != Member1.name) {
free(Member1.name);
Member1.name = NULL;
}
system("pause");
return 0;
}

運(yùn)行如下:

從中我們看到,改變Member2的值,Member1的值也改變了,這說明一片空間被兩個不同的子對象共享了,改變一個對象的值另外一個也會隨之改變。

我們改變Member2寫法,申請內(nèi)存的代碼如下:

//在win10_64位+vs2017
//來源:技術(shù)讓夢想更偉大
//作者:李肖遙
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct {
char *name;
int age;
}Member;
int main() {
Member Member1;
Member1.name = malloc(sizeof(char) * 64);
if (NULL == Member1.name)
{
printf("malloc failed\n");
}
memset(Member1.name, 0, 64);
//strcpy(Member1.name, "LiMing");
snprintf(Member1.name, 64, "LiMing");
Member1.age = 18;
Member Member2;
Member2.name = malloc(sizeof(char) * 64);
if (NULL == Member2.name)
{
printf("malloc failed\n");
}
memset(Member2.name, 0, 64);
//strcpy(Member2.name, "LiMing");
snprintf(Member2.name, 64, "LiXiaoYao");
Member2.age = 29;
Member1 = Member2;
printf("%s, %d\n", Member2.name, Member2.age
if (NULL != Member1.name) {
free(Member1.name);
Member1.name = NULL;
}
if (NULL != Member2.name) {
free(Member2.name);
Member2.name = NULL;
}
system("pause");
return 0;
}

運(yùn)行如下:

從中我們看到,當(dāng)數(shù)據(jù)成員中有指針時,兩個類中的兩個指針將指向同一個地址,當(dāng)對象快結(jié)束時,會調(diào)用兩次free函數(shù),此時Member2已經(jīng)是野指針(圖中有X的錯誤標(biāo)志),這個野指針指向的內(nèi)存空間已經(jīng)被釋放掉,再次釋放會報異常錯誤,要解決這個問題就要涉及到深拷貝了。

深拷貝

深拷貝除了拷貝其成員本身的值之外,還拷貝成員指向的動態(tài)內(nèi)存區(qū)域內(nèi)容,深拷貝會在堆內(nèi)存中另外申請空間來儲存數(shù)據(jù)。

解決的思路是在釋放掉被賦值指針變量的舊指向內(nèi)存時,重新對其開辟新內(nèi)存,這種情況下兩個結(jié)構(gòu)體中指針地址不同,但是指向的內(nèi)容是一致的。代碼如下:

//在win10_64位+vs2017
//來源:技術(shù)讓夢想更偉大
//作者:李肖遙
#include <stdio.h>
#include <stdlib.h>
typedef struct {
char *name;
int age;
}Member;
int main() {
Member Member1;
Member1.name = malloc(sizeof(char) * 64);
if (NULL == Member1.name)
{
printf("malloc failed\n");
}
memset(Member1.name, 0, 64);
//strcpy(Member1.name, "LiMing");
snprintf(Member1.name, 64, "LiMing");
Member1.age = 18;
Member Member2;
Member2.name = malloc(sizeof(char) * 64);
if (NULL == Member2.name)
{
printf("malloc failed\n");
}
memset(Member2.name, 0, 64);
//strcpy(Member2.name, "LiMing");
snprintf(Member2.name, 64, "LiXiaoYao");
Member2.age = 29;
if (Member1.name != NULL) {
free(Member1.name);
Member1.name = NULL;
}
Member1.name = malloc(strlen(Member2.name) + 1);
strcpy(Member1.name, Member2.name);
printf("%s, %d\n", Member1.name, Member1.age);
if (NULL != Member1.name) {
free(Member1.name);
Member1.name = NULL;
}
if (NULL != Member2.name) {
free(Member2.name);
Member2.name = NULL;
}
system("pause");
return 0;
}

運(yùn)行如下:

結(jié)論

使用C語言來說,深拷貝淺拷貝的概念我們不需要深究,在進(jìn)行結(jié)構(gòu)體拷貝的時候,結(jié)構(gòu)體成員是非指針的話,那么直接賦值是沒有任何問題的,建議使用這種方式,避免淺拷貝這類不易發(fā)現(xiàn)的錯誤產(chǎn)生。

如果成員有指針類型,我們就需要重寫拷貝函數(shù),自己定義拷貝行為了,這一點(diǎn)我們需要尤為注意。

責(zé)任編輯:龐桂玉 來源: C語言與C++編程
相關(guān)推薦

2021-07-16 12:33:24

Javascript深拷貝淺拷貝

2017-08-16 13:30:05

Java深拷貝淺拷貝

2021-10-18 09:01:01

前端賦值淺拷貝

2021-09-27 11:07:11

深拷貝淺拷貝內(nèi)存

2022-07-26 08:07:03

Python淺拷貝深拷貝

2018-09-26 14:37:17

JavaScript前端編程語言

2021-06-28 07:12:28

賦值淺拷貝深拷貝

2009-05-19 17:28:44

深拷貝淺拷貝clone()

2020-08-03 08:24:26

原型模式拷貝

2021-01-08 06:15:09

深拷貝淺拷貝寫時拷貝

2020-10-12 08:35:22

JavaScript

2023-05-17 08:42:46

深拷貝Golang

2018-05-10 14:20:18

前端JavaScript深拷貝

2024-02-05 22:56:16

C++拷貝開發(fā)

2024-03-15 15:03:23

2024-04-17 09:01:08

Python深拷貝淺拷貝

2023-05-17 07:36:00

淺拷貝深拷貝對象

2020-06-23 08:41:47

JavaScript開發(fā)技術(shù)

2023-09-22 12:21:33

Python深拷貝淺拷貝

2019-02-25 08:58:16

Python深拷貝淺拷貝
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 国产精品中文在线 | 免费黄色日本 | 美女一级毛片 | 午夜性色a√在线视频观看9 | 久久亚洲春色中文字幕久久久 | 亚洲国产精品视频一区 | 精品久久久久久久人人人人传媒 | 中文字幕免费在线 | 中国一级毛片免费 | 日韩三 | 亚洲视频中文字幕 | 九九热视频这里只有精品 | 国产精品国产三级国产aⅴ无密码 | 亚洲午夜精品 | 欧美日韩免费 | 在线国产一区二区三区 | 亚洲欧洲日韩精品 中文字幕 | 亚洲精品乱码久久久久久蜜桃 | 亚洲欧美在线观看 | 男女视频在线观看 | 国产操操操 | 久久精品视频9 | 91久色 | 亚洲一区电影 | 日韩视频在线观看 | 免费成人在线网站 | 久久精品久久精品久久精品 | 久久成人高清视频 | 欧美精品在线一区二区三区 | 色免费在线视频 | 国产精品亚洲精品久久 | 一区亚洲 | 欧美精品第一区 | 在线免费观看成年人视频 | 91成人免费看片 | 中文字幕一区二区三区在线观看 | 91精品国产综合久久久密闭 | 美女中文字幕视频 | 午夜影院操| 久久一二 | 欧美在线看片 |