大家对于指针并不陌生,具有一定编程基础的都知道指针的概念:
- 指针是个变量,用来存放地址,地址唯一标识一块内存空间
- 指针大小是固定的4/8个字节(32位平台/64位平台)
- 指针是有类型的,指针的类型决定了指针+ - 整数的步长,指针解引用时候的权限
- 指针的运算
接下来我们讨论一下指针类型:字符指针,数组指针以及数组传参,函数指针
字符指针
字符指针 char *
常见的两种使用方式:
- 存放一个字符的地址
int main()
{
char c = 'a';
char *cp = &c;
return 0;
}
- 指向一个常量字符串
int main()
{
char *pc = "hello world";
printf("%s", pc);
return 0;
}
其中方式2中的本质是把字符串"hello world"
首字符的地址放在了pc
中,如下图所示:
看下边代码是否理解:
int main()
{
char str1[] = "hello";
char str2[] = "hello";
char *str3 = "hello";
char *str4 = "hello";
if (str1 == str2)
{
printf("str1 and str2 are same\n");
}
else
{
printf("str1 and str2 are same\n");
}
if (str3 == str4)
{
printf("str3 and str4 are same\n");
}
else
{
printf("str3 and str4 are same\n");
}
return 0;
}
这里
str3
和str4
指向同一个常量字符串。C/C++会把常量字符串存储到单独的内存区域,当几个指针指向同一个字符串时,实际指向了同一块内存。但是用相同的常量字符串去初始化不同的数组时就会开辟不同内存块,所以str1
和str2
不同。
数组指针
数组指针是指向数组的指针。
定义方式:int (*p)[10] //p指针是指向一个存放十个整形数组的指针
那么 &数组名 和 数组名 有什么区别呢?
int main()
{
int arr[10] = { 0 };
printf("%p\n", &arr);
printf("%p\n", arr);
printf("%p\n", &arr + 1);
printf("%p\n", arr + 1);
return 0;
}
数组指针常用于二维数组。
数组指针和指针数组:数组指针是指向的数组。指针数组是存放指针的数组。
数组传参
我们用两种方式打印二维数组arr[3][5]
。
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,11,12,13,14,15 };
print_arr1(arr, 3, 5);
//数组名arr,表示数组首元素地址
//二维数组数组名的首元素是二维数组的第一行
//这里传递的arr相当于第一行地址,是一维数组的地址,可以用数组指针接收
print_arr2(arr, 3, 5);
return 0;
}
函数指针
指向函数的指针
定义方式:void (*pfunc)()
;
比如加法函数:int add(int a, int b)
其函数指针为:int (*add)(int, int)