🎈个人主页:JAMES别扣了
💕在校大学生一枚。对IT有着极其浓厚的兴趣
✨系列专栏目前为C语言初阶、后续会更新c语言的学习方法以及c题目分享.
😍希望我的文章对大家有着不一样的帮助,欢迎大家关注我,我也会回关,大家一起交流一起互动,感谢大家的多多支持哈!🎉欢迎 👍点赞✍评论⭐收藏
前言
指针在C语言的版块中是有着极其重要的作用,通过两篇关于指针基础文章以及读者不断的代码实现可以很好掌握指针的基础知识,接下来我将结合C语言中数组方面的知识给读者们展示不一样的指针内容,这个系列文章我会写三篇左右,谢谢大家支持哈!
往期相关内容高质量文章(93+质量分)
C语言数组:一篇文章让你秒懂基础!-CSDN博客
C语言指针(上):一篇文章让你秒懂基础!-CSDN博客
C语言指针(下):一篇文章让你秒懂基础!(5000字)-CSDN博客
1.数组名的理解
在之前的数组文章中,我们一起探讨过数组的基本知识,现在让我们更深一步理解数组的相关概念。
这⾥我们使⽤ &arr[0] 的⽅式拿到了数组第⼀个元素的地址,但是其实数组名本来就是地址,⽽且 是数组⾸元素的地址,我们来做个测试。
#include <stdio.h>
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("arr = %p\n", arr)
return 0;
}
输出结果:
我们发现数组名和数组⾸元素的地址打印出的结果⼀模⼀样,数组名就是数组⾸元素(第⼀个元素)的地 址。
这时候有读者会有疑问?数组名如果是数组⾸元素的地址,那下⾯的代码怎么理解呢?
#include <stdio.h>
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("%d\n", sizeof(arr));
return 0;
}
输出的结果是:40,如果arr是数组⾸元素的地址,那输出应该的应该是4/8才对。 其实数组名就是数组⾸元素(第⼀个元素)的地址是对的,但是有两个例外:
• sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩, 单位是字节
• &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素 的地址是有区别的)
除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。
这时有好奇的同学,在试⼀下这个代码:
#include <stdio.h>
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("arr = %p\n", arr);
printf("&arr = %p\n", &arr);
return 0;
)
哈哈,这时会有读者问那么arr与&arr有什么区别么?
#include <stdio.h>
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("&arr[0]+1 = %p\n", &arr[0]+1);
printf("arr = %p\n", arr);
printf("arr+1 = %p\n", arr+1);
printf("&arr = %p\n", &arr);
printf("&arr+1 = %p\n", &arr+1);
return 0;
}
输出结果:
&arr[0] = 0077F820
&arr[0]+1 = 0077F824
arr=0077F820
arr+1=0077F824
&arr=0077F820
&arr+1 = 0077F848
这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1相差4个字节,是因为&arr[0]和arr都是 ⾸元素的地址,+1就是跳过⼀个元素。 但是&arr和&arr+1相差40个字节,这就是因为&arr是数组的地址,+1操作是跳过整个数组的。 到这⾥⼤家应该搞清楚数组名的意义了吧。
2. 使⽤指针访问数组
有了前⾯知识的⽀持,再结合数组的特点,我们就可以很⽅便的使⽤指针访问数组了。
#include <stdio.h>
int main()
{
int arr[10] = {0}; //输⼊
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0]); //输⼊
int* p = arr;
for(i=0; i<sz; i++)
{
scanf("%d", p+i);
//scanf("%d", arr+i);//也可以这样写
} //输出
for(i=0; i<sz; i++)
{
printf("%d ", *(p+i));
}
return 0;
}
这个代码搞明⽩后,我们再试⼀下,如果我们再分析⼀下,数组名arr是数组⾸元素的地址,可以赋值 给p,其实数组名arr和p在这⾥是等价的。那我们可以使⽤arr[i]可以访问数组的元素,那p[i]是否也可 以访问数组呢?
#include <stdio.h>
int main()
{
int arr[10] = {0}; //输⼊
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0]); //输⼊
int* p = arr;
for(i=0; i<sz; i++)
{
scanf("%d", p+i);
//scanf("%d", arr+i);//也可以这样写
} //输出
for(i=0; i<sz; i++)
{
printf("%d ", p[i]);
}
return 0;
}
将*(p+i)换成p[i]也是能够正常打印的,所以本质上p[i]是等价于*(p+i)。
同理arr[i] 应该等价于*(arr+i),数组元素的访问在编译器处理的时候,也是转换成⾸元素的地址+偏移 量求出元素的地址,然后解引⽤来访问的。
3. ⼀维数组传参的本质
数组我们学过了,之前也讲了,数组是可以传递给函数的,这个⼩节我们讨论⼀下数组传参的本质。 ⾸先从⼀个问题开始,我们之前都是在函数外部计算数组的元素个数,那我们可以把函数传给⼀个函 数后,函数内部求数组的元素个数吗?
#include <stdio.h>
void test(int arr[])
{
int sz2 = sizeof(arr)/sizeof(arr[0]);
printf("sz2 = %d\n", sz2);
}
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int sz1 = sizeof(arr)/sizeof(arr[0]);
printf("sz1 = %d\n", sz1);
test(arr);
return 0;
}
输出的结果:
我们发现在函数内部是没有正确获得数组的元素个数。
这就要学习数组传参的本质了,上个⼩节我们学习了:数组名是数组⾸元素的地址;那么在数组传参 的时候,传递的是数组名,也就是说本质上数组传参本质上传递的是数组⾸元素的地址。 所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。那么在函数内部我们写 sizeof(arr) 计算的是⼀个地址的⼤⼩(单位字节)⽽不是数组的⼤⼩(单位字节)。正是因为函 数的参数部分是本质是指针,所以在函数内部是没办法求的数组元素个数的。
void test(int arr[])//参数写成数组形式,本质上还是指针
printf("%d\n", sizeof(arr));
}
void test(int* arr)//参数写成指针形式
{
printf("%d\n", sizeof(arr));//计算⼀个指针变量的⼤⼩
}
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
test(arr);
return 0;
}
{
结论:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。
4.⼆级指针
指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥?答案是:更高级别的指针!
这就是 ⼆级指针。
对于⼆级指针的运算有:
1.*ppa 通过对ppa中的地址进⾏解引⽤,这样找到的是
int b = 20;
*ppa = &b;// 等价于 pa = &b; pa , *ppa 其实访问的就是 pa .
2.**ppa 先通过 *ppa 找到 pa ,然后对pa进行解应用操作:*pa,那找到的是a。
**ppa = 30;
// 等价于 *pa = 30;
// 等价于 a = 30;
5. 指针数组
指针数组是指针还是数组? 我们类⽐⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。 那指针数组呢?是存放指针的数组.
指针数组的每个元素都是⽤来存放地址(指针)的。 如下图:
指针数组的每个元素是地址,⼜可以指向⼀块区域.
6. 指针数组模拟⼆维数组
#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]就是整型⼀维数 组中的元素。 上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并⾮是连续的。
总结
C语言中数组与指针的深入探索
在C语言编程中,数组和指针是两个紧密相关的概念。数组是一种数据结构,用于存储相同类型的元素集合;而指针则是一种变量,用于存储另一个变量的内存地址。理解数组和指针之间的关系,对于深入掌握C语言至关重要。
一、数组名的理解
在C语言中,数组名实际上是一个指向数组首元素的常量指针。这意味着数组名本身并不表示整个数组,而是指向数组第一个元素的地址。因此,我们不能对数组名进行赋值操作,但可以获取其地址或进行解引用操作。
二、使用指针访问数组
通过指针,我们可以直接访问数组中的元素。假设我们有一个整型数组int arr[5];,我们可以使用指针int *p = arr;来指向数组的首元素。此时,*p就表示数组的第一个元素,*(p+1)表示第二个元素,以此类推。通过这种方式,我们可以遍历整个数组。
三、一维数组传参的本质
在C语言中,当我们将一维数组作为参数传递给函数时,实际上传递的是数组首元素的地址。因此,函数内部接收到的参数实际上是一个指针,指向数组的首元素。这使得我们可以在函数内部通过指针来访问和修改数组的元素。
四、二级指针
二级指针是指向指针的指针。它的声明形式为int **pp;,其中pp是一个指向整型指针的指针。二级指针常用于动态内存分配、函数返回多个值以及处理指针数组等场景。
五、指针数组
指针数组是指数组的每个元素都是指针。例如,int *pArr[5];表示一个包含5个整型指针的数组。每个指针可以指向一个整型变量或整型数组的首元素。指针数组常用于存储多个字符串的地址或处理多个动态分配的数组。
六、指针数组模拟二维数组
在C语言中,二维数组在内存中是连续存储的,但我们可以使用指针数组来模拟二维数组的行为。通过为每个指针分配一个一维数组,我们可以创建一个不规则的“二维数组”。这种方式在处理稀疏矩阵或动态调整数组大小时非常有用。
通过上述内容的阐述,我们可以看到数组和指针在C语言中的紧密关系。理解并掌握这些知识点,不仅有助于我们更加灵活地操作数组和指针,还能提升我们的编程能力和代码质量。希望本文对你有所帮助,祝你学习进步!