结构体中指针赋值问题的分析及C代码示例

开发 开发工具
本文对结构体中指针赋值问题进行了分析,并用C代码演示了指针的赋值方法。在实际的C语言项目中,很多程序出现问题,就是对指针的处理不当造成的。因此,熟练掌握各种指针的使用方法,是对一个合格的软件开发人员的基本要求。

[[178653]]

问题描述

某结构体的定义如下:

  1. typedef struct 
  2.     int     iAge;                // 年龄 
  3.     char    szAddr1[100];        // 地址1 
  4.     char   *pszAddr2;            // 地址2 
  5.     char  **pszAddr3;            // 地址3 
  6. } T_PeopleInfo; 

请问如何对结构体中的各个成员变量(尤其是指针变量)进行赋值?

问题分析及C代码示例

我们可以看到,在结构体T_PeopleInfo中,pszAddr2和pszAddr3均为指针,其中pszAddr2为一级指针,pszAddr3为二级指针。本文的重点,就是要找到对一级指针和二级指针赋值的正确方法。

我们把结构体T_PeopleInfo放到具体的C代码中,以直观地展现对结构体中的各个成员变量的赋值方法。

我们首先编写如下程序(程序1):

  1. /********************************************************************** 
  2. * 版权所有 (C)2016, Zhou Zhaoxiong。 
  3. * 文件名称:PointerTest.c 
  4. * 文件标识:无 
  5. * 内容摘要:演示指针的用法 
  6. * 其它说明:无 
  7. * 当前版本:V1.0 
  8. * 作    者:Zhou Zhaoxiong 
  9. * 完成日期:20160712 
  10. **********************************************************************/ 
  11. #include <stdio.h> 
  12.  
  13.  
  14. // 重定义数据类型 
  15. typedef signed   int        INT32; 
  16. typedef unsigned int        UINT32; 
  17. typedef unsigned char       UINT8; 
  18.  
  19. // 结构体定义 
  20. typedef struct 
  21.     UINT32   iAge;                    // 年龄 
  22.     UINT8    szAddr1[100];            // 地址1 
  23.     UINT8   *pszAddr2;                // 地址2 
  24.     UINT8  **pszAddr3;                // 地址3 
  25. } T_PeopleInfo; 
  26.  
  27.  
  28. /**************************************************************** 
  29. * 功能描述: 主函数 
  30. * 输入参数: 无 
  31. * 输出参数: 无 
  32. * 返 回 值: 0-执行完成 
  33. * 其他说明: 无 
  34. * 修改日期       版本号        修改人        修改内容 
  35. ------------------------------------------------------------- 
  36. * 20160712        V1.0     Zhou Zhaoxiong     创建 
  37. ****************************************************************/ 
  38. INT32 main(void) 
  39.     T_PeopleInfo tPeopleInfo = {0}; 
  40.  
  41.     // 结构体变量赋值 
  42.     // 对iAge赋值 
  43.     tPeopleInfo.iAge = 10; 
  44.  
  45.     // 对szAddr1赋值 
  46.     strncpy(tPeopleInfo.szAddr1, "Chongqing, China!", strlen("Chongqing, China!")); 
  47.  
  48.     // 对pszAddr2赋值 
  49.     strncpy(tPeopleInfo.pszAddr2, "Chengdu, China!", strlen("Chengdu, China!")); 
  50.  
  51.     // 对pszAddr3赋值 
  52.     strncpy(tPeopleInfo.pszAddr3, "Wuhan, China!", strlen("Wuhan, China!")); 
  53.  
  54.     // 打印变量的值 
  55.     printf("Age=%d, Addr1=%s, Addr2=%s, Addr3=%s", tPeopleInfo.iAge, tPeopleInfo.szAddr1, tPeopleInfo.pszAddr2, tPeopleInfo.pszAddr3); 
  56.  
  57.     return 0; 
  58.  

