C++一分钟之-动态内存管理:new与delete

简介: 【6月更文挑战第19天】在C++中,`new`和`delete`用于动态内存管理,分配和释放堆内存。不正确使用可能导致内存泄漏和悬挂指针。要避免这些问题,确保每次`new`都有匹配的`delete`,释放内存后设指针为`nullptr`。使用`delete[]`释放数组,避免重复释放。智能指针如`std::unique_ptr`可自动管理内存,减少手动管理的风险。通过实例展示了如何使用智能指针进行安全的内存操作。

在C++编程的广阔天地中,动态内存管理是一项不可或缺的技能。通过newdelete操作符,开发者能够按需分配和释放内存,这对于处理不确定大小的数据结构或在运行时调整资源尤为关键。然而,不当的使用也可能导致内存泄漏、悬挂指针等问题。本文旨在深入浅出地讲解newdelete的使用方法,剖析常见问题与易错点,并通过实例代码展示如何避免这些陷阱。
image.png

动态内存分配基础

new操作符

  • 基本用法new用于在堆上分配内存,返回所分配内存的首地址。例如,int *p = new int; 分配一个整型大小的内存,并将地址赋给指针p
  • 初始化分配new还可以直接初始化分配的内存,如int *p = new int(10); 初始化为10。
  • 数组分配:分配数组时使用new[],如int *arr = new int[5]; 分配一个包含5个整数的数组。

delete操作符

  • 释放单个对象delete p; 释放由p指向的内存。
  • 释放数组:释放数组时必须使用delete[],如delete[] arr;,否则会导致未定义行为。

常见问题与易错点

1. 内存泄漏

问题描述:分配的内存没有被及时释放。 示例

void function() {
   
   
    int *ptr = new int;
    // ...操作ptr
    // 忘记释放ptr指向的内存
}

避免方法:确保每一块new分配的内存都有对应的delete,使用智能指针可以自动管理内存。

2. 悬挂指针

问题描述:指向已释放内存的指针。 示例

int *ptr = new int;
delete ptr;
// ptr现在是一个悬挂指针,如果再次使用可能导致未定义行为
cout << *ptr;

避免方法:释放内存后立即将指针置为nullptr,如delete ptr; ptr = nullptr;

3. 误用deletedelete[]

问题描述:使用delete释放数组内存或使用delete[]释放单个对象的内存。 示例

int *arr = new int[5];
delete arr; // 错误,应使用delete[]

避免方法:严格区分对象和数组的释放方式,遵循分配与释放匹配的原则。

4. 重复释放内存

问题描述:对同一块内存多次执行deletedelete[]操作。 示例

int *ptr = new int;
delete ptr;
delete ptr; // 重复释放,错误

避免方法:释放后立即将指针置为nullptr,或者使用智能指针避免手动管理内存。

实践代码示例

下面的代码演示了如何正确使用newdelete,并利用智能指针避免内存管理错误:

#include <iostream>
#include <memory>

void properMemoryManagement() {
   
   
    // 使用unique_ptr自动管理单个对象的内存
    std::unique_ptr<int> ptr(new int(10));
    std::cout << *ptr << std::endl;

    // 使用unique_ptr管理数组
    std::unique_ptr<int[]> arr(new int[5]{
   
   1, 2, 3, 4, 5});
    for(int i = 0; i < 5; ++i) {
   
   
        std::cout << arr[i] << " ";
    }
    std::cout << std::endl;
    // 不需要手动调用delete,智能指针自动处理
}

int main() {
   
   
    properMemoryManagement();
    return 0;
}

通过上述内容,我们不仅学习了newdelete的基本用法,还深入探讨了动态内存管理中常见的问题与解决方案。掌握这些原则,结合实践中的不断应用,你将能够在C++编程中更加游刃有余地控制内存,编写出既高效又健壮的代码。

目录
相关文章
|
7月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
251 26
|
12月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
8月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
157 1
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
586 68
|
11月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
12月前
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
513 0
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
622 0
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
279 3
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
716 4
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。