指针初阶(超详解)

指针初阶

    • 1.指针是什么
    • 2.指针和指针类型
      • 2.1 指针+-整数
      • 2.2 指针的解引用
    • 3.野指针
      • 3.1 野指针成因
      • 3.2如何避免野指针
    • 4.指针运算
      • 4.1 指针+-整数
      • 4.2 指针-指针
      • 4.3 指针的关系运算
    • 5.指针和数组
    • 6.二级指针
    • 7.指针数组

1.指针是什么

指针是什么?

指针理解的2个要点:

  1. 指针是内存中一个最小单元的编号,也就是地址
  2. 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量
    (单元编号 == 地址 == C语言中:指针)

那我们可以这样理解:

内存:
在这里插入图片描述

指针变量
我们可以通过&(取地址操作符)取出变量的内存其实地址,把地址可以存放到一个变量中,这个
变量就是指针变量

int main()
{int a = 10;//是向内存中的栈区空间申请4个字节的空间,这4个字节用来存放10这个数值int* pa = &a;//这里我们对变量a,取出它的地址,可以使用&操作符。//a变量占用4个字节的空间,这里是将a的4个字节的第一个字节的地址存放在p变量中,p就是一个之指针变量。//pa 0x000112233return 0;
}

总结:

指针变量,用来存放地址的变量。(存放在指针中的值都被当成地址处理)。
那这里的问题是:
1.一个小的单元到底是多大?(1个字节)
2.如何编址?
经过仔细的计算和权衡我们发现一个字节给一个对应的地址是比较合适的。
对于32位的机器,假设有32根地址线,那么假设每根地址线在寻址的时候产生高电平(高电压)和低电平(低电压)就是(1或者0);
那么32根地址线产生的地址就会是:
在这里插入图片描述
在这里插入图片描述

  1. 内存被划分为一个个内存单元,每个内存单元的大小是1个字节
  2. 每个字节的内存单元都有一个编号,这个编号就是地址,地址就是C语言中的指针
  3. 地址要储存的话放在指针变量中
  4. 每个内存单元都有唯一的地址来标识
  5. 在32位机器上地址的大小是4个字节,所以指针变量的大小也是4个字节
    同理:在64位机器上地址的大小是8个字节,所以指针变量的大小也是8个字节

2.指针和指针类型

我们看以下代码:

int main()
{printf("%d\n", sizeof(char*));printf("%d\n", sizeof(short*));printf("%d\n", sizeof(int*));printf("%d\n", sizeof(double*));
}

在这里插入图片描述
在这里插入图片描述

2.1 指针±整数

代码演示:

int main()
{int a = 0;int* pa = &a;char* pc = &a;printf("%d\n", pa);printf("%d\n", pa+1);printf("%d\n", pc);printf("%d\n", pc+1);
}

运行结果:
在这里插入图片描述

指针类型是有意义的
指针类型决定了指针+1/-1跳过几个字节
char的指针+1跳过1个字节
short
的指针+1跳过2个字节
int的指针+1跳过4个字节
double
的指针+1跳过8个字节

2.2 指针的解引用

在这里插入图片描述
在这里插入图片描述
指针类型是有意义的
指针类型决定指针解引用时访问几个字节
比如:char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。

3.野指针

概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)

3.1 野指针成因

  1. 指针未初始化
    代码演示:
int main()
{int* p;//未初始化*p = 20;return 0;
}
  1. 指针访问越界
    代码演示:
int main()
{int arr[10] = { 0 };int* p = arr;int i = 0;for (i = 0; i <= 10; i++)//指针访问越界{//当指针指向的范围超出数组arr的范围时,p就是野指针*p = -1;p++;}return 0;
}
  1. 指针指向的空间释放
    代码演示:
int* test()
{int a = 10;//0x0040fe44return &a;//地址已返回出函数销毁
}
int main()
{//0x0040fe44int* p = test();//p就是野指针printf("%d", p);return 0;
}

