学习了一些指针的基础知识,我们来深入理解指针。
一 字符指针变量
有一种指针类型叫字符指针。
一般形式为:
char ch = 'w';
char* p = &ch;//p就是字符指针
我们来看一下如下代码
#include <stdio.h>
int main()
{
char* p = "abcdef";
printf("%c\n", *p);
return 0;
}
上述代码,很多人都会认为是把 字符串"abcdef"整个放在字符指针p里了,其实不然,而是把字符串里第一个字符的地址存放在p中了,我们看一下结果:
我们再来看一个代码
#include <stdio.h>
int main()
{
const char* p = "abcdef";
printf("%s\n", p);
return 0;
}
上述代码,是把常量字符串的首字符a的地址存放在了字符指针变量p里了,指针变量p只能读取字符串的内容,不能修改。
我们再来分析一下如下代码
#include <stdio.h>
int main()
{
char str1[] = "hello CSDN";
char str2[] = "hello CSDN";
const char* str3 = "hello CSDN";
const char* str4 = "hello CSDN";
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;
}
上述代码,数组str1和str2,数组名都是首元素的地址,但是数组str1和str2都是独立的数组,数组内容可以改变,所以str1不等于str2;而str3和str4都是字符指针变量,指向的是常量字符串,存放的是常量字符串的首字符的地址,即‘h’的地址,不可以被修改,所以str3等于str4。
要点:内容相同的常量字符串只会保存一份
二 数组指针变量
刚看到这个名词,很多人都会疑惑这个是数组还是指针变量,我们可以通过类比的方式来确认。
之前我们已经知道整型指针变量,它能存放整型变量的地址,它是指向整型数据的指针;字符型指针变量,它能存放字符型变量的地址,它是指向字符型数据的指针;通过类比,我们知道数组指针变量应该是存放数组的地址,能够指向数组的指针变量。
我们来分析一下下面代码
int* p1[5];
int(*p2)[5];
上述代码,p1是一个数组,其元素个数是5,每个元素的类型是int*,因此p1是一个指针数组;通过符号优先级顺序,知道p2是指针,指针指向的是数组,其元素个数为5,每个元素的类型是int,因此p2是指向数组的指针,即数组指针。
int main()
{
int a = 1;
int* p1 = &a;//p1是整型指针变量
char b = 'w';
char* p2= &b;//p2是字符指针变量
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int (*p3)[10] = &arr;//取出的是arr的地址,p3就是一个数组指针变量
return 0;
}
数组指针类型解析:
三 函数指针变量
要点:1.函数名是函数的地址
2.函数的地址即为函数指针,可以通过函数指针执行或者调用对应的函数
我们也是通过类比,来理解函数指针变量的,函数指针变量应该是用来存放函数地址的,后面运用地址来调用函数。那么函数是否会有地址呢,我们来看下面代码:
#include <stdio.h>
void func()
{
printf("hello CSDN");
}
int main()
{
printf("&func: %p\n", &func);
printf("func: %p\n", func);
return 0;
}
上述代码,我们通过%p占位符打印出了地址,证明了函数是有地址的,函数名就是其地址,且&函数名的地址和其一样。如果我们要把函数的地址存放起来,我们就得运用到函数指针变量了,该定义形式和数组指针变量大致相同。
一般形式如下:
类型说明符 (*指针变量名)();
例如:
void func()
{
printf("hello CSDN");
}void (*p1)() = func;
void (*p2)() = &func;
函数指针类型解析:
四 函数指针数组
之前我们已经学习了数组,数组是存放相同类型数据的存储空间。那把一个函数的地址存放在数组中,那是什么呢?那就是函数指针数组。
通过前面的学习,我们可以知道函数指针数组应该是这样定义的:
int (*p[5])();
上述代码,p先和[]结合,说明了p是数组,数组的内容是int (*)()类型的函数指针。