目录
目录
1.指针的概念
这次的内容想讲一下c语言中关于指针的一些概念。
首先,我们需要对指针有一个明确的认识,指针是什么,如何运算?大致分为以下4点:
1.指针是一个变量,用来存放地址,地址唯一标识着一块内存空间。
2.指针的大小为4/8个字节(32位平台/64位平台)
3.指针有类型,指针的类型决定了指针加减整数的步长(这一点后面会讲到),指针解引用时所操作的权限。
4.指针的运算。
2.字符指针
在指针的类型中我们知道有一种指针为字符指针char*;
一般定义如下
int main()
{
char ch = 'w'; //定义ch为char的字符变量
char* pc = &ch; //将ch的地址存储在字符指针pc中
*pc = 'w'; //pc的值解引用之后存储的字符w
return 0;
}
也有另外一种使用方法
int main()
{
const char* ptr = "hello c";
printf("%s", ptr);
return 0;
}
这种使用方法是将hello c字符串复制到了char* ptr中吗?其实不是,其实本质上是将hello c中的h的地址放到了ptr中,在printf函数打印的时候,会从起始地址开始打印,直到遇到'\0'之后停止。本质上就如下图所示。
3.数组指针
3.1数组指针的定义
数组指针本质上还是指针,相比于之前的整型指针和字符指针,那么数组指针就是能够指向数组的指针。接下来先分辨以下行代码,到底哪个是数组指针。
int *p1[10];
int (*p2)[10];
第一行是一个指针数组,它声明了一个名为p1的数组,其中包含10个整形指针元素。每个元素指向一个int类型的变量。这里的p1是一个由指针组成的数组。
第二行是数组指针,因为[ ]的优先级是高于*的,p先和*结合意味着p是一个指针,然后指向一个10个大小的整型数组,叫做数组指针。
3.2数组名vs&数组名
int arr[10];
arr和&arr分别代表什么?
arr代表数组名,而在数组中数组名代表数组的首元素的地址,那么&arr数组名到底是什么?
对比以下两端代码可以看出两者的区别;
#include <stdio.h>
int main()
{
int arr[10] = {0};
printf("%p\n", arr);
printf("%p\n", &arr);
return 0;
结果如下:
可以看到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分别+1,那么我们再来看一下运行的结果。
我们发现,arr+1和&arr+1的地址不一样了?
上面是用16进制表示的,那么转成10进制则是差了40个字节,一个int占4个字节,那么&arr和&arr+1之间刚好差了10个整型,所以&arr代表的是整个数组的地址,而arr则代表数组首元素的地址。
3.3数组指针的使用
那么了解了数组指针,那么数组指针是如何使用的?
既然数组指针指向的是数组,那么数组指针中存放的应该是数组的地址
如下代码:
void print_arr1(int arr[3][5], int row, int col)
{
int i = 0;
int j = 0;
for (i = 0; i < row; i++)
{
for (j = 0; j < col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
void print_arr2(int (*arr)[5], int row, int col)
{
int i = 0;
int j = 0;
for (i = 0; i < row; i++)
{
for (j = 0; j < col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
int main()
{
int arr[3][5] = { 1,2,3,4,5,6,7,8,9,10 };
print_arr1(arr,3,5);
print_arr2(arr,3,5);
return 0;
}
如上所示,打印的结果为
看似打印的结果一样,但是在函数传参的时候却是不一样的
print_arr1函数相当于是arr数组的拷贝,但是二维数组的首元素是二维数组的第一行。而print_arr2函数则相当于是声明指向了具有5列的整数数组的指针。
接下来再看一下,以下代码的区别:
int arr[5];
int *parr1[10];
int (*parr2)[10];
int (*parr3[10])[5];
int arr[5]是一个普通的整数数组
int *parr1[10] 是一个指针数组,每个元素是一个指向整数的指针
int (*parr2)[10]小括号内的先结合,说明parr2是一个指针,指向一个有10个整型的数组,所以parr2是一个数组指针。也就是指向数组的指针
int (*parr3[10])[5]小括号内parr3先和[10]结合说明parr3是一个指针数组,里面存放了10个指针变量,每个指针变量指向了有5个int类型的数组。
4.下期预告
下期根据这期的内容会往后拓展,然后讲关于数组传参,函数指针和函数指针数组的概念。