作者主页:paper jie的博客_CSDN博客-C语言,算法详解领域博主
本文作者:大家好,我是paper jie,感谢你阅读本文,欢迎一建三连哦。
本文录入于《系统解析C语言》专栏,本专栏是针对于大学生,编程小白精心打造的。笔者用重金(时间和精力)打造,将C语言基础知识一网打尽,希望可以帮到读者们哦。
其他专栏:《C语言》《算法详解》《C语言-语法篇》
内容分享:本期对C语言中的指针的进阶知识进行进行具体讲解,各位看官姥爷快搬好小板凳坐好叭。
-------- 不要998,不要98,只要一键三连,三连买不了吃亏,买不了上当
目录
😄前言
🤣回顾知识
🫠字符指针
😜指针数组
🤫 数组指针
🤔数组指针的定义
😶🌫️ &数组名和数组名
🤓数组指针的使用
😴 数组参数,指针参数
🤤一维数组传参
🫥二维数组传参
🤕一级指针传参
🤯二级指针传参
🥹总结
😄前言
在前面的《指针不可怕,请爱它呵护它》http://t.csdn.cn/m7yf2 这篇文章中,已经对指针的基础知识:指针是什么,指针的类型,野指针,二级指针指针的运算与使用这些知识进行了基础且具体的讲解。学完这些知识大家对平常中的指针问题已经基本可以解决了。但是指针的知识远远不止这些,接下来的内容可以在帮助大家对C语言中的指针再提高一个理解。
🤣回顾知识
1. 指针就是一个变量,用来存放地址,地址唯一标识一块内存空间。
2. 指针的大小就是固定的4/8个字节(32位平台/64位平台)
3. 指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用的权限。
4. 指针的运算
🫠字符指针
在指针的类型中我们知道有一种指针类型为字符指针char*,它一般这么使用
还有一种方式大家看看:
这个代码大家就是容易把const char* pstr认为里面存的是hello bit,但其实本质上里面存的是字符串首元素的地址。所以它解引用出来肯定也是首元素。用const是因为hello bit 是一个常量,不能被修改,就要用const限定。
理解了上面的内容的话,我们看下面一道代码:
#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;
}
我们来分析一下为什么是这个结果:
C语言中,常量字符串一般都是储存到一块单独的内存空间。当几个指针同时指向这个字符串的时候,就是指向同一块内存,同一个地址。所以str3==str4
但是同相同的常量字符串去初始化不同的数组,因为数组随时都有可以发生不同的改变,为了安全起见,就会开辟不同的内存块。所以str1!=str2
😜指针数组
在《指针不可怕,请爱它呵护它》中我们也讲过了指针数组,它就是一个存放指针的数组。
🤫 数组指针
🤔数组指针的定义
数组指针是数组还是指针呢?答案当然是指针啦。我们在之前的文章中已经知道:
整型指针:int *p 能够指向整数据的指针
浮点数指针:float *pf 能够指针浮点数数据的指针
所以数组指针:能够指向数组的指针
判断一下,哪个是数组指针:
解释:
注意:[]的优先级是高于*的,所以要加上()保证p和*先结合
😶🌫️ &数组名和数组名
这里有一个数组 int arr[10] ,我们知道arr是数组名,数组名是首元素地址。那这样的话,&arr又是什么捏?
通过代码观察:
我们发现arr和&arr的地址都是一样的,难道它们是一样的含义吗?
接下来我们再看一段代码:
#include <stdio.h>
int main()
{
int arr[10] = { 0 };
printf("arr = %p\n", arr);
printf("&arr= %p\n", &arr);
printf("arr+1 = %p\n", arr+1);
printf("&arr+1= %p\n", &arr+1);
return 0;
}
通过代码,我们发现,虽然它们的地址一样,但是意义是不一样的。
实际上:&arr是数组的地址,不是数组首元素的地址。&arr加一是跳过整个数组,而arr加一是跳过一个元素 。
int(*arr)[10]中&arr的类型是int(*)[10],是一种数组指针。
🤓数组指针的使用
数组指针指向的是数组,那里面存放的就是数组的地址。
代码:
一个数组指针的正常使用场景:
理解:arr是数组首元素的地址,在二维数组中,首元素的地址就是第一行的地址,就可以用数组指针来接收。
//用数组指针打印二维数组
void print_arr2(int(*arr)[5], int row, int col)
{
int i = 0;
for (i = 0; i < row; i++)
{
int j = 0;
for (j = 0; j < col; j++)
{
printf("%d ", *(*(arr + i)+j));
//我们可以这么理解
//*(arr+i)就是就是数组的地址&arr解引用得到了数组arr
//arr在解引用就得到了元素
}
printf("\n");
}
}
int main()
{
int arr[3][5] = { {1,2,3,4,5},{6,7,8,9,10},{11,12,13,14,15} };
print_arr1(arr, 3, 5);
//数组名arr,表示首元素的地址
//但是二维数组的首元素是二维数组的第一行
//所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址
//可以数组指针来接收
print_arr2(arr, 3, 5);
return 0;
}
现在我们就可以理解下面代码的意思了:
😴 数组参数,指针参数
在写代码的时候难免要把数组或者指针传给函数,那函数的参数怎么写呢?
🤤一维数组传参
#include <stdio.h>
void test(int arr[])//ok?
//ok
//数组传参,数组接受,空格不影响,本质上函数是接受的首元素的地址,不会创建
{}
void test(int arr[10])//ok?
//ok
//理由同上
{}
void test(int* arr)//ok?
//ok
//arr就是一个整型地址,可以拿一个整型指针接收
{}
void test2(int* arr[20])//ok?
//ok
//指针数组传参,指针数组接收
{}
void test2(int** arr)//ok?
//ok
//arr2是指针数组首元素的地址就是==指针的地址==地址的地址
//拿二级指针接收
{}
int main()
{
int arr[10] = { 0 };
int* arr2[20] = { 0 };
test(arr);
test2(arr2);
}
🫥二维数组传参
void test(int arr[3][5])//ok?
//ok
//二维数组传参 二维数组接收
{}
void test(int arr[][])//ok?
//err
//不可以省略列
{}
void test(int arr[][5])//ok?
//ok
{}
//总结:二维数组传参,函数形参的设计只能省略第一个[]的数字。
//因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
//这样才方便运算。
void test(int* arr)//ok?
//err
//arr是二维数组第一行的地址,是一个一维数组,不能用int类型的指针接收
{}
void test(int* arr[5])//ok?
//err
//参数只可以用二维数组或者数组指针接收
//他是指针数组
{}
void test(int(*arr)[5])//ok?
//ok
//他就是数组指针
{}
void test(int** arr)//ok?
//err
//它是二级指针
//arr是二维数组第一行的地址,只能用数组指针接收或者二维数组
{}
int main()
{
int arr[3][5] = { 0 };
test(arr);
}
🤕一级指针传参
#include <stdio.h>
void print(int* p, int sz)
{
int i = 0;
for (i = 0; i < sz; i++)
{
//p+i就是相当与指针指向的地址不断向后加一
printf("%d\n", *(p + i));
}
}
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9 };
//一级指针p存放的是arr首元素的地址
int* p = arr;
int sz = sizeof(arr) / sizeof(arr[0]);
//一级指针p,传给函数
print(p, sz);
//p==arr
return 0;
}
当函数参数部分为一级指针的时候,函数可以接收什么参数呢?
看代码:
void test1(int* p)
{}
//test1函数能接收什么参数?
//可以接收 arr数组的数组名 整型变量的地址 一级指针
void test2(char* p)
{}
//test2函数能接收什么参数?
//字符变量的地址 一级指针
int main()
{
int a = 0;
int arr[10] = { 0 };
int* pa = &a;
test1(&a);//整型变量的地址
test1(pa);//pa就是整型a的地址
test1(arr);//arr是首元素的地址
char ch = 'w';
char* p = &ch;
test2(&ch); //字符变量的地址
test2(p); // p这个指针变量中就存放了字符的地址
return 0;
}
🤯二级指针传参
#include <stdio.h>
void test(int** ptr)//用二级指针接收
{
printf("num = %d\n", **ptr);
}
int main()
{
int n = 10;
int* p = &n;//一级指针 存的是n的地址
int** pp = &p;//二级指针 存的是一级指针p的地址
test(pp);//传二级指针 就是传的一级指针的地址
test(&p);// &p 取出的是一级指针p的地址
return 0;
}
当函数的参数部分为二级指针时,函数可以接收什么参数呢?
代码:
void test(char** p)//二级指针接收
{
}
int main()
{
char c = 'b';
char* pc = &c;
char** ppc = &pc;
char* arr[10];//指针数组
test(&pc); // &pc时取出一级指针pc的地址就是二级指针
test(ppc);//ppc 是二级指针
test(arr);//Ok?
//ok
//arr是指针数组首元素的地址,首元素是一级指针 所以它也相当于一个二级指针
return 0;
}
🥹总结
对于指针的进阶,里面的内容还是很多的,我打算分两篇文章来介绍。这篇文章我们重点介绍了数组指针和数组的传参,指针的传参。以代码的形式具体的分析了其中知识点,想必大家对于数组指针应该可以清楚的理解它,数组和指针的传参也知道什么可以传参,什么可以接收参数。接下来我们还会继续对进阶指针剩下的内容完善的,敬请期待!