c++栈内存和堆内存的基本使用介绍
1.c++栈内存和堆内存的基本使用
#include <iostream>
// 定义一个简单的结构体
struct Person {
std::string name;
int age;
};
int main() {
// 栈内存分配
int a = 10; // 基本数据类型的栈内存分配
Person person; // 结构体的栈内存分配
person.name = "John";
person.age = 30;
std::cout << "Stack memory usage:" << std::endl;
std::cout << "a = " << a << std::endl;
std::cout << "Person name: " << person.name << ", age: " << person.age << std::endl;
// 堆内存分配
int* p = new int; // 基本数据类型的堆内存分配
*p = 20;
Person* personPtr = new Person; // 结构体的堆内存分配
personPtr->name = "Alice";
personPtr->age = 25;
std::cout << "Heap memory usage:" << std::endl;
std::cout << "*p = " << *p << std::endl;
std::cout << "PersonPtr name: " << personPtr->name << ", age: " << personPtr->age << std::endl;
// 释放堆内存
delete p;
delete personPtr;
return 0;
}
编译运行输出:
【栈(Stack):由编译器自动分配和释放,存放函数的参数,局部变量、临时变量、函数返回地址等。
堆(Heap):由程序员分配和释放,否则会引起内存泄漏。程序在运行的时候用malloc或new申请任意大小的内存,程序员自己负责在适当的时候用free或 delete释放内存,生命周期由开发者控制,未释放时可能导致内存泄漏(程序结束后由操作系统回收)。】
本例展示了栈(Stack)内存和堆(Heap)内存的分配方式:
栈内存通过直接声明变量(如int a = 10;)分配。
堆内存通过new关键字分配,并通过delete释放。
生命周期说明:
栈内存的生命周期与作用域相关,变量在作用域结束时自动销毁。
堆内存的生命周期由程序员手动管理,需要显式调用delete或delete[]释放。
2.动态数组的使用和内存管理
#include <iostream>
int main() {
// 动态数组的堆内存分配
int size = 5;
int* array = new int[size];
// 初始化数组
for (int i = 0; i < size; ++i) {
array[i] = i * 10;
}
// 打印数组
std::cout << "Dynamic array:" << std::endl;
for (int i = 0; i < size; ++i) {
std::cout << "array[" << i << "] = " << array[i] << std::endl;
}
// 释放堆内存
delete[] array;
return 0;
}
编译运行输出:
本例演示了如何分配和管理动态数组的内存,使用new[]进行分配,使用delete[]进行释放。这是处理动态数组内存的正确方式。
使用new[]分配的内存必须用delete[]释放,而new分配的内存必须用delete释放,否则可能导致未定义行为。
3.内存泄漏及其解决示例
#include <iostream>
void memoryLeakExample() {
int size = 5;
int* leakArray = new int[size] {1, 2, 3, 4, 5}; // 分配并初始化数组; 这个数组若没有被释放,则会造成内存泄漏
// 打印数组元素
for (int i = 0; i < size; ++i) {
std::cout << "leakArray[" << i << "] = " << leakArray[i] << std::endl;
}
// 正确的处理方式:释放数组
delete[] leakArray;
}
int main() {
// 调用
memoryLeakExample();
return 0;
}
编译运行输出:
本例展示了如何防范内存泄漏,及正确的处理方式。
更多内存泄漏的常见还有:
忘记释放指针。
异常导致程序提前退出,未执行释放代码。
使用std::vector等容器时,错误地管理内存。
4.智能指针的使用
C++11引入了智能指针,可以自动管理内存,避免内存泄漏。
#include <iostream>
#include <memory> // 需要包含这个头文件
class MyClass {
public:
MyClass() {
std::cout << "Constructor called" << std::endl;
}
~MyClass() {
std::cout << "Destructor called" << std::endl;
}
void display() {
std::cout << "Display method called" << std::endl;
}
};
int main() {
// 使用unique_ptr
std::unique_ptr<MyClass> ptr1(new MyClass());
ptr1->display();
// 使用shared_ptr
std::shared_ptr<MyClass> ptr2 = std::make_shared<MyClass>();
ptr2->display();
return 0;
}
编译运行输出:
本例介绍了C++11引入的智能指针(unique_ptr和shared_ptr),并展示了它们的基本用法。智能指针能够自动管理内存,有助于避免内存泄漏。关于智能指针的更多情况可见C++11中智能指针的原理、使用、实现 - wxquare - 博客园
https://zhuanlan.zhihu.com/p/150555165
OK!