深入浅出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.数组名的理解

请看下面一段代码:

cpp 复制代码
#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.指针访问数组

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

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

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

cpp 复制代码
#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.一维数组传参的本质

再请看下面代码:

cpp 复制代码
#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语言中,指针是一个变量,其值为另一个变量的地址。而二级指针是一个指向指针的指针,即它的值是另一个指针的地址。

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

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

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

2.二级指针的内存表示

3.二级指针的解引用

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

cpp 复制代码
#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.指针指向单个字符

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

cpp 复制代码
int main()
{
 char ch = 'w';
 char *pc = &ch;
 *pc = 'w';
 return 0;
}

2.指针指向字符串

cpp 复制代码
#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》中有一道这样的面试题:

cpp 复制代码
#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");
	else
		printf("str1 and str2 are not same\n");

	if (str3 == str4)
		printf("str3 and str4 are same\n");
	else
		printf("str3 and str4 are not same\n");

	return 0;
} 

答案如下:

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

总结如下:

①**.str1==str2比较的是地址,而不是内容。**

每个数组对象都会分配属于自己的存储空间,地址各不相同。

指针变量并不分配存储区,而是指向常量字符串所在的静态存储区,由于指向的是同一个常量,所以str3=str4。

四、指针数组

1.指针数组的概念

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

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

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

cpp 复制代码
int *arr[5];

2.初始化指针数组

在声明时进行初始化:

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

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

cpp 复制代码
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.指针数组模拟二维数组

cpp 复制代码
#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语言中,数组指针是一种特殊的指针类型,它指向的是一个数组。 更具体地说,数组指针是一个指针,其指向的是包含特定数量元素的数组 。数组指针与指向单个变量的指针不同,它指向的是一个数组的首元素, 但是它的解引用操作会返回整个数组,而不是单个元素。
声明:数据类型(*指针名)[数组长度]

cpp 复制代码
int(*ptr)[5];

2.初始化数组指针

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

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

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

3.二维数组传参的本质

有了数组指针的理解,我们就能够讲⼀下⼆维数组传参的本质了。
过去我们有⼀个⼆维数组的需要传参给⼀个函数的时候,我们是这样写的:

cpp 复制代码
#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] 。那就意味着⼆维数组传参本质上也是传递了地址,传递的是第⼀
⾏这个⼀维数组的地址,那么形参也是可以写成指针形式的。如下:

cpp 复制代码
#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, ... 是函数的参数列表。

cpp 复制代码
int(*fubPtr)(int,int);

2.初始化函数指针

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

cpp 复制代码
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.函数指针的应用

通过函数指针调⽤指针指向的函数

cpp 复制代码
#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语言中,函数指针数组是一个数组,其元素是函数指针。这意味着函数指针数组中的每个元素都是一个指针,指向一个函数。函数指针数组可以用来存储多个函数的地址,并允许你在运行时根据需要选择其中一个函数来执行。
声明:返回类型 (*数组名)[数组长度];

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

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

2.初始化函数指针数组

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

cpp 复制代码
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.函数指针数组的应用

转移表

cpp 复制代码
#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语言程序。加油!祝你更上一层楼!!!

相关推荐
lulu_gh_yu22 分钟前
数据结构之排序补充
c语言·开发语言·数据结构·c++·学习·算法·排序算法
~yY…s<#>2 小时前
【刷题17】最小栈、栈的压入弹出、逆波兰表达式
c语言·数据结构·c++·算法·leetcode
EricWang13584 小时前
[OS] 项目三-2-proc.c: exit(int status)
服务器·c语言·前端
我是谁??4 小时前
C/C++使用AddressSanitizer检测内存错误
c语言·c++
希言JY5 小时前
C字符串 | 字符串处理函数 | 使用 | 原理 | 实现
c语言·开发语言
午言若5 小时前
C语言比较两个字符串是否相同
c语言
TeYiToKu7 小时前
笔记整理—linux驱动开发部分(9)framebuffer驱动框架
linux·c语言·arm开发·驱动开发·笔记·嵌入式硬件·arm
互联网打工人no17 小时前
每日一题——第一百二十四题
c语言
爱吃生蚝的于勒7 小时前
深入学习指针(5)!!!!!!!!!!!!!!!
c语言·开发语言·数据结构·学习·计算机网络·算法
羊小猪~~7 小时前
数据结构C语言描述2(图文结合)--有头单链表,无头单链表(两种方法),链表反转、有序链表构建、排序等操作,考研可看
c语言·数据结构·c++·考研·算法·链表·visual studio