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

深入浅出C语言指针(进阶篇)

深入浅出C语言指针(基础篇)

深入浅出C语言指针(进阶篇)

目录

引言

一、指针和数组

1.数组名的理解

2.指针访问数组 

3.一维数组传参的本质

二、二级指针

1.二级指针的概念

2.二级指针的内存表示

3.二级指针的解引用

三、字符指针

1.指针指向单个字符

2.指针指向字符串

3.一道面试题

四、指针数组

1.指针数组的概念

 2.初始化指针数组

3.指针数组模拟二维数组

五、数组指针

1.数组指针的概念

2.初始化数组指针

3.二维数组传参的本质

六、函数指针

1.函数指针的概念

2.初始化函数指针

3.函数指针的应用

七、函数指针数组

1.函数指针数组的概念

2.初始化函数指针数组

 3.函数指针数组的应用

 总结


引言

在C语言中,指针是至关重要的一部分,掌握指针的用法对于编写高效、简洁的代码具有极大帮助。本文将带您深入了解C语言指针的高级用法,助您迈向编程高手之路。

一、指针和数组

1.数组名的理解

请看下面一段代码:

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };printf("arr =   %p\n", arr);printf("&arr[0]=%p\n", &arr[0]);return 0;
}

运行结果如下: 

我们发现数组名和数组⾸元素的地址打印出的结果⼀模⼀样, 数组名就是数组⾸元素的地址

 两个特例

sizeof(数组名) ,sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩,
单位是字节
&数组名 ,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的)
除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。

2.指针访问数组 

我们搞清楚数组名就是数组首元素的地址之后,那么这样写代码就是可行的:

int arr[10] = {1,2,3,4,5,6,7,8,9,0};
int *p = arr;//p存放的是数组首元素的地址

当我们把数组名当成地址存放到指针中,就可以用指针访问数组了:

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int len = sizeof(arr) / sizeof(arr[0]);int* p = arr;//使用指针来访问数组int i = 0;for (i = 0; i < len; i++){printf("%d ", *(p + i));//这里*(p+i)等价于arr[i]}return 0;
}

可以看到成功访问了数组每个元素: 

3.一维数组传参的本质

再请看下面代码:

#include <stdio.h>
void test1(int arr[])
{printf("%d\n",sizeof(arr));
}
void test2(int *arr)
{printf("%d\n", sizeof(arr));
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };test1(arr);test2(arr);return 0;
}

输出结果如下:

上面我们学习了:数组名是数组⾸元素的地址;那么在数组传参的时候,传递的是数组名,也就是说本质上 数组传参本质上传递的是数组⾸元素的地址。
所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。那么在函数内部我们写
sizeof(arr) 计算的是⼀个地址的⼤⼩(单位字节)⽽不是数组的⼤⼩(单位字节)
由此我们可以总结

数组传参的本质是传递了数组首元素的地址,所以形参访问的数组和实参访问的数组是同一个数组。(所以形参的数组并不会再开辟空间,故也可以省略数组大小只写成arr[])
⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。

二、二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥呢?

答案是二级指针

1.二级指针的概念

在C语言中,指针是一个变量,其值为另一个变量的地址。而二级指针是一个指向指针的指针,即它的值是另一个指针的地址。

二级指针的定义可以表示为:

int a = 10;        // 声明一个整型变量a
int *p = &a;       // 声明一个指向整型变量a的指针p
int **pp = &p;     // 声明一个指向指针p的二级指针pp

在这个例子中,a的地址存放在p中,p的地址存放在pp中。p是一级指针,存储a的地址;pp 是一个二级指针,它存储的是指针 p 的地址。

2.二级指针的内存表示

3.二级指针的解引用

要获取二级指针所指向的指针所指向的变量的值,需要进行两次解引用:

#include <stdio.h>
int main() {int a = 10;int* p = &a;int** pp = &p;// 通过二级指针修改a的值**pp = 20;printf("a = %d\n", a);return 0;
}

这里的** pp 首先通过* pp 获取 p 的值(即 a 的地址)然后通过* 解引用这个地址,从而修改 a 的值。

