三.指针运算
指针的运算有:指针加减整数,指针减指针,指针大小关系比较。
指针加减整数前面讲过了,就是什么样的真正类型,跳过多少个内存单元。我们主要是看一下指针减去指针和指针关系比较。
指针减去指针表示的是之间的元素个数;这里是9个元素。可以这样理解&arr[9]-&arr[0]下标9减0是9。
补充:指针相减的绝对值才是元素个数,因为指针相减可以是负数。指针与指针相减的前提是指针类型相同、指向同一块空间,基本用在数组里面。
指针的关系运算:
#include <stdio.h> #define n_ele 5 //方式一: int main() { float* vp = NULL; float varr[n_ele]; for(vp = &varr[n_ele]; vp > &varr[0];)//第三个条件省略 { *--vp = 0; } return 0; } //方式二: int main() { float* vp = NULL; float varr[n_ele]; for(vp = &varr[n_ele-1]; vp >= &varr[0]; vp--) { *vp = 0; } return 0; }
使用两个指针的关系比较充当for循环里的入口条件测试。
第一种方式vp第一次从下标为5的位置和下标为0的位置比较,注意下标为5的位置其实不在数组内;*--vp是先把指向数组下标为5的指针变成指向数组下标为4的指针,再*解引用找到内存空间改成数值0,没有越界操作。
第二种方式第一次从下标为4开始比较,最后停止的时候,是用指向相当于数组下标为-1的指针和数组下标为0的比较,不符合循环条件,退出循环,也没有越界访问。这两种方式完成的功能都一样,把数组元素都赋值成0,哪一种好呢?
其实第一种方式会更好,因为他是C语言标准允许的。C语言标准规定:允许指向数组元素的指针和指向数组最后一个元素的下一个位置的指针进行比较,但不允许和指向第一个元素前一个位置的指针进行比较。
四.指针和数组的关系
相信在前面,读者多多少少感觉到了数组的下标访问,和指针运算之间有种莫名其妙但又说不出来的感觉。其实数组的和指针是非常相似的!
以前说过数组名一般表示的是首元素地址,既然是地址,其实也就是指针!但和指针不同的是arr它表示的是首元素地址,也就是说 arr = arr + 1;这句代码试图将arr指向第二个元素,这是不允许的。如果是这种情况 :p = arr + 1;指针p在没有再次修改的情况下,都是指向数组第二个元素。
4.1下标与解引用的等价替换
arr[0]访问的是数组的第一个元素,用指针可以*(parr+0)访问;arr[1]访问的是数组的第二个元素,用指针可以*(parr+1)访问;所以下标访问实际上就是让arr(首元素地址)加上一个整数【指针加减整数运算】后,解引用访问。
总结:arr[i] = *(arr+i) = *(parr+i) = parr[i]
阴间写法:*(arr+i)和*(i+arr)道理是一样的,前面是arr[i],后面可以这样写i[arr]!!!
4.2指针数组
指针数组,存放指针的数组~
#include <stdio.h> int main() { char* arr[3] = {"hello,world","C so good","I like it"}; //字符串常量时首字母地址,也就是说"hello,world"实际上表示的是h的地址 //是字符的地址,用字符指针char* ,arr是一个接收字符指针的数组---指针数组。 return 0; }
五.二级指针
指针是存放变量的地址的,指针变量也是变量,它也有地址。存放一级指针变量地址的指针叫做二级指针。
一级指针pa的类型是int*,这颗*不是解引用的那颗星,而是用来标志pa是一个指针的,这是一种理解;另一种理解是*pa的类型是int,也就是说pa解引用后指向的是一个整型。
二级指针ppa的类型是int**。一样的道理,可以把靠近pa的一颗星看做是一个标志ppa是一个指针,然后是指向int*的;也可以是*pa后是一个一级整型指针。
指针的内容不止有这些,学习指针的路还很长,后面再回~
结语:希望读者读完有所收获!在学C的路上,祝福我们能越来越C!✔
读者对本文不理解的地方,或是发现文章在内容上有误等,请在下方评论区留言告诉博主哟~,也可以对博主提出一些文章改进的建议,感激不尽!最后的最后!
❤求点赞,求关注,你的点赞是我更新的动力,一起努力进步吧。