在C语言的学习过程中,指针是躲不掉的一大困难,开始的时候,可能你会觉得初始化整形指针和解引用不过如此,但是当类型逐渐复杂起来以后,没有对指针和类型的深入理解,想要看懂和很好的运用指针就比较困难了,希望通过我的这篇文章,让大家对指针能有更好更深入的认识。
在计算机中,必不可少的不只是CPU的运算,还需要有内存对数据的存储,然而在存储数据的过程中,必然会面对这样的问题:那么多的内存单元,数据到底存在哪?CPU运算时又要从哪读取数据呢?这时候就要想了:如果你有一个朋友住在了宾馆,让你去找他,你会怎么找呢?肯定不是一个一个房间敲开找,要问道朋友的房间号,才能一次性找到他的位置。那么我们是否可以给每个内存都给一个编号呢?答案是,可以。这便有了指针。
1.内存被划分成一个个的单元,一个内存单元大小是一个字节
2.每个内存单元给一个编号,这个编号就是地址,C语言中简称:指针
大概就是以下的换算
内存单元的编号==地址==指针
#include
int main()
{
int a = 10;
int *pa = &a;//*说明pa是一个指针变量(指向int类型),其中存放的是指针,也就是a的地址
//&a表示的是取出a变量的地址,然后通过语句赋给pa
*pa = 20;//此处的*为解引用,相当于通过pa存的地址找到a,可以理解为*pa==a
printf("%d %d\n",*pa,a);//此时打印的两值都为20
return 0;
}
通过以上的讲解,大概认识了指针是个什么东西,并且了解了其简单使用了,下面专们来讲讲指针变量的大小。
指针变量大小:
指针变量专门用来存地址,指针变量的大小取决于地址存放所要多大的空间
1.32位机器上:地址线32根,二进制32bit位,要将此地址存起来,需要四个字节的空间
2.64位机器:同理,一个地址需要八个字节
注:地址和类型所占的空间不同,64位机器就像一个相比32位机器更大的宾馆,所需要的编号位数需要更大,但是每个房间的大小却不变一样。故:int类还是占4个字节的空间,只是找到这个空间的编号地址变复杂了而已。
在刚刚的代码块中,我们谈到了int *pa,意思是pa是一个指针,指向的变量是int类型的数据,那我们合理联想一下,是不是有int*,就有char*,long long int*呢?答案是:是的,char *p意思是开辟指向char类型的指针空间,long long int*就是指向long long int类型的。前面提到的指针解引用*p中,int *p中的*p可以访问4个字节的空间,而char *p的*p可以访问1个字节的空间,以此类推。
根据上述指针变量类型,我们就可以对指针进行加减运算,见下方代码
#include
int main()
{
int a = 10;
int *pa = &a;
char *pc = (char*)&a;
//pa + 1---->地址 + 4
//pc + 1---->地址 + 1
//以上加减跳过的字节数是以指针类型为依据的,注意观察
return 0;
}
而在C语言中,有数组这样的一种数据类型,他的下标随地址的增加而增加,且在内存中连续存放,那我们是否可以通过指针而非[i]来访问数组元素呢?,见代码
#include
int main()
{
int arr[10]={0,1,2,3,4,5,6,7,8,9};
int *p = arr;//数组名是首元素地址
for(int i = 0;i < 10;i++){
printf("%d ",arr[i]);
}
for(int i = 0;i < 10;i++){
printf("%d ",*(p + i));//上下两块代码打印的结果是一样的,其实运行的效果也是一样的
}
return 0;
}//代码可以CV自测一下
指针减指针得到的是int类型的数据,但是相减时一定要满足这样一个条件,就是:两个相减的指针指向的是同一块空间。eg:指向同一个数组里的不同元素,这样大指针减小指针便可以得到指针间的元素个数了。
#include
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
printf("%d\n",&arr[9]-&arr[0]);//将打印指针间的元素个数
return 0;
}
在指针中,有这样一种特殊的指针,那就是void*(无具体类型指针),在用指针变量存放不同类型的地址时,不是相同类型的地址,是不能相互赋值的,要想放入指针变量类型不同的地址,只有两个办法,一个是强制类型转换,另一个便是void*,见代码
char a = 'm';
int *p = &a;//这样写会报错
int *p = (int*)&a;//将a的地址类型强制转换成int*,放入时不报错
void *p = &a;//将a的地址放入void*类型的p中,不报错
void*类型的指针可以放其他任意类型的指针,比如char*,int*等等,虽然放的指针类型很多,但还是有一定的局限性
注:
1.void*类型不能直接解引用
2.void*类型指针不可加减整数
虽然void*可以包含许多类型的指针,但这恰恰让其只知道指向的位置,却不知道访问多大的空间,故产生以上问题。
const使变量有了常属性,见下代码
#include
int main()
{
const int a = 10;//给a加上const
a = 20;//这时修改a的值时编译器就会报错
int *pa = &a;
*pa = 20;//这时候a的值将被成功改掉,pa执政相当于一个后门,间接改a
const int *pb = &a;//给*pb加上const属性
*pb = 0;//这时候通过*pb来改a时就会报错
return 0;
}
其中,const虽然有了常属性,但本质上还是属于变量范畴(通常叫:常变量),const对于变量只是在语法上做了限制。
当const放在不同位置时,起到的作用也是不同的
const int a = 10;//不能通过a改变a变量的值
const int *pa = &a;//不能通过*pa改变a的值,但pa指针可以被直接改变
int const *pb = &a;//pb和pa的性质相同
int *const pc = &a;//可以通过*pc改变a的值,但pc的值不能直接改变
const int * const pd = &a;//不可通过*pd改变a,同时也不能直接改变pd
注意观察,其实不难发现,根据const所在的位置,就可以判断其在哪个变量上做了限制了。
比的是地址的高低(感觉没啥讲的)
我们在C语言学习过程中,肯定不免听到别人说“野指针”这三个字,现在我们来聊聊野指针吧。
现在我来给大家生成一个野指针
#include
int main()
{
int *ptr;
*ptr = 20;//非法访问
return 0;
}
其中ptr就是一个野指针,也就是所指向不可知地址的指针 ,由于生成的指针ptr指向方向不定,所以在访问或改变其指向的值的时候是极其不安全的,因为没有人会知道他指向的是哪一块空间。
避免生成野指针:
1.给指针初始化
2.不知道初始化什么,给指针赋NULL
3.指针用完记得置NULL
4.避免返回局部变量的地址
(在局部变量销毁,空间释放后局部变量指针就成为野指针而引发错误)
如果你在为害怕误用指针而烦恼,建议来看看assert吧,它能一定程度上帮助你判断指针是否可以使用,在assert.h中定义了宏assert();
#include
#include
int main()
{
int *pa = NULL;
assert(pa != NULL);//当pa为空指针时报错
return 0;
}
在函数调用过程中,有时候会遇到传递参数的问题,但有时候单纯传递数却无法解决一些问题
比如:写一个函数用来交换两个变量的值
#include
void Swap1(int x,int y)
{
int tmp = x;
x = y;
y = tmp;
}
void Swap2(int* pa,int* pb)
{
int tmp = *pa;
*pa = *pb;
*pb = tmp;
}
int main()
{
int a = 10;
int b = 20;
Swap1(a,b);
printf("%d %d\n",a,b);//这里将会打印 10 20,可以看到变量并没有交换
Swap2(&a,&b);
printf("%d %d\n",a,b);//这里打印 20 10,变量成功交换
return 0;
}
可以看出,当要用来改变传递变量的值的时候,可以传递它的地址,而直接传递其数值就达不到这样的效果。
传址:让函数与主调函数建立联系
指针的内存用来存放地址数据,那么能否创建一个指向指针变量的指针呢?
当然可以,见代码
#include
int mian()
{
int a = 10;
int * p = &a;//取a的地址
int ** pp = &p;//取指针p的地址,pp为二级指针
int *** ppp = &pp;//取二级指针pp的地址,ppp为三级指针(用的很少)
//其中**pp==a
printf("%d\n",**p);
return 0;
}
什么是指针数组,字面意思:存放指针的数组
#include
int main()
{
int a = 10;
int b = 20;
int c = 30;
int* arr[3] = {&a,&b,&c};//其中arr数组的每个元素都为int类型的指针变量
return 0;
}
也是字面意思:指向数组的指针
注:在看这些名字的时候,可以注意一下它的命名特点和方式,比如数组在后其本质就是数组,指针在后其就是指针。
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int (*parr)[10] = &arr;//其中parr是数组指针,&arr取得是整个数组的指针
//[]的优先级 > *
//*代表parr是一个指针,而[10]代表parr指向的数组有十个元素,每个元素是int类型
//parr + 1会跳过整个数组
//用parr访问元素(*parr)[i]
#include
void print1(int arr[3][5],int r,int c){....}//这两种传参方式都对
void print2(int (*arr)[5],int r,int c){....}
int main()
{
int arr[3][5];
print(arr,3,5);//二维数组传参
return 0;
}
字面意思:指向函数的指针
#include
int Add(int x,int y)
{
return x + y;
}
int main()
{
int (*pf)(int,int)=&Add;
int ret = (*pf)(4,9);
int (*pf2)(int,int) = Add;//Add本身就可以作为地址赋给pf2
int ret2 = (*pf2)(4,9);
int ret3 = pf(4,9);//调用是pf可不用解引用
printf("%d %d %d\n",ret,ret2,ret3);//打印的三个值是相同的
return 0;
}
总的来说其实还是要看名字,就是存放函数指针的数组
使用举例
#include
int add(int x, int y)
{
return x + y;
}
int sub(int x, int y)
{
return x - y;
}
int mul(int x,int y)
{
return x * y;
}
int div(int x, int y)
{
return x / y;
}
void menu()
{
printf("1.add\n");
printf("2.sub\n");
printf("3.mul\n");
printf("4.div\n");
printf("0.exit\n");
}
int main()
{
int a, b;
menu();
int input;
int (*parr[5])(int, int) = { NULL,add,sub,mul,div };
//上面的函数指针数组里分别放了四个函数指针
scanf("%d", &input);
while (input < 5 && input>0) {
scanf("%d%d", &a, &b);
int ret = parr[input](a, b);//通过调用函数指针数组来间接访问各个函数
printf("%d", ret);
}
return 0;
}
以上的代码,完成了一个简单的计算器,同时规避了一堆if else的冗杂,这便是函数指针数组的一个妙用。
以上讲了指针的基本类型,但是关于指针的细节还有很多很多,再往后深入就是要注意指针的不同类型和运算,才是深入了解和应用指针的关键。那么今天的指针内容就先分享到这里,如果感觉对你还有帮助的话,记得留一个小小的赞再走哦!