3.2如何避免野指针

  1. 指针初始化
    在这里插入图片描述

  2. 小心指针越界

  3. 指针指向空间释放即使置NULL

  4. 避免返回局部变量的地址

  5. 指针使用之前检查有效性

int main()
{int* p = NULL;*p = 20;//空指针不允许访问//....if (p != NULL){//...确保非控指针在使用}return 0;
}

4.指针运算

4.1 指针±整数

代码演示1:

int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };// 0 1 2 3 4 5 6 7 8 9 //是指针打印数组内容int* p = arr;int i = 0;//p --> arr//p == arr//p+i == arr+i//*(p+i) == *(arr+i) == arr[i]//*(arr+i) == arr[i]//*(i+arr) ==i[arr]//[]仅为操作符for (i = 0; i < 10; i++){printf("%d ", *(p + i));//printf("%d ", *(arr + i));//printf("%d ", arr[i]);//printf("%d ", i[arr]);//p指向的是数组首元素//p+i是数组下标为i的元素地址//p+i起始时跳过i*sizeof(int)个字节}return 0;
}

运行结果:
在这里插入图片描述
代码演示2:

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

分析:
在这里插入图片描述

4.2 指针-指针

案例1:

int main()
{int arr[10] = { 0 };////指针-指针的前提:两个指针指向同一块区域,指针类型时相同的//指针-指针差值的绝对值,指针和指针之间的元素个数//printf("%d\n", &arr[9] - &arr[0]);printf("%d\n", &arr[0] - &arr[9]);return 0;
}

运输结果:
在这里插入图片描述
案例2:

//模拟实现strlen
//1.计算器
//2.递归
size_t my_strlen(char* str)
{char* start = str;while (*str){str++;}return str - start;
}
int main()
{char arr[] = "abcdef";size_t len = my_strlen(arr);printf("%zd\n", len);return 0;
}

运输结果:
在这里插入图片描述

4.3 指针的关系运算

代码演示1:

int main()
{float values[N_VALUES];float* vp;for (vp = &values[N_VALUES]; vp > &values[0];){*--vp = 0;}return 0;
}

分析:
在这里插入图片描述
代码2(代码1修改如下):

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

分析:
在这里插入图片描述

实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。
标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与
指向第一个元素之前的那个内存位置的指针进行比较。

5.指针和数组

  • 指针就是指针,指针变量就是一个变量,存放的地址,指针变量的大小是4/8
  • 数组就是数组,可以存放一组数,数组的大小取决于元素类型和个数
  • 数组的数组名是首元素地址
  • 通过一个指针可以访问数组元素

数组名表示数组首元素地址,但有两个例外

  • sizeof(数组名),数组名单独放在sizeof内部,数组名表示整个数组,计算的是数组的大小,单位是字节
  • &数组名,数组名表示整个数组,取出的事数组的地址,数组的地址和数组首元素地址的值是一样的,但是类型和意义不同

代码案例1:

int main()
{int arr[10] = { 0 };printf("%d\n", arr);printf("%d\n", arr+1);//printf("%d\n", &arr[0]);printf("%d\n", &arr[0]+1);//printf("%d\n", &arr);printf("%d\n", &arr+1);//printf("%d\n", sizeof(arr));
}

在这里插入图片描述
代码案例二:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
int main()
{int arr[10] = { 0 };int* p = arr;int i = 0;for (i = 0; i < 10; i++){printf("%p===%p\n", p+i, arr+i);}return 0;
}

运行结果:
在这里插入图片描述

6.二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?
这就是 二级指针
在这里插入图片描述
二级指针变量是存放一级指针变量地址的

代码案例1:

int main()
{int a = 10;int* pa =  &a;//pa是指针变量,一级指针变量int* * ppa = &pa;//ppa指针变量,二级指针变量**ppa = 20;//*ppa得出pa-->**ppa==*pa==aprintf("%d", a);return 0;
}