三、字符指针

1.指针指向单个字符

这是一般的使用方法,这里进阶篇我们就不过多赘述:

int main()
{char ch = 'w';char *pc = &ch;*pc = 'w';return 0;
}

2.指针指向字符串

#include<stdio.h>
int main()
{const char* pstr = "abcdef";//这里是把一个字符串放到pstr指针变量里了吗?printf("%s\n", pstr);//%s打印字符串后面需提供地址return 0;
}

代码 const char* pstr = "abcdef"; 特别容易让人以为是把字符串abcdef放到字符指针pstr里了,但是本质是把字符串abcdef的首字符a的地址放到了pstr中。由于字符串是连续存放的,故也可以通过通过指针访问到整个字符串。

3.一道面试题

《剑指offer》中有一道这样的面试题:

#include <stdio.h>
int main()
{char str1[] = "hello bit.";char str2[] = "hello bit.";const char* str3 = "hello bit.";const char* str4 = "hello bit.";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");return 0;
} 

答案如下:

我们来分析一下:这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4相同。

总结如下:
.str1==str2比较的是地址,而不是内容。
每个数组对象都会分配属于自己的存储空间,地址各不相同。
指针变量并不分配存储区,而是指向常量字符串所在的静态存储区,由于指向的是同一个常量,所以str3=str4。

四、指针数组

1.指针数组的概念

指针数组是指针还是数组?
我们类⽐⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。
那指针数组呢?是存放指针的数组

指针数组:在C语言中,指针数组是一种特殊类型的数组,指针数组常用于存储一系列的地址,这些地址可以是变量的地址、数组元素的地址或者其他指针的地址。简单来说指针数组是一个数组,其中的每一个元素都是指向某种数据类型的指针。

声明数据类型*指针数组名[数组长度]

int *arr[5];

 2.初始化指针数组

在声明时进行初始化:

int a=1,b=2,c=3,d=4,e=5;
int *arr[5]={&a,&b,&c,&d,&e};

 也可以先声明一个指针数组,然后在后续代码中赋值:

int a=1,b=2,c=3,d=4,e=5;
int *arr[5];
arr[0]=&a;
arr[1]=&b;
arr[2]=&c;
arr[3]=&d;
arr[4]=&e;

3.指针数组模拟二维数组

#include <stdio.h>
int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 2,3,4,5,6 };int arr3[] = { 3,4,5,6,7 };//数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中int* parr[3] = { arr1, arr2, arr3 };int i = 0;int j = 0;for (i = 0; i < 3; i++){for (j = 0; j < 5; j++){printf("%d ", parr[i][j]);}printf("\n");}return 0;
}

parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数组中的元素。上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并⾮是连续的。

五、数组指针

1.数组指针的概念

数组指针是指针?还是数组?
答案是:指针。
我们已经熟悉:
整形指针: int * pi ; 能够指向整形数据的指针。
浮点型指针: float * pf ; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。
数组指针:在C语言中,数组指针是一种特殊的指针类型,它指向的是一个数组。 更具体地说,数组指针是一个指针,其指向的是包含特定数量元素的数组。数组指针与指向单个变量的指针不同,它指向的是一个数组的首元素, 但是它的解引用操作会返回整个数组,而不是单个元素
声明:数据类型(*指针名)[数组长度]
int(*ptr)[5];

2.初始化数组指针

数组指针可以通过取数组的地址来初始化:

int arr[5] = {1, 2, 3, 4, 5};
int (*ptr)[5] = &arr; // 初始化数组指针,使其指向数组arr|    |    ||    |    ||    |    ptr指向数组的元素个数|    ptr是数组指针变量名ptr指向的数组的元素类

在这里,&arr 表示取数组 arr 的地址(而不是首元素的地址),这个地址就是数组指针 ptr 的初始值。

3.二维数组传参的本质

 有了数组指针的理解,我们就能够讲⼀下⼆维数组传参的本质了。

