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

【C语言】“指针的运算”、“指针与数组”

文章目录

  • 一、指针运算
    • 1.指针 + - 整数
    • 2.指针-指针
    • 3.指针关系运算
  • 二、指针与数组
  • 三、二级指针
  • 四、指针数组
  • 完结


一、指针运算

指针可以进行±整数,指针-指针,还有关系运算,其他的运算会被编译器阻止。

1.指针 + - 整数

对指针进行±的时候一定要注意不要越界访问,不然会产生野指针。

void print(int* arr)
{for (int* i = arr; i < &arr[arr_lenght];){printf("%d", *i++);	//对指针进行++达到遍历数组的效果}
}int main()
{int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };print(arr);
}

在这里插入图片描述

  • int* i = arr(数组名为首地址),此时i = 0xffee0000 对int*类型进行±整数就是对地址进行±一个int类型大小,+1之后为0xffee0004,也就是arr[1]的地址,再+1就是arr[2]的地址,依次类推,再在 i 的前面加上 * 号,就可以用指针做到遍历数组。

2.指针-指针

指针+整数为地址,所以指针-指针也是一个整数

  • 应用场景举例:我们可以在求字符串长度的时候,用字符串结尾地址-字符串首地址,得到字符串长度。
int my_strlen(char *str)		
{	char *pstr = str;			//先将字符串首地址进行存储while(*pstr != '\0' )		//strlen不应计算\0,所以遇到\0停止{pstr++;					//将字符串地址进行++}return pstr-str;			//用\0位置的地址减去首地址则为字符串长度。
}

3.指针关系运算

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

在这里插入图片描述

正确

void print(int* arr)
{for (int* i = arr; i < &arr[arr_lenght];){printf("%d", *i++);	//对指针进行++达到遍历数组的效果}
}int main()
{int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };print(arr);
}

错误

  • 此种办法在一些编译器下可以运行,但是我们要避免这么写!
void print(int* arr)
{for (int* i = &arr[arr_lenght-1]; i >= &arr[0];){printf("%d", *i--);}
}int main()
{int arr[arr_lenght] = { 1,2,3,4,5,6,7,8,9,0 };print(arr);
}

二、指针与数组

代码示例:

#include <stdio.h>
int main()
{int arr[10] = {1,2,3,4,5,6,7,8,9,0};printf("%p\n", arr);printf("%p\n", &arr[0]);return 0;
}

在这里插入图片描述

​ 我们看到数组名的地址与首元素的地址相等,那么数组名就是首元素的地址,这是在大多数情况,不过也有例外。

  • 两种特殊情况,数组名代表整个数组

(1)当sizeof(数组名)

#include <stdio.h>
int main()
{int arr[10] = {1,2,3,4,5,6,7,8,9,0};int sz = sizeof(arr);printf("%d",sz);			//输出40
}

数组大小为40,40就是整个数组的大小

(2)当 &数组名 时

#include <stdio.h>
int main()
{int arr[10] = {1,2,3,4,5,6,7,8,9,0};printf("%p\n", arr);printf("%p\n", &arr);printf("%d\n", arr + 1);printf("%d\n", &arr + 1);
}

在这里插入图片描述

我们看到&arr+1加了28(10进制的40),加的其实就是整数数组的大小

三、二级指针

​ 我们知道指针存放的其实就是变量的地址,但实质上指针也是一个变量,那么就需要一个可以用来存储指针的变量,我们称它为二级指针。(存放二级指针还可以用三级指针,存放三级指针还可以….)。

代码示例

#include <stdio.h>int main()
{int a = 10;int *pa = &a;			//int 是指针指向变量的类型 *表示pa是个指针变量。int **ppa = &pa;		//int* 是指针指向变量的类型 *表示ppa是个指针。printf("%d",a);			//10printf("%d",*pa);		//10 要拿到一级指针的值需要解引用一次  printf("%d",**pa);		//10 要拿到二级指针的值需要解引用两次
}

在这里插入图片描述

  • pa存放的是a的地址,ppa存放的是pa的地址

四、指针数组

指针数组是存放指针的数组,注意要和数组指针区分开,数组指针是指存放数组的指针。

  • 数组不仅可以存储char、short、int等等的基本数据类型,还可以存储指针

#include <stdio.h>
int main()
{int a = 0;int b = 0;int c = 0;int* arr[3] = {&a,&b,&c};		//指针数组就是数组的每个位置都存储一个地址
}

完结

创作不易,还请各位小伙伴多多点赞👍关注✨收藏⭐

请添加图片描述

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

相关文章:

  • Linux高级命令之查找文件命令
  • PyCharm+Docker:打造最舒适的深度学习炼丹炉
  • 【mock】手把手带你用mock写自定义接口+mock常用语法
  • 2023 年腾讯云服务器CVM快速配置购买教程,新手上云必备!
  • opencv显示图像
  • C++:类和对象(中)
  • 53. 最大子数组和
  • 基于Java+SpringBoot+SpringCloud+Vue前后端分离医院管理系统设计与实现
  • QT基础入门【环境配置篇】linux桌面QT开发环境的构建以及问题解决
  • Linux系统之部署企业内部静态导航页
  • 2023备战金三银四,Python自动化软件测试面试宝典合集(四)
  • 算法训练营 day43 动态规划 不同路径 不同路径 II
  • 关联查询的SQL有几种情况
  • 查缺补漏三:事务隔离级别
  • 没有她的通讯录(C语言实现)
  • Spring Security 从入门到精通
  • 微信小程序Springboot vue停车场车位管理系统
  • 看完这篇 教你玩转渗透测试靶机vulnhub——Hack Me Please: 1
  • nodejs+vue地铁站自动售票系统-火车票售票系统vscode
  • Spring Security in Action 第十二章 OAuth 2是如何工作的?
  • 天工开物 #5 我的 Linux 开发机
  • 【沁恒WCH CH32V307V-R1开发板输出DAC实验】
  • Linux进程控制详解
  • C语言深度剖析之程序环境和预处理
  • 【Spark分布式内存计算框架——Spark Core】9. Spark 内核调度(上)
  • Vulkan教程(15): Graphics pipeline之Render passes(渲染通道)
  • 乐观锁、雪花算法、MyBatis-Plus多数据源
  • 详解Redisson分布式限流的实现原理
  • [python入门㊹] - python测试类
  • Web 框架 Flask 快速入门(二)表单