运行结果:
在这里插入图片描述

  • *ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa
  • **ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作: *pa ,那找到的是 a .

代码案例2:

7.指针数组

指针数组是指针还是数组?

是数组。是存放指针的数组。
数组我们已经知道整形数组,字符数组。
int arr1[5];
char arr2[5]
指针数组:
int* arr3[5];
在这里插入图片描述

代码案例:

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

运行结果:
在这里插入图片描述

不知不觉,指针初阶以告一段落。通读全文的你肯定收获满满,让我们继续为C语言学习共同奋进。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/18089.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Class.forName和ClassLoader区别

【反射中&#xff0c;Class.forName和ClassLoader区别】_lfsun666的博客-CSDN博客

docker启动报错:Cannot connect to the Docker daemon

Couldn’t connect to Docker daemon at httpdocker://localunixsocket - is it running? 解决方式&#xff1a;正确的是将当前用户加入docker组解决步骤 sudo groupadd docker #添加docker用户组 sudo gpasswd -a $USER docker #将登陆用户加入到docker用户组中 ne…

SDWAN的技术背景及其发展历程

一、SDWAN&#xff08;软件定义广域网&#xff09;技术的背景 传统广域网&#xff08;WAN&#xff09;主要基于专线连接&#xff0c;例如MPLS&#xff08;多协议标签交换&#xff09;网络&#xff0c;用于连接企业的分支机构和总部。这些传统的WAN网络在过去是为了满足数据中心…

BladeX框架开源-工作-笔记-Docker部署-Jenkins配置

BladeX框架开源-工作-笔记-Docker部署-Jenkins配置 文章目录 BladeX框架开源-工作-笔记-Docker部署-Jenkins配置第一章-概要-BladeX框架简介与git地址第二章-BladeX框架前后端项目Docker部署与DockerFile配置文件2.1-开始部署阶段&#xff0c;默认服务器上面已有Nacos服务2.2-采…

maven下载按照及初次使用相关配置

maven下载按照及初次使用相关配置 一、下载 与安装 依赖Java&#xff0c;需要配置JAVA_HOME设置MAVEN自身的运行环境&#xff0c;需要配置MAVEN_HOME测试环境配置结果 MVN测试成功&#xff01;&#xff01;&#xff01; 二、本地仓库配置 Maven启动后&#xff0c;会自动保…

基于Amoeba读写分离(三十六)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 今天要学的是基于Amoeba读写分离。Amoeba是一个开源的关系型数据库管理系统&#xf…

Spark性能调优之数据序列化

前言 在使用Spark进行数据开发的时候,避不开的一个问题就是性能调优。网上一搜一大堆所谓的调优策略很多作者自己都不知所云,导致读者看了后只会更加困惑。我们在研究一个技术的时候第一手资料永远都请参考官网,官网对性能优化不一定是最全甚至最优,但是可以解决大部分问题…

六、初始化和清理(4)

本章概要 数组的初始化 动态数组创建可变参数列表 枚举类型 数组初始化 数组是相同类型的、用一个标识符名称封装到一起的一个对象序列或基本类型数据序列。数组是通过方括号下标操作符 [] 来定义和使用的。要定义一个数组引用&#xff0c;只需要在类型名加上方括号&#xf…

NestJS 的 拦截器 学习

拦截器会用到RxJs&#xff0c;所以在学习拦截器之前可以先了解一下它。 拦截器是使用Injectable()装饰器装饰的类并且实现了接口NestInterceptor。 拦截器受到 AOP(面向切面编程)技术的启发&#xff0c;具有如下的功能&#xff1a; 在方法执行之前/之后绑定额外的逻辑转换函…

express学习笔记2 - 三大件概念

中间件 中间件是一个函数&#xff0c;在请求和响应周期中被顺序调用&#xff08;WARNING&#xff1a;提示&#xff1a;中间件需要在响应结束前被调用&#xff09; 路由 应用如何响应请求的一种规则 响应 / 路径的 get 请求&#xff1a; app.get(/, function(req, res) {res…

