其回顾之前基础篇讲过关于指针的简单概念:
1.
指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
2.
指针的大小是固定的
4/8
个字节(
32
位平台
/64
位平台)。
3.
指针是有类型,指针的类型决定了指针的
+-
整数的步长,指针解引用操作的时候的权限。
4.
指针的运算。
今天介绍几种特殊的指针
1.字符串指针(char* )
字符串指针存放字符串中第一个字符的地址,使用方法如下:
int main()
{
char ch = 'w';
char *pc = &ch;
*pc = 'w';
return 0;
}
int main()
{
const char* pstr = "hello bit.";
printf("%s\n", pstr);
return 0;
}
其本质是把字符串 hello bit. 首字符的地址放到了pstr中。
这里有一道关于字符串指针的面试题,我们可以探讨下
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不同。运行结果如图
2. 指针数组
int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5];//二级字符指针的数组
以整形指针的数组为例,介绍他在内存中的情况,定义一个指针数组:
int a1[5],a2[5],a3[5],a4[5]; int * arr[4]={a1,a2,a3,a4}
3. 数组指针
数组指针是指针?还是数组? 答案是:指针。
我们已经熟悉:
整形指针:
int * pint
;
能够指向整形数据的指针。
浮点型指针:
float * pf
;
能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。其定义如下:
int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个
指针,指向一个数组,叫数组指针。
//这里要注意:[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。
下面讨论一下对于数组指针的赋值:是int (*p)[10] = &arr;还是int (*p)[10] = arr;
可见其结果是一样的,但是意义不同。&arr 表示的是数组的地址,而不是数组首元素的地址。
下面这段代码能更好的解释两者的区别
这里可以回顾几种定义,辨析一下各自的类型
int arr[5]; //数组:存放了5个int类型元素的数组
int *parr[5]; //指针数组:存放了5个指针元素的数组,每个指针指向一个int类型的数据
int (*parr)[5]; //数组指针:指向一个数组,数组中存放了5个int类型的数据
int (*parr[10])[5]; //数组指针数组:存放了5个指针,每个指针指向一个存放了10个int数据的数组
4.函数指针
函数指针及指向函数是指针,指针中存放了函数的地址,那函数的地址怎样表示,是函数名还是&函数名
由此可见,两种方式都对,那函数指针定义可以为:
pfun1
先和
*
结合,说明
pfun1
是指针,指针指向的是一个函数,指向的函数无参
数,返回值类型为
void
。
void test()
{
printf("hehe\n");
}
void (*pfun1)();