C语言中野指针和空指针的区别

 更新时间:2024年11月26日 09:06:47   作者:AzurSatr  
本文主要介绍了野指针和空指针的概念、用途、原理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

野指针

定义

野指针是指那些指向了未知、随机、不正确或已经被释放的内存地址的指针变量。它们可能指向了程序栈上的某个临时变量(该变量可能已经出栈并被覆盖),或者指向了已经被freedelete释放的内存块,或者根本就是一个随机的、未初始化的内存地址。
注:野指针不会直接引发错误,操作野指针指向的内存区域才会出问题。

用途

(实际上,野指针应该被避免):
野指针通常是由于编程错误造成的,比如忘记初始化指针、释放内存后未将指针置为NULL等。由于它们指向的内存状态是不确定的,因此使用野指针往往会导致程序崩溃、数据损坏等严重问题。

原理

野指针的产生往往与内存管理不当有关。在C语言中,需要手动管理内存(包括分配和释放),如果在这个过程中出现了疏忽,就可能导致野指针的产生。

指针未初始化

在C语言中,指针变量在声明时不会自动初始化为NULL或任何其他值。如果在声明指针后没有立即为其分配一个有效的内存地址或显式地将其初始化为NULL,那么这个指针就会包含一个随机的内存地址,成为野指针。

int *ptr; // 声明了一个指针变量ptr,但未初始化  
*ptr = 10; // 尝试解引用ptr,这是未定义行为,因为ptr是野指针

指针越界访问

当指针被用来访问数组或其他连续内存区域时,如果访问的索引超出了其有效范围,那么指针就会指向一个未知的内存地址,从而成为野指针。

int arr[10];  
int *ptr = arr;  
for (int i = 0; i <= 12; i++) { // 注意循环条件,i会超出数组范围  
    *(ptr + i) = i; // 当i=10和i=11时,ptr+i成为野指针  
}

释放内存后未置NULL

使用freedelete等函数释放了指针所指向的内存后,如果没有将指针置为NULL,那么这个指针仍然会保留原来的内存地址。此时,如果程序继续尝试通过该指针访问内存(尽管内存可能已经被系统重新分配给其他变量),就会导致未定义行为,因为该指针已经成为了野指针。

int *ptr = (int*)malloc(sizeof(int));  
if (ptr != NULL) {  
    *ptr = 10;  
    free(ptr); // 释放内存  
    // 忘记将ptr置为NULL  
    // ...  
    // 如果之后再次使用ptr,如*ptr = 20; 则ptr是野指针  
}

返回局部变量的地址

在函数中,如果返回了一个指向局部变量的指针,那么在函数返回后,局部变量会被销毁,但返回的指针仍然指向那个已经被销毁的内存地址。此时,该指针就成为了野指针。

int* getPointer() {  
    int local = 10;  
    return &local; // 返回指向局部变量的指针  
}  
  
int main() {  
    int *ptr = getPointer(); // ptr成为野指针,因为local已经被销毁  
    // ...  
}

指针运算错误

在进行指针运算时(如指针加法、减法或递增、递减),如果运算结果超出了合法内存范围,那么指针就会指向一个未知的内存地址,从而成为野指针。

如何避免出现

初始化指针

在声明指针变量时,应立即将其初始化为NULL或指向一个有效的内存地址。这可以防止指针在未经初始化的情况下就被使用,从而避免野指针的产生。指针变量刚被创建时不会自动成为NULL指针,它的缺省值是随机的,它所指的空间是随机的。

int *ptr = NULL; // 初始化为NULL  
int array[10];  
int *arrayPtr = array; // 指向有效数组的首地址

检查指针是否为NULL

在解引用指针之前,应始终检查它是否为NULL。这可以确保只有在指针指向有效内存地址时才进行解引用操作。

if (ptr != NULL) {  
    *ptr = 10; // 只有当ptr不是NULL时才解引用  
}

释放内存后置NULL

在释放指针所指向的内存后,应立即将指针置为NULL。这可以防止在之后的代码中错误地尝试再次解引用该指针。

free(ptr);  
ptr = NULL; // 避免成为野指针

避免返回局部变量的地址

函数内部定义的局部变量在函数返回时会被销毁,因此不应该返回指向这些局部变量的指针。如果需要返回数据,可以考虑返回数据的副本、使用动态分配的内存、或者通过参数传递指针并在函数外部分配内存。

// 反例 
int* badFunction() {  
    int local = 10;  
    return &local; // 返回指向局部变量的指针,可能导致野指针  
}  

// 范例之一:返回动态分配的内存  
int* goodFunction() {  
    int *ptr = (int*)malloc(sizeof(int));  
    if (ptr != NULL) {  
        *ptr = 10;  
    }  
    return ptr; // 注意:调用者需要负责释放内存  
}

使用智能指针(仅限C++)

使用std::unique_ptrstd::shared_ptr等智能指针可以自动管理内存的生命周期,从而减少野指针的风险。