以beam search为例,详解transformers中generate方法(上)

以beam search为例&#xff0c;详解transformers中generate方法&#xff08;上&#xff09; 1. generate的代码位置2. GenerationMixin概览3. generate签名4. generate过程4.1 读取并更新generation config4.2 补充没有传入的参数4.3 定义模型输入4.4 定义模型的其他参数4.5 对…

Mysql存储过程基本语法

目录 存储过程MYSQL基础语法游标&#xff08;作用范围存储过程&#xff09;事务&#xff08;Demo&#xff09;其他操作Demo 存储过程 MYSQL 基础语法 #创建存储过程 #参数格式 #in输入参数 out输出参数 inout既可以输入也可以输出 create or replace procedure 存储过程名…

react常用知识点

React是一个用于构建用户界面的JavaScript库。以下是React常用的知识点&#xff1a; 组件&#xff1a;React将用户界面分解成小而独立的组件&#xff0c;每个组件都有自己的状态和属性&#xff0c;并且可以通过组合这些组件来构建复杂的用户界面。 // 函数组件示例 function We…

torch.load 报错 ModuleNotFoundError 或 AttributeError

Python 3.11.3 (main, Apr 7 2023, 19:25:52) [Clang 14.0.0 (clang-1400.0.29.202)] on darwin Type "help", "copyright", "credits" or "license" for more information.正常情况下&#xff0c;我们会使用 torch.save 保存模型的 …

向 Maven 中央仓库上传一个修改过的基于jeecg的autoPOI的 jar包记录

1、注册https://issues.sonatype.org/账号 下面就代表注册好了&#xff0c;同时提交的工单也通过了 2、这里主要是goupId 需要进行认证&#xff0c;需要到域名注册商近一个txt的解析&#xff0c;以便确保这个是你的 通过下面来验证你的域名信息&#xff0c;这里主要是上面的工…

git命令分类合集

配置 git config --global user.name <name>&#xff1a;设置全局用户名 git config --global user.email <email>&#xff1a;设置全局用户邮箱 git config --global core.editor <editor>&#xff1a;设置全局文本编辑器创建与克隆仓库 git init&#xf…

面试题:说一说深拷贝和浅拷贝?

JavaScript中存在两大数据类型&#xff1a; 基本类型 和 引用类型 基本类型数据保存在在栈内存中 引用类型数据保存在堆内存中&#xff0c;引用数据类型的变量是一个指向堆内存中实际对象的引用&#xff0c;存在栈中 深拷贝和浅拷贝都只针对于引用类型。 一、 浅拷贝&#xff1…

【2023年11月第四版教材】《第1章-信息化发展之<3 现代化创新发展>》

第1章-信息化发展之&#xff1c;3 现代化创新发展&#xff1e; 3 现代化创新发展3.1 农业现代化3.2 两化融合&#xff08;17下2&#xff09;&#xff08;18下2&#xff09; &#xff08;22下12)3.3 智能制造3.4 消费互联网 3 现代化创新发展 3.1 农业现代化 要素具体内容农业…

Cpp7 — 继承和多态

继承 -------- 面向对象的三大特性之一 面向对象的三大特性&#xff1a;封装、继承、多态 封装&#xff1a;把数据和方法都封装在一起&#xff0c;想给你访问的变成共有&#xff0c;不想给访问的&#xff0c;写成私有。 继承&#xff1a;继承是类设计层次的复用 多态&#…

[SQL挖掘机] - 索引

介绍: 当你在数据库中进行查询时&#xff0c;索引是一种用于提高查询性能的重要工具。索引是对表中的一列或多列进行排序的数据结构&#xff0c;它可以快速定位到满足特定条件的记录&#xff0c;从而减少了查询所需的时间和资源。 在数据库中使用索引的主要好处包括&#xff…