C语言基础:野指针、空指针、空悬指针

devtools/2025/1/11 20:53:00/

野指针、空指针、空悬指针

野指针

定义:只想一块未知区域(以及销毁或者访问受限的内存区域外的已存在或不存在的内存区域)的指针,被称作野指针。野指针是危险的。

危害:

① 引用野指针,相当于访问了非法的内存,常常会导致段错误(segmentation fault),也有可能编译运行不报错。

② 引用野指针,可能会破坏系统的关键数据,导致系统崩溃等严重后果

野指针产生的场景:

  1. 变量未初始化,通过指针访问该变量

     int a;int *p = &a;//p是野指针printf("%d\n",*p);//访问野指针,数据不安全
  2. 指针变量未初始化

     int *p; // p 是野指针printf("%d\n",*p);​int a = get();p = &a;
  3. 指针指向的内存空间被(free)回收了

     int *p = malloc(4);*p = 12;//此时的p不是野指针free(p);printf("%d\n",*p); // 此时的p就是野指针
  4. 指针函数中直接返回了局部变量的地址

     int *get_num(){int a = 15;int *p = &a;//此时p对应的数据是一个局部作用域的数据return p;}main(){int *p = get_num();//此时p是野指针}

如何避免野指针?

  1. 指针变量要及时初始化,如果暂时没有对应的值,建议赋初值NULL。

  2. 数组操作(遍历和指针运算)时,注意数组的长度,避免越界

  3. 指针指向的内存空间被回收,建议给这个指针遍历赋值为NULL

     int *p = (int *)malloc(10);free(p);p = NULL;
  4. 指针遍历使用之前要检查它的有效性(非空检验)

     int *p = NULL;// if(p == NULL)if(!p){return -1;}

空指针

很多情况下,我们不可避免的会遇到野指针,比如刚定义的指针无法立即为其分配一块恰当的内存,又或者指针指向的内存已经被释放了等等。一般的做法是将这些危险的野指针指向一块确定的内存,比如零地址内存(NULL)。

定义:空指针即保存了零地址的指针(赋值为NULL的指针),也就是指向零地址的指针。(NULL是空常量,它的值是0,这个NULL一般存放在内存0x0000 0000的位置,这个地址只能存NULL,不能被其他程序修改)

示例:

 // 1.刚定义的指针,让其指向零地址以确保安全char *p1 = NULL;int *p2 = NULL;​// 2.被释放了内存的指针,让其指向零地址以确保安全char *p3 = malloc(100);free(p3);p3 = NULL;

空悬指针

在C语言中,悬空指针指的是指向已删除(或释放)的内存位置的指针。如果一个指针指向的内存已经被释放,但指针本身并未重新指向其他有效的内存地址,那么这个指针就变成了悬空指针。悬空指针会引发不可预知的错误,并且如果一旦发生,就很难定位,因此在编程中尽量避免使用悬空指针。

 // 2.被释放了内存的指针,让其指向零地址以确保安全char *p3 = malloc(100);free(p3);printf("%p,%c\n",p3,*p3);//此时地址依然可以访问,但是地址对应的原本数据不可访问

void与void*的区别

定义:

  • void:是空类型,是数据类型的一种

  • void*:是指针类型,是指针类型的一种,可以匹配任意类型的指针,类似与通配符,又被叫做万能指针。

void:

  • 说明:void作为返回值类型说明,表示没有返回值;作为形参,表示形参列表为空,在调用的时候不能给实参

  • 举例:

 //函数定义void fun(void){..}//等效于 void fun(){..}//函数调用fun();

void*:

  • 说明:

    • void*是一个指针类型,但该指针的数据类型不明确,无法通过解引用获取内存中的数据,因为void*不知道访问几个内存单元。

    • void*是一种数据类型可以作为函数返回值类型,也可以作为形参类型

    • void*类型的变量在使用之前必须强制类型转换,明确它能够访问几个字节的内存空间

     int *p = (int*)malloc(4);
  • 说明:

    • void*作为返回值类型,这个函数可以返回任意类型的指针

    • void*作为形参类型,这个函数在调用时,可以给任意类型的指针

  • 总结:

    • void*类似于通配符,不能对void*类型的变量解引用(因为不明确数据类型,所以无法确定内存单元的大小)

    • void*在间接访问(解引用)前要强制类型转换,但不能太随意,否则存和取的数据类型不一致


http://www.ppmy.cn/devtools/149695.html

相关文章

Python网络爬虫简介-科普版

Python网络爬虫简介 一、什么是网络爬虫 网络爬虫(Web Crawler),又称为网页蜘蛛、网页机器人,是一种按照一定规则自动抓取互联网信息的程序。它通过模拟浏览器的行为,访问网页,获取网页内容,并…

探索电商宝藏:用Java打造1688商品详情爬虫,挖掘商业价值

在数字化浪潮的推动下,电商行业正以前所未有的速度蓬勃发展,成为现代商业版图中举足轻重的一环。作为国内领先的B2B电商平台,1688汇聚了海量商品和众多供应商,宛如一座蕴藏丰富商机的宝藏。然而,如何高效地挖掘这座宝藏…

jenkins入门10--自动化构建

build periodically:设定类似cron周期性时间触发构建 * * * * * (五颗星,中间用空格隔开) 第一颗表示分钟,取值0~59 第二颗表示小时,取值0~23 第三颗表示一个月的第几天,取值1~31 第四颗表示第几月&#xf…

使用宝塔面板,安装 Nginx、MySQL 和 Node.js

使用ssh远程链接服务器 在完成使用ssh远程链接服务器后 可使用宝塔面板,安装 Nginx、MySQL 和 Node.js 宝塔网站 一、远程链接服务器 二、根据服务器系统安装宝塔 wget -O install.sh https://download.bt.cn/install/install_lts.sh && sudo bash inst…

深入讲解 Docker 及实践

Docker 是现代化应用开发、测试和生产环境部署中不可或缺的工具。它能够为开发人员提供与生产环境一致的开发环境,同时支持高效的容器化部署、资源隔离、容器编排等高级功能。尤其在微服务架构和云原生应用中,Docker 更是提供了简化的流程和高效的可扩展…

数组分割函数

这是一个数组分割函数,它的作用是将一个大数组按照指定的长度分割成多个小数组。 参数说明: array: 需要被分割的原始数组 subGroupLength: 每个小数组的长度 工作原理: splitArray(array, subGroupLength) {let index 0; …

Redis-代理(解决redis压力)

Redis-proxy 是一种用于解决 Redis 压力的代理解决方案。它可以作为一个中间层,将来自客户端的请求转发到后端的一个或多个 Redis 服务器,并处理负载均衡、高可用性等问题。 以下是一个简单的 Redis-proxy 示例,使用了开源的项目 twemproxy …

一个基于Spring Boot的简单网吧管理系统

一个基于Spring Boot的简单网吧管理系统的案例代码。这个系统包括用户管理、电脑管理、上机记录管理等功能。代码结构清晰,适合初学者学习和参考。 1. 项目结构 src/main/java/com/example/netbarmanagement├── controller│ ├── ComputerController.jav…