过去我们有⼀个⼆维数组的需要传参给⼀个函数的时候,我们是这样写的:
#include <stdio.h>
void test(int a[3][5], int r, int c)
{int i = 0;int j = 0; for (i = 0; i < r; i++){for (j = 0; j < c; j++){printf("%d ", a[i][j]);}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };test(arr, 3, 5);return 0;
} 
这⾥实参是⼆维数组,形参也写成⼆维数组的形式,那还有什么其他的写法吗?
⾸先我们再次理解⼀下⼆维数组,⼆维数组起始可以看做是每个元素是⼀维数组的数组,也就是⼆维数组的每个元素是⼀个⼀维数组。那么⼆维数组的⾸元素就是第⼀⾏,是个⼀维数组。
如下图:

所以,根据数组名是数组⾸元素的地址这个规则, ⼆维数组的数组名表⽰的就是第⼀⾏的地址,是
维数组的地址,类型是数组指针类型 。根据上⾯的例⼦,第⼀⾏的⼀维数组的类型就是 int [5] , 所以第⼀⾏的地址的类
型就是数组指针类型 int(*)[5] 。那就意味着⼆维数组传参本质上也是传递了地址,传递的是第⼀
⾏这个⼀维数组的地址,那么形参也是可以写成指针形式的。如下:
#include <stdio.h>
void test(int(*p)[5], int r, int c)
{int i = 0;int j = 0;for (i = 0; i < r; i++){for (j = 0; j < c; j++){printf("%d ", *(*(p + i) + j));}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };test(arr, 3, 5);return 0;
} 

运行结果:

总结:⼆维数组传参,形参的部分可以写成数组,也可以写成指针

六、函数指针

1.函数指针的概念

函数指针:在C语言中,函数指针是一种特殊类型的指针,它存储的是函数的地址,而不是变量的地址。通过函数指针,你可以间接地调用函数,类似于通过普通指针间接访问变量。

声明返回类型 (*函数指针名)(参数类型1, 参数类型2, ...);

这里的返回类型是函数返回的类型,函数指针名是你给这个指针起的名字,而参数类型1, 参数类型2, ... 是函数的参数列表。

int(*fubPtr)(int,int);

2.初始化函数指针

函数指针可以通过取函数的地址来初始化:

int add(int a, int b) {return a + b;
}
int (*funcPtr)(int, int) = &add; // 初始化函数指针,使其指向add函数|    |       ----------          //也可以写成int (*funcPtr)(int, int)=add;|    |           ||    |         funPtr指向函数的参数类型和个数的交代|   函数指针变量名funPtr指向函数的返回类型

在这里,&add 表示取函数 add 的地址,这个地址就是函数指针 funcPtr 的初始值。

(这里add和&add是一样的,函数名和&函数名都表示函数地址,二者没有区别)

3.函数指针的应用

通过函数指针调⽤指针指向的函数
#include <stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int(*pf3)(int, int) = Add;printf("%d\n", Add(1, 2));//通过函数名调用printf("%d\n", (*pf3)(2, 3));//通过函数指针调用printf("%d\n", pf3(3, 5));//由于pf存储的是函数的地址,也可以通过函数地址直接调用,故也可以不写*号return 0;
}

运行结果如下:

七、函数指针数组

1.函数指针数组的概念

函数指针数组:在C语言中,函数指针数组是一个数组,其元素是函数指针。这意味着函数指针数组中的每个元素都是一个指针,指向一个函数。函数指针数组可以用来存储多个函数的地址,并允许你在运行时根据需要选择其中一个函数来执行。

声明:返回类型 (*数组名)[数组长度];

这里的 返回类型 是函数返回的类型,数组名 是数组的名字,而 数组长度 是一个常量表达式,表示数组中函数指针的数量。

int(*arr[5])(int,int);//在这个声明中,arr是一个数组,它包含5个指向整型函数的指针。

2.初始化函数指针数组

函数指针数组可以通过取函数的地址来初始化:

int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}int (*funcArray[2])(int, int) = {add, subtract}; // 初始化函数指针数组

在这里,funcArray 是一个数组,它包含两个指向整型函数的指针,分别指向 add 和 subtract 函数。

 3.函数指针数组的应用

