当前位置: 首页 > news >正文

详解指针(2)(初阶版)

前言:内容包括:指针运算,指针和数组,二级指针,指针数组

详解指针(1)(点击即跳转)

 part 1:指针运算

1 指针+-整数

以如下代码为例:初始化数组内容全为0

#define N_VALUES 5
float values[N_VALUES];
float *vp;for (vp = &values[0]; vp < &values[N_VALUES];)
{*vp++ = 0;
}

*vp++:后置++只作用于vp指针,后置++是先使用再++,故而此操作是*vp=0,vp++

vp++:跳过一个float类型的大小,由于数组中的元素都是float类型,故而vp++,跳过一个元素

最终for循环结束的条件是vp=&values[N_VALUES],即指针vp指向了最后一个元素后面的那一块空间,但需要注意,这不是越界访问,因为vp指针只是指向了那一块不属于数组的空间,却并没有访问

2 指针-指针 

注意:两个指针必须指向同一块区域,两个指针的类型必须相同

           指针-指针就是指针和指针之间的元素个数  (指针就是地址)

           但有正负之分:小地址-大地址:元素个数带负号

                                     大地址-小地址:元素个数是正数

以如下代码为例:求字符串长度

int my_strlen(char *s)
{char *p = s;while(*p != '\0' )p++;return p-s;
}

字符型指针接收字符串的首个字母的地址

 故而求出的字符串长度是6

3 指针的关系运算

 以如下代码为例:初始化数组内容为0

#define N_VALUES 5
float values[N_VALUES];
float *vp;for(vp = &values[N_VALUES]; vp > &values[0];)
{*--vp = 0;
}

*--vp:vp先-1,然后对vp解引用:*vp

或许有人会这么简化以上代码:可能更容易理解

for(vp = &values[N_VALUES-1]; vp >= &values[0];vp--)
{*vp = 0;
}

 

虽然再绝大部分编译器上这样做是没有问题的,但是我们还是要避免这样些,因为标准并不保证它可行 

标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与
指向第一个元素之前的那个内存位置的指针进行比较

 黑色的是属于数组的空间,蓝色是不属于数组的空间

标准规定:指针p1可以和p2比较,但是指针p1不能与p3比较

故而第二种即使容易理解,但是我们要避免写出这种代码

part 2:指针和数组

指针可以指向数组元素,故而可以通过指针来访问数组的每一个元素

int main()
{int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };int *p = arr; //指针存放数组首元素的地址int sz = sizeof(arr) / sizeof(arr[0]);//数组元素个数int i = 0;for (i = 0; i<sz; i++){printf("%d ", *(p + i));}return 0;
}

p+i:指向的是arr数组中下标为i的元素

*(p+i)就可以拿到下标为i的元素

part 3:二级指针

二级指针存放的是一级指针的地址

一级指针存放的是变量的地址

这里所有的地址都是假设的,现有变量a=10,一级指针pa,二级指针ppa

int a = 10;
int*pa = &a;
int**ppa =&pa

int*pa : *表示pa是指针,int表示pa指向的对象(a)是int类型

int**ppa:*表示ppa是指针,int*表示ppa指向的对象(pa)是int*类型(即一级指针) 

 *pa(对一级指针解引用一次)可以找到a

**ppa(对二级指针解引用两次)可以找到a

**ppa = 30;
//等价于*pa = 30;
//等价于a = 30;

*ppa:找到的是pa,访问到的是pa

int b = 20;
*ppa = &b;//等价于 pa = &b;

part 3:指针数组

指针数组是数组,是存放指针的数组

int* arr3[5];

arr3是一个数组,有五个元素,每个元素是一个整形指针
实例:使用指针数组模拟二维数组

int main()
{int arr1[] = { 1,2,3,4 };int arr2[] = { 5,6,7,8 };int* str[] = { arr1,arr2 };int i = 0;for (i = 0; i < 2; i++){int j = 0;for (j = 0; j < 4; j++){printf("%d ", str[i][j]);}printf("\n");}return 0;
}

模拟两行4列的二维数组,每一行都是一个一维数组

str[i][j]:str[i]就是某个一维数组的数组名,j就是一行中某个元素的下标

             就像我们平常访问一个一维数组一样,现有int arr[]={1,2,3}

            我们是通过arr[i]的方式访问arr数组的每个元素的

             arr[0]:第一个元素,arr[1]:第二个元素,arr[2]:第三个元素

            我们可以发现,数组名[下标] :数组中的某个元素

http://www.lryc.cn/news/3266.html

相关文章:

  • 超详细讲解字符串查找函数(保姆级教程!!!)
  • LeetCode-1138. 字母板上的路径【哈希表,字符串】
  • Vue 可配置化的路由缓存(Vu2 Vue3)
  • Linux VPU驱动
  • spring 笔记
  • Java日志框架学习
  • 基础面试题:堆和栈的区别
  • (干货教程)在VSCode并使用chatgtp插件编写CC++语言程序
  • 【思维模型】概率思维的价值:找到你的人生算法,实现阶级跃迁!
  • SpringBoot + kotlin/java + Mybatis-Plus +Sqlite + Gradle多模块项目
  • Docker 容器与容器云读书笔记(一)
  • 软件设计(九)
  • FoveaBox原理与代码解析
  • Linux内核启动(1,0.11版本)启动BIOS与加载内核
  • python制作贪吃蛇小游戏,畅玩无限制
  • MySQL-InnoDB数据页结构浅析
  • Java、JSP职工人事管理系统设计与实现
  • 数据结构与算法这么难,为什么我们还要学习?
  • 剑指 Offer 52. 两个链表的第一个公共节点
  • 可以写进简历的软件测试电商项目,不进来get一下?
  • 蓝桥杯-算法-印章问题
  • 戴尔游匣G16电脑U盘安装系统操作教程分享
  • 2023数学建模美赛赛题思路分析 2023美赛 美国大学生数学建模数模
  • vue3与vue2的对比
  • 史上最全软件测试工程师常见的面试题总结(百度、oppo、中软国际、华为)备战金三银四
  • “深度学习”学习日记。卷积神经网络--用CNN的实现MINIST识别任务
  • JavaWeb--JDBC练习
  • 【LeetCode】2335. 装满杯子需要的最短总时长
  • Android 12.0 通过驱动实现禁用usb鼠标和usb键盘功能
  • C++入门——内存管理