笔者有幸得到一次和行业前辈交流的机会,前辈也分享了他对于职业规划、人生发展的理解和感悟,笔者也获益颇丰。另外由于笔者主要方向是C/C++,刚入门不久,前辈也是针对能力提升方面推荐了很多课程和书籍,其中就有这本 《高质量程序设计指南C/C++》作者林锐,第三版。
自己也是刚入门不久,可能会有些错误,欢迎大家一起学习,不吝赐教,有任何问题可以评论私信。
前面几篇:
林锐《高质量程序设计指南C/C++》笔记01:基本数据类型、类型转换
林锐《高质量程序设计指南C/C++》笔记02:断言、数组访问、位域、条件编译
林锐《高质量程序设计指南C/C++》笔记03:虚继承、C++重载内联
林锐《高质量程序设计指南C/C++》笔记04:异常处理
第十六章:内存管理
一、常用内存分配方式
- 静态内存分配
静态内存分配在编译时完成,内存在整个程序生命周期内保持分配状态。全局变量、静态变量和常量通常使用静态内存分配。
int globalVar = 10; // 全局变量
static int staticVar = 20; // 静态变量
int main() {
const int constVar = 30; // 常量
return 0;
}
- 自动内存分配
自动内存分配也称为栈内存分配,局部变量(包括形参) 在函数调用时分配内存,并在函数返回时自动释放。它的生命周期受作用域控制。
void function() {
int localVar = 40; // 自动分配(栈内存)
// 函数结束自动释放内存
}
int main() {
function();
return 0;
}
- 动态内存分配
动态内存分配在运行时进行,使用new和delete操作符在堆上分配和释放内存。动态内存分配需要程序员显式管理内存的生命周期。
int main() {
int* p = new int(50); // 动态分配单个整数
delete p; // 释放动态分配的内存
int* arr = new int[5]; // 动态分配数组
for (int i = 0; i < 5; ++i) {
arr[i] = i * 10;
}
delete[] arr; // 释放动态分配的数组内存
return 0;
}
二、当指针作为函数参数
如果函数参数是个指针,不能用该指针去申请动态内存。如:
void allocateMemory(int* ptr) {
ptr = new int[10]; // 在堆上分配10个int的数组
// 这里 ptr 指向了新分配的内存
}
int main() {
int* myPtr = nullptr;
allocateMemory(myPtr);
// 这里 myPtr 仍然是 nullptr,因为 allocateMemory 函数的修改只在函数内部有效
return 0;
}
这是因为当一个指针作为参数传递给函数时,传递的是指针的值(地址),而不是指针的引用。因此,函数内部对指针变量的修改并不会影响到函数外部的原始指针。这会导致内存泄漏。
这里将 myPtr的值(即 nullptr)按值传递给 allocateMemory函数的参数 ptr。此时,ptr是 myPtr的拷贝,它们都指向同一个地址(即 nullptr)。因此,ptr指向新分配的内存,但 myPtr仍然是 nullptr。
解决方法:
- 使用指针的引用
void allocateMemory(int*& ptr) {
ptr = new int[10]; // 分配动态内存
}
int main() {
int* myPtr = nullptr;
allocateMemory(myPtr);
// 这里 myPtr 指向新分配的内存
delete[] myPtr; // 释放内存
return 0;
}
引用相当于取了一个别名。函数内部对 ptr的修改会反映到 main函数中的 myPtr,确保 myPtr指向新分配的内存。
三、智能指针
见:c++11智能指针详解之shared_ptr代码实现:基本功能
总结
该系列:
林锐《高质量程序设计指南C/C++》笔记01
林锐《高质量程序设计指南C/C++》笔记02
林锐《高质量程序设计指南C/C++》笔记03

2万+

被折叠的 条评论
为什么被折叠?