注意指针运算

在进行指针运算时(如递增、递减、加法、减法),要确保指针始终在合法范围内。避免指针越界访问,这可能会导致野指针的出现(虽然不是直接的野指针,但可能导致不可预测的行为)。

代码审查和测试

使用静态分析工具、内存泄漏检测器和单元测试等工具。

空指针

定义

空指针是一个特殊的指针值,它不指向任何有效的内存地址。在C语言中,空指针通常用宏NULL来表示,实际上NULL通常被定义为((void*)0)或简单地0

用途

空指针主要用于表示指针当前不指向任何对象。在初始化指针时,将其设置为NULL可以明确地表明该指针当前不指向任何有效内存地址。此外,在释放指针所指向的内存后,将指针置为NULL可以防止野指针的产生。

原理

空指针的原理很简单,它就是一个特殊的值(通常是0),用于表示指针不指向任何内存地址。在C语言中,任何尝试解引用空指针的行为都是未定义的,通常会导致程序崩溃。

举例:

#include <stdio.h>  
  
int main() {  
    int *ptr = NULL; // 声明了一个指针变量ptr,并将其初始化为NULL,表示它不指向任何内存地址  
  
    if (ptr != NULL) {  
        // 如果ptr不是NULL,则执行这里的代码(但在这个例子中,这里的代码不会被执行)  
    } else {  
        printf("ptr is NULL\n"); // 输出ptr是空指针的信息  
    }  
  
    // 尝试解引用ptr(这是不安全的,因为ptr是NULL)  
    // *ptr = 10; // 这将导致未定义行为,因为ptr是空指针  
  
    return 0;  
}

在上面的例子中,ptr被初始化为NULL,表示它不指向任何内存地址。然后,我们通过if语句检查ptr是否为NULL,并根据结果打印相应的信息。注意,我们并没有尝试解引用ptr,因为这是一个空指针,解引用它是不安全的。

到此这篇关于C语言中野指针和空指针的区别的文章就介绍到这了,更多相关C语言 野指针和空指针内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Opencv实现轮廓提取功能

    Opencv实现轮廓提取功能

    这篇文章主要为大家详细介绍了Opencv实现轮廓提取功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-05-05
  • VS2022永久配置OpenCV开发环境的实现

    VS2022永久配置OpenCV开发环境的实现

    本文主要介绍了VS2022永久配置OpenCV开发环境的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-02-02
  • C++代码改造为UTF-8编码问题的总结(最新推荐)

    C++代码改造为UTF-8编码问题的总结(最新推荐)

    本文总结了如何将C++程序代码改造为UTF-8编码,包括操作系统、编译器和终端等各方面的设置,在实际操作中,可以通过渐进式更新的方式,只在新的代码项目中使用UTF-8编码,避免大规模修改旧代码,感兴趣的朋友一起看看吧
    2025-02-02
  • C/C++经典算法之约瑟夫问题详解

    C/C++经典算法之约瑟夫问题详解

    这篇文章主要给大家介绍了关于C/C++经典算法之约瑟夫问题的相关资料,约瑟夫环问题是一道经典的数据结构的题目,本文介绍了解决约瑟夫问题的三种方法,需要的朋友可以参考下
    2021-07-07
  • DSP中浮点转定点运算--浮点与定点概述

    DSP中浮点转定点运算--浮点与定点概述

    本文主要介绍DSP中浮点与定点概述,很值得学习一下,需要的朋友可以参考一下。
    2016-06-06
  • C语言深入分析数组指针和指针数组的应用

    C语言深入分析数组指针和指针数组的应用

    在C语言和C++等语言中,数组元素全为指针变量的数组称为指针数组,指针数组中的元素都必须具有相同的存储类型、指向相同数据类型的指针变量。指针数组比较适合用来指向若干个字符串,使字符串处理更加方便、灵活
    2022-04-04
  • C++使用string的大数取模运算(5)

    C++使用string的大数取模运算(5)

    这篇文章主要为大家详细介绍了C++使用string的大数取模运算,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-09-09
  • 实例详解C/C++中extern关键字

    实例详解C/C++中extern关键字

    这篇文章主要介绍了C/C++中extern关键字详解 的相关资料,需要的朋友可以参考下
    2016-04-04
  • C++ 头文件系列(set)详解

    C++ 头文件系列(set)详解

    一般而言,每个C++/C程序通常由头文件和定义文件组成。头文件作为一种包含功能函数、数据接口声明的载体文件,主要用于保存程序的声明,而定义文件用于保存程序的实现 。
    2017-02-02
  • QT输入框输入限制整理(正则表达式限制)

    QT输入框输入限制整理(正则表达式限制)

    我们有时需要限制文本框输入内容的类型,下面这篇文章主要给大家介绍了关于QT输入框输入限制的相关资料,文中通过代码介绍的非常详细,对大家的学习或者工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2024-04-04

最新评论