转移表

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int add(int a, int b)
{return a + b;
}
int sub(int a, int b)
{return a - b;
}
int mul(int a, int b)
{return a * b;
}
int div(int a, int b)
{return a / b;
}
int main()
{int x, y;int input;int ret = 0;int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表do {printf("**********计算器**********\n");printf("******1.加法  2.减法******\n");printf("******3.乘法  4.除法******\n");printf("*********0.退出**********\n");printf("请选择:\n");scanf("%d", &input);if (input >= 1 && input <= 4){printf("输入需要操作的两个数\n");scanf("%d%d", &x, &y);ret = (*p[input])(x, y);printf("结果为%d\n", ret);}else if (input = 0){break;}else{printf("输入有误,请重新输入!\n");}} while (input);
}

运行如下:

 总结

通过本文的学习,相信你对C语言指针有了更深入的理解,能够更好地运用指针来编写高效的C语言程序。加油!祝你更上一层楼!!!

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 自动化测试--WebDriver API
  • element表单disabled功能失效问题
  • eqmx上读取数据处理以后添加到数据库中
  • 华为Ascend C算子开发(中级)考试
  • web网站组成
  • 《华为数据之道》读书笔记六---面向自助消费的数据服务建设
  • powershell自定义命令别名
  • pglogical扩展的基本用法介绍
  • leetcode日记(51)不同路径Ⅱ
  • 数据治理之“财务一张表”
  • python-爬虫实例(1):获取京东商品评论
  • linux-conda环境安装配置教程
  • Angular哪些方法适合写在constructor中?
  • 总结一些vue3小知识3
  • Spring IoC控制反转思想 DI依赖注入(五大注解+一个方法注解)
  • 10个最佳ES6特性 ES7与ES8的特性
  • js
  • node入门
  • python 学习笔记 - Queue Pipes,进程间通讯
  • 大快搜索数据爬虫技术实例安装教学篇
  • 使用API自动生成工具优化前端工作流
  • 手机app有了短信验证码还有没必要有图片验证码?
  • 协程
  • 与 ConTeXt MkIV 官方文档的接驳
  • media数据库操作,可以进行增删改查,实现回收站,隐私照片功能 SharedPreferences存储地址:
  • ​必胜客礼品卡回收多少钱,回收平台哪家好
  • #Datawhale X 李宏毅苹果书 AI夏令营#3.13.2局部极小值与鞍点批量和动量
  • #define,static,const,三种常量的区别
  • #我与Java虚拟机的故事#连载12:一本书带我深入Java领域
  • #我与Java虚拟机的故事#连载16:打开Java世界大门的钥匙
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (C++17) std算法之执行策略 execution
  • (javascript)再说document.body.scrollTop的使用问题
  • (二)Linux——Linux常用指令
  • (附源码)apringboot计算机专业大学生就业指南 毕业设计061355
  • (附源码)springboot 校园学生兼职系统 毕业设计 742122
  • (六)库存超卖案例实战——使用mysql分布式锁解决“超卖”问题
  • (四)opengl函数加载和错误处理
  • (转)AS3正则:元子符,元序列,标志,数量表达符
  • .JPG图片,各种压缩率下的文件尺寸
  • .NET序列化 serializable,反序列化
  • .net中我喜欢的两种验证码
  • /etc/motd and /etc/issue
  • /usr/bin/perl:bad interpreter:No such file or directory 的解决办法
  • @ModelAttribute使用详解
  • @Pointcut 使用
  • @RestControllerAdvice异常统一处理类失效原因
  • [ element-ui:table ] 设置table中某些行数据禁止被选中,通过selectable 定义方法解决
  • [ Linux ] git工具的基本使用(仓库的构建,提交)
  • []利用定点式具实现:文件读取,完成不同进制之间的
  • [2010-8-30]
  • [202209]mysql8.0 双主集群搭建 亲测可用
  • [AIGC] Nacos:一个简单 yet powerful 的配置中心和服务注册中心
  • [bbk5179]第66集 第7章 - 数据库的维护 03
  • [BZOJ 3680]吊打XXX(模拟退火)