前言
这篇文章将简单的介绍指针的相关知识,定义,类型,指向内容,运算以及指针与数组和函数的联系。
定义
- 指针是一个变量
一个指针变量只能指向同一数据类型的变量。 - 指针只能存地址
- 指针占据8个字节
指针是几个字节与语言无关,而是跟系统的寻址能力有关,比如以前是16位系统,指针就是2个字节,32位则为4个字节,64位则为8个字节。 - 指针是一种保存变量地址的变量
注:计算字节的方法如下
例,int *a;
printf("%d\n",sizeof(a));
声明
- 指针的声明相对于普通变量的声明多了一个一元运算符“*”。
- 运算符“*”是间接寻址或者间接引用运算符。当它作用于指针时,将访问指针所指向的对象。
- p 是一个指针,保存着一个地址,该地址指向内存中的一个变量;*p则会访问这个地址所指向的变量。
- 声明一个指针变量并不会自动分配任何内存。
- 在对指针进行间接访问之前,指针必须进行初始化;或者是他指向现有的内存,或者给他动态分配内存,否则这个指针会变成野指针。
初始化
例:int a = 1;int *p = &a;
*:定义的时候表明是一个指针变量,使用的时候表示取地址的值。
&;取某一变量地址。
- 一定要进行指针初始化。
因为不初始化,它的指向就是不确定的,就有可能指向系统中某一重要的内存位置。
指针的类型
int **p;//p是一个指向整型数的指针的指针。
p与 * 结合,说是p是一个指针,然后再与 * 结合,说明指针所指向的元素是指针,然后再与int结合,说明该指针所指向的元素是一个整型数据。
指针的运算
加减运算
- 指针p和正整数n,
所以p +/- n 的实际值为p +/- n* sizeof(数据类型)
* m = * p1++;
先取值* p1,p再后移。
* m = (*p1)++;
先取值,再把这个值自增1
* m = * ++p1;
p先后移,再取值
比较运算
多用于比较两个指针指向的对象在内存中的位置关系和判断指针是否为空。
比较运算的结果为boolen型;true或false。
指针与数组
指针数组
是一个数组,数组中的每一个元素都是指针。
因为指针数组定义中的运算符“[]”的优先级高于“* ”,所以表达式* p1[n]等价于*(pl[n]),
其中pl[n]表示名为pl,包含n个元素的指针数组 。
一个常用用法是用来存储若干行长度不等的字符串。
数组指针
数组指针,是一个指针,它指向一个数组。
数组指针是指向数组的指针。
例:
int a[3][3],( * p)[3];
p= a;
(* p)[3]通知编译器,它是一个指向长度为3的整型数组的指针。
这样用指针访问其所指向的内存单元时可以用((p+i)+j)来表示a[i][j].
而如果
int a[3][3], * p;
p= a;
则需用 * (p+3*i+j)来表示a[i][j]。
指针与字符串
在西C++中,除了用来初始化数组的字符串,都称为字符串常量
可以对字符数组中数组元素的值进行修改,但不能对字符指针引用的字符串常量中的值进行修改
字符串常量在内存中,以‘\0’结尾.
指针与函数
指针作为函数参数
函数参数间的传递方式主要有两种方式,按值传递和按地址传递。
按值传递的方式是单向传递.即函数调用时只能将实参的值赋给相应的形参,而形参的改变对实参的值没有任何影响。
按地址传递的方式实际是通过指针传递,即传递指针的值,指针既可以作为函数的形参也可以作为函数的实参。
当指针作为函数参数时,在调用过程中实参将某地址值传递给形参,使实参和形参指向相同的内存地址,这样被调用函数能够改变实参指针所指向变量的值,但不能改变实参指针的值。
因此,如果从指针作为函数参数无法实现形参指针变量(地址)改变实参指针变量(地址)这个角度,按地址传递是一种特殊的按值传递方式。
当需要传递的数据存放在连续的内存空间(如数组),如果采用按值传递会产生大量调用函数的开销;如果采用按地址传递就可以只传递数据的起始地址.从而减少开销提高效率,尤其当处理字符串时,采用指针作参数是最直接、效率最高的。
指针函数
指针函数是指返回指针的函数。
当函数返回一个指针时,必须保证指针所指向的对象时实际存在的
指向函数的指针
指向函数的指针是专门用于存放该函数代码首地址的指针变量,一旦定义了某个指向函数的指针,它就与函数名一样,具有调用函数的作用。
数组名是一个表示地址的指针常量。
函数名实际也是一个指针常量,他指向该函数代码的首地址指向函数的指针就是专门存放函数代码的入口地址也可简称为函数指针
函数指针增加了函数调用的灵活性,函数指针不能进行++,--,+,-运算。
作者有话说
指针这个内容吧,只能看缘分了!