news 2026/6/15 17:20:23

深究指针_2

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
深究指针_2

文章目录

  • 深究指针_2
    • void *类型
    • 内存管理
      • 内存布局
      • malloc
      • calloc()
      • 调整函数realloc()
    • 数组与指针
    • 指针和字符串操作

深究指针_2

void *类型

**(void *)**是任意类型指针,更准确地说,是“通用指针”或者是“无类型指针”。与函数定义void相似,多用在动态内存分配等地方,重点在于其通用性

  • void *就是一个指向未知类型数据的内存地址的指针
  • 它只包含一个地址值,但不包含关于这个地址上存储的数据类型的信息。

特性

  • 可以指向任何类型的数据

  • inta=10;floatb=3.14;charc='X';intarr[5]={1,2,3,4,5};void*vp;// 声明一个void指针vp=&a;// 可以指向intvp=&b;// 可以指向floatvp=&c;// 可以指向charvp=arr;// 可以指向数组
  • 不能直接解引用,不能进行指针算术运算,使用前要进行类型转换

  • inta=10;void*p;int*a_ptr=(int*)p;

内存管理

内存布局

在此之前我们先了解C程序内存布局。

栈(stack)局部变量,函数参数(自动管理)
······
堆(Heap)动态内存分配(手动管理)
BSS段没有初始化的全局/静态变量
数据段(data)已经初始化的全局/静态变量
代码段(text)程序代码

malloc

#include<stdio.h>#include<srdlib.h>intmain(){intn,sum,i,*p;printf("Enter n:");scanf("%d",&n);if((p=(int*)malloc(n*sizeof(int)))==NULL){printf("Fail to allocate memory.\n");exit(1);// 终止程序执行}//强制转换为整形指针,并且判断是否分配成功//p已经指向基地址printf("Enter %d integers:",n);for(i=0;i<n;i++){scanf("%d",p+i);//本身就是地址,也可以是p[i]//p已经是数组名}for(i=0;i<n;i++){sum+=*(p+i);}free(p);//用完后一定要释放return0;}

总结

确定需要多少内存空间

利用内存动态分配函数<srdlib.h>

用指针指向此空间,指向基地址即可,只可以通过指针访问

内存使用完毕释放

void*malloc(unsighnedchar)

在动态存储去,无符号整型,大小为字节数

申请成功,则返回一个指向分配内存空间的起始地址的指针

类型为 void *(上述通用指针),不成功返回NULL

int*p;if((p=(int*)malloc(n*sizeof(int)))==NULL){printf("Not able to allocate memory.\n");exit(1);}

calloc()

自动对整个区域进行初始化,而malloc()不执行操作

void*calloc(unsignedn;unsignedsize)

在动态存储区连续分配n个存储块,每个存储块长度为size,并且分配后全部初始化为0;

空间释放函数

voidfree(void*ptr);//指向释放空间首地址指针free(p);

调整函数realloc()

void*realloc(void*ptr,unsignedsize)

更改以前的存储分配

ptr必须是以前通过动态内存分配得到的指针

参数size是现在需要的空间大小

如果调整失败,返回NULL,ptr指向存储块的内容不变

如果调整成功,返回指向大小为size的存储块的起始地址的指针,并保证该块的内容与原块一致。

  • 如果size小于原块,内容为前size范围内的数据
  • 如果size大于原块,则原有数据存在新快的前一部分

如果分配成功,原存储快内容就可能改变了,因此不能再通过ptr去使用

数组与指针

数组名作为函数的参数,在函数调用时,将实参数组首元素的地址传给形参(指针变量),因此,形参也指向实参数组的首元素,如果改变形参所指向的单元的值,就改变实参数组元素的值。

源于数组名是指针常量,指针作为函数的参数。

指针和字符串操作

我们先来分析一个实例

#include<stdio.h>intmain(){char*str[]={"Hello","World","C"};char**pp=str;printf("%c\n",**pp);// 输出 'H'printf("%s\n",*(pp+1));// 输出 "World"printf("%c\n",*(*pp+1));// 输出 'e'printf("%c\n",*((*pp+1)+2);//输出'r'return0;}

我们来逐行分析这段代码

char*str[]={"Hello","World","C"};

这一行定义了一个指针数组

  • str是一个数组,包含了三个char*类型元素,分别指向定义的三个元素,str[i]指向第i个字符串。
  • 内存布局:
    • str[0]→ 指向字符串 “Hello”
    • str[1]→ 指向字符串 “World”
    • str[2]→ 指向字符串 “C”
    • *str[0] = “Hello”
char**pp=str;
  • pp是一个指向指针的二级指针
  • str作为数组名,退化为指向数组第一个元素的指针,也就是&str[0]
  • pp = &str[0]
  • *pp = str[0] = ‘Hello’ 这是指向第一个字符的指针
  • **pp = *(*pp) = *str[0] = ‘H’
printf("%s\n",*(pp+1));
  • pp+1 = &str[1] 移动一个指针大小的距离

  • *(pp+1) = str[1] = “World”

printf("%c\n",*(*pp+1));
  • *pp = str[0](数组名)是指向字符串"Hello"的指针,存储第一个字母的地址,*pp+1指向第二个字母的指针,即’e‘存储的位置。
  • *(*pp+1) = ‘e’

事实上,以上仅仅作为理解,写代码过于弯绕繁琐,通常指针数组可以直接: str[i]等效于(pp + i),str[i][j] 等效于 *(*(pp + i)+ j)*

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/14 18:56:45

数据结构:并查集

数据结构&#xff1a;并查集 并查集&#xff08;Disjoint Set Union&#xff0c;简称 DSU&#xff09;是一种用于高效管理和合并不相交集合的数据结构&#xff0c;核心支持两种操作&#xff1a; 查找&#xff08;Find&#xff09;&#xff1a;确定某个元素属于哪个集合&#xf…

作者头像 李华
网站建设 2026/6/15 12:44:25

5分钟搞定视频字幕:OpenSubtitlesDownload新手完整指南

5分钟搞定视频字幕&#xff1a;OpenSubtitlesDownload新手完整指南 【免费下载链接】OpenSubtitlesDownload Automatically find and download the right subtitles for your favorite videos! 项目地址: https://gitcode.com/gh_mirrors/op/OpenSubtitlesDownload 还在…

作者头像 李华
网站建设 2026/6/15 14:20:39

TeslaMate完整教程:从零开始搭建特斯拉数据监控平台

TeslaMate完整教程&#xff1a;从零开始搭建特斯拉数据监控平台 【免费下载链接】teslamate 项目地址: https://gitcode.com/gh_mirrors/tes/teslamate 你是否曾好奇&#xff1a;我的特斯拉电池健康度到底如何&#xff1f;驾驶习惯对能耗有多大影响&#xff1f;充电成本…

作者头像 李华
网站建设 2026/6/15 14:21:08

Higress网关健康检查:5个关键配置优化微服务稳定性

Higress网关健康检查&#xff1a;5个关键配置优化微服务稳定性 【免费下载链接】higress Next-generation Cloud Native Gateway | 下一代云原生网关 项目地址: https://gitcode.com/GitHub_Trending/hi/higress 在云原生架构中&#xff0c;Higress网关的健康检查功能是…

作者头像 李华