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

【C语言】初阶指针(指针及其类型以及野指针)

简单不先于复杂,而是在复杂之后。

89efcc89ac61428db4d5b6639b2bd948.jpeg

目录

1. 指针是什么?

2. 指针和指针类型 

2.1  指针+-整数

2.2 指针的解引用 

3. 野指针 

3.1 野指针成因 

3.2 如何规避野指针 


1. 指针是什么?

指针理解的两个要点:

1. 指针是内存中最小单元的编号,也就是地址。

2. 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量。

总结:指针就是地址,口语中说的指针通常是指针变量。

 

int main()
{
int a = 10;//a是整型变量,占用4个字节的内存空间
int* pa = &a;//pa是一个指针变量,用来存放地址的
//取地址时取的是第一个字节的地址return 0;
}

总结:

指针变量,用来存放地址的变量。(存放在指针中的值都被当成地址处理。)

  • 一个小的内存单元多大?

            ---1byte

  • 如何编址?

经过仔细的计算和权衡我们发现一个字节给一个对应的地址是比较合适的。

对于32位的机器,假设有32根地址线,那么假设每根地址线在寻址的时候产生高电平(高电压)和低电平(低电压)就是1或者0;

那么32根地址线产生的地址就会是:

00000000 00000000 00000000 00000000

00000000 00000000 00000000 00000001

...

11111111 11111111 11111111 11111111

 总结:

1. 指针是用来存放地址的,地址是唯一标识一块地址空间的。

2. 指针的大小在32位平台是4个字节,在64位平台是8个字节。

 

C语言的%zu是用来格式化输出size_t类型数据的格式控制符。

size_t是C标准库中定义的一个无符号整型类型,用于表示内存空间大小和数组索引等非负整数值。在C语言中,sizeof操作符用于计算指定类型或表达式的大小(即占用的字节数),并返回一个size_t类型的值。

需要注意的是,%zu只能用于格式化输出size_t类型的值,不能用于其他类型的数据。

如果把一个非size_t类型的数据使用%zu格式控制符进行输出,可能会导致输出结果不正确,甚至导致程序崩溃。

2. 指针和指针类型 

变量有不同的类型,整型、浮点型等。

指针也有类型。

当有这样的代码: 

 

 

 

 

 

结论:指针类型决定了指针在被解引用的时候访问几个字节

1. 如果是int*的指针解引用访问4个字节

2. 如果是char*的指针,解引用访问1个字节

在这段代码中,整型指针变量pa和字符型指针变量pc所指向的地址是相同的,即它们都指向变量a的内存地址。

这是因为在C语言中,变量在内存中的存储方式是连续的,而不同类型的指针变量只是指向这些内存地址的方式不同。

在这段代码中,变量a在内存中的存储方式是以4个字节为一个单位进行存储,因此它的地址是一个整型地址。

当将变量a的地址赋给整型指针变量pa时,pa所指向的地址与a的地址是相同的,因为它们都是整型地址。

当将变量a的地址强制转换为字符型指针类型并赋给字符型指针变量pc时,pc所指向的地址也与a的地址相同,这是因为强制类型转换只改变了指针变量的解释方式,而并没有改变指针变量所指向的地址。

因此,虽然整型指针变量和字符型指针变量的类型不同,但它们所指向的地址是相同的。

2.1  指针+-整数

 

 

总结:指针的类型决定了指针向前或向后走一步(+1-1操作)有多大,也就是指针的步长。

2.2 指针的解引用 

 

 

 总结:

 指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。

 比如:char* 的指针解引用就只能访问一个字节,而int* 的指针解引用就能访问四个字节。

3. 野指针 

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

3.1 野指针成因 

1. 指针未初始化

int main()
{
int* p;//没有初始化,就意味着没有明确的指向
//一个局部变量不初始化的话,放的是随机值:0xcccccccc*p = 10;//非法访问内存,这里的p就是野指针return 0;
}

 2. 指针越界访问

int main()
{
int arr[10] = { 0 };
int* p = arr;//&arr[0]
int i = 0;
for (i = 0; i <= 10; i++)
{
*p = i;
p++;
}
//当指针指向的范围超出数组arr的范围时,p就是野指针
return 0;
}

3. 指针指向的空间释放(动态内存开辟时讲解)

int* test()
{
int a = 10;
return *a;
}
int main()
{
int*p = test();return 0;
}

3.2 如何规避野指针 

  • 指针初始化
  • 小心指针越界
  • 指针指向空间释放时及时置NULL 
  • 避免返回局部变量的地址
  • 指针使用之前检查有效性

 

零地址禁止访问,指针没有指向有效空间。

这段代码有一个明显的问题,就是在函数test()中声明的变量a是一个局部变量,在函数执行完毕后将被销毁。

因此,返回该变量的地址是不安全的,因为该地址将指向一块已经被销毁的内存。

这将导致指针变量p所指向的地址变成了一个野指针,可能会引发不可预测的错误,这是一种非常危险的行为。

如果想要正确地使用返回指针,可以考虑使用动态内存分配函数malloc()来分配一块内存,并将变量的值存储在该内存块中,然后返回该内存块的地址。

动态内存分配在之后的博客会有所体现。

这样可以确保返回的地址始终有效,并且可以安全地使用指针变量。

 

int main()
{
int a = 0;
int* p = &a;
*p = 20;
//null -> 0int* p2 = NULL;
*p2 = 100;//errint* p3 = NULL;if (p3 != NULL)
{
*p3 = 100;//ok
}return 0;
}

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

相关文章:

  • UDS统一诊断服务【六】访问时序参数0X83服务
  • Linux应用编程(文件属性与目录)
  • 第十四届蓝桥杯嵌入式详解
  • 新建论文三线表模板,一键格式刷
  • 攻防世界-web2(逆向加密算法)
  • C语言学习1--------Visual Studio集成开发环境的搭建
  • 腾讯云轻量应用服务器搭建网站教程(WordPress为例)
  • mac上的PCB设计软件现状
  • 【面试题】JavaScript 你常用的 函数有哪些呢? (12个)
  • Java集合——Set接口学习总结
  • 2023最全的自动化测试入门基础知识(建议收藏)
  • 【RabbitMQ】SpringBoot整合RabbitMQ、实现RabbitMQ五大工作模式(万字长文)
  • ES6(函数扩展、数组扩展)
  • postman汉化教程
  • java day8
  • 口令暴力破解--Telnet协议暴力破解、数据库暴力破解与远程桌面暴力破解
  • [译]什么是SourceMap
  • saga模式、Seata saga模式详解
  • java开发工程师碰到技术难题怎么办?我来聊聊我的做法
  • 高比例可再生能源电力系统的调峰成本量化与分摊模型(Matlab代码实现)
  • Hive安装与操作
  • oracle centos7安装Oracle12(附oracle所有版本安装包)
  • ESP32学习二-更新Python版本(Ubuntu)
  • 【19】核心易中期刊推荐——人工智能 | 遥感信息处理
  • MySQL运维10-MySQL数据的导入导出
  • 全国计算机等级考试——二级JAVA完整大题题库【五十三道】
  • 【算法题解】24. 模拟机器人行走
  • PyTorch 深度学习实战 |用 TensorFlow 训练神经网络
  • 【进阶C语言】静态版通讯录的实现(详细讲解+全部源码)
  • 【JavaWeb】后端(Maven+SpringBoot+HTTP+Tomcat)