在程序1中,我们按照对结构体中的数组的赋值方法对指针赋值,程序可以编译通过,但运行的时候,程序便会挂掉。究其原因,是因为没有为pszAddr2和pszAddr3指针分配内存空间。

我们对程序1进行改进,编写出以下程序(程序2):

  1. /********************************************************************** 
  2. * 版权所有 (C)2016, Zhou Zhaoxiong。 
  3. * 文件名称:PointerTest.c 
  4. * 文件标识:无 
  5. * 内容摘要:演示指针的用法 
  6. * 其它说明:无 
  7. * 当前版本:V1.0 
  8. * 作    者:Zhou Zhaoxiong 
  9. * 完成日期:20160712 
  10. **********************************************************************/ 
  11. #include <stdio.h> 
  12.  
  13.  
  14. // 重定义数据类型 
  15. typedef signed   int        INT32; 
  16. typedef unsigned int        UINT32; 
  17. typedef signed   char       INT8; 
  18.  
  19. // 结构体定义 
  20. typedef struct 
  21.     UINT32   iAge;                    // 年龄 
  22.     INT8     szAddr1[100];            // 地址1 
  23.     INT8    *pszAddr2;                // 地址2 
  24.     INT8   **pszAddr3;                // 地址3 
  25. } T_PeopleInfo; 
  26.  
  27.  
  28. /**************************************************************** 
  29. * 功能描述: 主函数 
  30. * 输入参数: 无 
  31. * 输出参数: 无 
  32. * 返 回 值: 0-执行完成 
  33. * 其他说明: 无 
  34. * 修改日期       版本号        修改人        修改内容 
  35. ------------------------------------------------------------- 
  36. * 20160712        V1.0     Zhou Zhaoxiong     创建 
  37. ****************************************************************/ 
  38. INT32 main(void) 
  39.     T_PeopleInfo tPeopleInfo = {0}; 
  40.  
  41.     // 结构体变量赋值 
  42.     // 对iAge赋值 
  43.     tPeopleInfo.iAge = 10; 
  44.  
  45.     // 对szAddr1赋值 
  46.     strncpy(tPeopleInfo.szAddr1, "Chongqing, China!", strlen("Chongqing, China!")); 
  47.  
  48.     // 对pszAddr2赋值 
  49.     tPeopleInfo.pszAddr2 = (INT8 *)malloc(100); 
  50.     if (tPeopleInfo.pszAddr2 == NULL
  51.     { 
  52.         return -1; 
  53.     } 
  54.     strncpy(tPeopleInfo.pszAddr2, "Chengdu, China!", strlen("Chengdu, China!")); 
  55.  
  56.     // 对pszAddr3赋值 
  57.     tPeopleInfo.pszAddr3 = (INT8 *)malloc(100); 
  58.     if (tPeopleInfo.pszAddr3 == NULL
  59.     { 
  60.         return -2; 
  61.     } 
  62.     strncpy(tPeopleInfo.pszAddr3, "Wuhan, China!", strlen("Wuhan, China!")); 
  63.  
  64.     // 打印变量的值 
  65.     printf("Age=%d, Addr1=%s, Addr2=%s, Addr3=%s\n", tPeopleInfo.iAge, tPeopleInfo.szAddr1, tPeopleInfo.pszAddr2, tPeopleInfo.pszAddr3); 
  66.  
  67.     return 0; 

在程序2中,我们先使用malloc为pszAddr2和pszAddr3分配了内存空间(注意,执行malloc之后,要判断指针是否为空),此时就可以将变量值赋给它们。程序编译和运行都是正常的,输出结果如下:

  1. ~/zhouzx/Test/PointerTest> PointerTest  
  2. Age=10, Addr1=Chongqing, China!, Addr2=Chengdu, China!, Addr3=Wuhan, China! 

除了程序2可以实现对一级指针和二级指针的正常赋值之外,我们还可以编写如下程序(程序3):

  1. * 版权所有 (C)2016, Zhou Zhaoxiong。 
  2. * 文件名称:PointerTest.c 
  3. * 文件标识:无 
  4. * 内容摘要:演示指针的用法 
  5. * 其它说明:无 
  6. * 当前版本:V1.0 
  7. * 作    者:Zhou Zhaoxiong 
  8. * 完成日期:20160712 
  9. **********************************************************************/ 
  10. #include <stdio.h> 
  11.  
  12.  
  13. // 重定义数据类型 
  14. typedef signed   int        INT32; 
  15. typedef unsigned int        UINT32; 
  16. typedef signed   char       INT8; 
  17.  
  18. // 结构体定义 
  19. typedef struct 
  20.     UINT32   iAge;                    // 年龄 
  21.     INT8     szAddr1[100];            // 地址1 
  22.     INT8    *pszAddr2;                // 地址2 
  23.     INT8   **pszAddr3;                // 地址3 
  24. } T_PeopleInfo; 
  25.  
  26.  
  27. /**************************************************************** 
  28. * 功能描述: 主函数 
  29. * 输入参数: 无 
  30. * 输出参数: 无 
  31. * 返 回 值: 0-执行完成 
  32. * 其他说明: 无 
  33. * 修改日期       版本号        修改人        修改内容 
  34. ------------------------------------------------------------- 
  35. * 20160712        V1.0     Zhou Zhaoxiong     创建 
  36. ****************************************************************/ 
  37. INT32 main(void) 
  38.     T_PeopleInfo tPeopleInfo = {0}; 
  39.  
  40.     // 结构体变量赋值 
  41.     // 对iAge赋值 
  42.     tPeopleInfo.iAge = 10; 
  43.  
  44.     // 对szAddr1赋值 
  45.     strncpy(tPeopleInfo.szAddr1, "Chongqing, China!", strlen("Chongqing, China!")); 
  46.  
  47.     // 对pszAddr2赋值 
  48.     tPeopleInfo.pszAddr2 = "Chengdu, China!"
  49.  
  50.     // 对pszAddr3赋值 
  51.     tPeopleInfo.pszAddr3 = "Wuhan, China!"
  52.  
  53.     // 打印变量的值 
  54.     printf("Age=%d, Addr1=%s, Addr2=%s, Addr3=%s\n", tPeopleInfo.iAge, tPeopleInfo.szAddr1, tPeopleInfo.pszAddr2, tPeopleInfo.pszAddr3); 
  55.  
  56.     return 0; 
  57.  

在程序3中,我们直接将字符串赋给了pszAddr2和pszAddr3,也就是将这两个字符串的首地址赋给了指针。那么,指针所指向的地址中存放的内容就是字符串的值。程序编译和运行都是正常的,输出结果如下:

  1. ~/zhouzx/Test/PointerTest> PointerTest  
  2. Age=10, Addr1=Chongqing, China!, Addr2=Chengdu, China!, Addr3=Wuhan, China! 

另,对于二级指针的赋值,我们还可以编写如下程序(程序4):

  1. /********************************************************************** 
  2. * 版权所有 (C)2016, Zhou Zhaoxiong。 
  3. * 文件名称:PointerTest.c 
  4. * 文件标识:无 
  5. * 内容摘要:演示指针的用法 
  6. * 其它说明:无 
  7. * 当前版本:V1.0 
  8. * 作    者:Zhou Zhaoxiong 
  9. * 完成日期:20160712 
  10. **********************************************************************/ 
  11. #include <stdio.h> 
  12.  
  13.  
  14. // 重定义数据类型 
  15. typedef signed   int        INT32; 
  16. typedef unsigned int        UINT32; 
  17. typedef signed   char       INT8; 
  18.  
  19. // 结构体定义 
  20. typedef struct 
  21.     UINT32   iAge;                    // 年龄 
  22.     INT8     szAddr1[100];            // 地址1 
  23.     INT8    *pszAddr2;                // 地址2 
  24.     INT8   **pszAddr3;                // 地址3 
  25. } T_PeopleInfo; 
  26.  
  27.  
  28. /**************************************************************** 
  29. * 功能描述: 主函数 
  30. * 输入参数: 无 
  31. * 输出参数: 无 
  32. * 返 回 值: 0-执行完成 
  33. * 其他说明: 无 
  34. * 修改日期       版本号        修改人        修改内容 
  35. ------------------------------------------------------------- 
  36. * 20160712        V1.0     Zhou Zhaoxiong     创建 
  37. ****************************************************************/ 
  38. INT32 main(void) 
  39.     T_PeopleInfo tPeopleInfo = {0}; 
  40.  
  41.     // 结构体变量赋值 
  42.     // 对iAge赋值 
  43.     tPeopleInfo.iAge = 10; 
  44.  
  45.     // 对szAddr1赋值 
  46.     strncpy(tPeopleInfo.szAddr1, "Chongqing, China!", strlen("Chongqing, China!")); 
  47.  
  48.     // 对pszAddr2赋值 
  49.     tPeopleInfo.pszAddr2 = "Chengdu, China!"
  50.  
  51.     // 对pszAddr3赋值 
  52.     tPeopleInfo.pszAddr3 = (INT8 *)malloc(100); 
  53.     if (tPeopleInfo.pszAddr3 == NULL
  54.     { 
  55.         return -1; 
  56.     } 
  57.     *(tPeopleInfo.pszAddr3) = "Wuhan, China!"
  58.  
  59.     // 打印变量的值 
  60.     printf("Age=%d, Addr1=%s, Addr2=%s, Addr3=%s\n", tPeopleInfo.iAge, tPeopleInfo.szAddr1, tPeopleInfo.pszAddr2, *(tPeopleInfo.pszAddr3)); 
  61.  
  62.     return 0; 

在程序4中,我们先用malloc为pszAddr3分配了内存空间,然后便可以使用该指针来接收字符串变量的值(注意,这里是将“Wuhan, China!”赋给了*(tPeopleInfo.pszAddr3))。程序编译和运行都是正常的,输出结果如下:

  1. ~/zhouzx/Test/PointerTest> PointerTest  
  2. Age=10, Addr1=Chongqing, China!, Addr2=Chengdu, China!, Addr3=Wuhan, China! 

总结

本文对结构体中指针赋值问题进行了分析,并用C代码演示了指针的赋值方法。

在实际的C语言项目中,很多程序出现问题,就是对指针的处理不当造成的。因此,熟练掌握各种指针的使用方法,是对一个合格的软件开发人员的基本要求。

【本文是51CTO专栏作者周兆熊的原创文章,作者微信公众号:周氏逻辑(logiczhou)】

责任编辑:武晓燕 来源: csdn博客
相关推荐

2022-01-12 08:30:55

结构体指针STM32

2009-08-13 15:41:50

C#结构体指针

2023-10-10 13:58:00

C语言代码结构体

2021-05-11 11:31:52

C语言类型指针

2009-08-31 15:02:22

C#解析结构体指针

2011-07-20 16:43:34

C++

2014-04-01 10:11:33

C语言指针

2011-04-11 14:18:37

CC++指针

2022-09-30 15:03:09

C语言深拷贝浅拷贝

2021-11-25 10:36:04

DNS命令Linux

2016-12-20 11:12:11

C代码自测开发

2017-05-27 06:32:36

JSONC代码函数

2009-08-14 11:05:28

C#语言的结构体

2009-08-13 11:18:50

C#结构体

2009-08-13 14:46:03

C#结构体定义

2022-01-09 23:04:19

语言打印结构体

2009-08-27 15:53:30

C#中using wo

2009-08-13 13:29:04

C#结构体使用

2014-02-10 15:05:37

C语言封装

2009-08-13 14:24:44

C#结构体构造函数
点赞
收藏

51CTO技术栈公众号