动态内存基础
data:image/s3,"s3://crabby-images/33980/33980d9a8c223709959f0b511c6637b326b63ca3" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/92643/9264325ee3485e17ba570fa132bd433fecd55956" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/c4ea2/c4ea29531f60aa2bcd27aeec8e3fb3552e58303f" alt="在这里插入图片描述" 栈的对象在内存中开辟的空间都比较紧密,连续创建的对象x和y,他们在栈空间是非常紧密的。函数调用,栈针就会出栈。函数结束,对象就自动销毁了。
堆是在运行期动态扩张。因为有的时候,需要多少内存,是在运行期才能确定的。所以就需要堆。比如刚开始分配了10个int的内存,使着过程中,发现不够用了,再分配20个int。这就是动态的扩展内存,动态缩小内存也同理。
对比栈的内存,栈针弹出会自动销毁。但是堆不会,需要通过delete进行释放。
如果控制堆内存呢?我们通过new和delete来构造、销毁对象。
栈内存:
int main()
{
int x;
x = 2;
cout << x << endl;
}
堆内存:
int main()
{
int* y = new int(2);
cout << *y << endl;
delete y;
}
int* fun()
{
int* res = new int(2);
return res;
}
int main()
{
int* y = fun();
cout << *y << endl;
delete y;
}
data:image/s3,"s3://crabby-images/8f28a/8f28a7ba44353275d983ed9f9bd522c3ceb7b3b8" alt="在这里插入图片描述"
int* fun()
{
int res = 2;
return &res;
}
int main()
{
int* y = fun();
cout << *y << endl;
delete y;
}
对象的构造:1分配内存;2在内存上构造对象; 对象的销毁:1对象销毁;2把之前分配的内存归还给系统;
data:image/s3,"s3://crabby-images/7b39d/7b39def54319d91be83c6908ee21c1692e0ee194" alt="在这里插入图片描述"
int main()
{
int* y = new int[5]{1, 11, 22, 33, 44};
cout << y[1] << endl;
delete[] y;
}
内存片段:有时候我们分配了一大块连续的内存,但是释放的时候,可能只是释放了其中一部分零散的内存,所以不连续。
通过nothrow new来判断分配内存是否成功。如果失败了,系统不抛出异常,而是返回一个nullptr的指针。
#include <iostream>
#include <new>
using namespace std;
int main()
{
int* y = new(std::nothrow) int[5]{1, 11, 22, 33, 44};
if (y == nullptr)
{
}
else
{
cout << y[1] << endl;
delete[] y;
}
}
palecement new:不需要分配内存了,只需要在已分配的内存上构造对象。vector就是用这个原理。vector动态增长。
vector增加一个元素的原理不是:若来了一个新的元素,先开辟一块新的内存,把原有的值拷贝过来,再把新的元素放进来。vector一般不这么做,而是多分配一些。这样不用一直拷贝。 data:image/s3,"s3://crabby-images/3299e/3299e6e045d342cde9943d04d12db340306c0ff8" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/cd5dd/cd5dd403fb724db70ba60dc4abac9dd0bf051437" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/87691/8769182b1e9475ab5c50642203d2ebe4c02f6fad" alt="在这里插入图片描述"
int main()
{
int* y = new auto(3);
}
new与对象对齐 data:image/s3,"s3://crabby-images/7b4b3/7b4b3d07f6a22cb326547c981b594deca63437b8" alt="在这里插入图片描述" delete
int main()
{
int* ptr = new int;
delete ptr;
int* ptr2 = new int[5];
delete[] ptr2;
}
placement delete:只关注内存上构造的对象销毁掉,但是不会把内存归还给系统。还是vector的删除一个元素的原理,只会把最后一个元素销毁掉,但是对象对应的内存并不归还给系统。
对于内建的类型,不需要考虑placement delete。只有我们定义了类的析构函数。析构函数会刷新缓存,关掉文件。
data:image/s3,"s3://crabby-images/4169c/4169c56b54920440c0d9ce458673899e3e363473" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/c33b3/c33b33a4ba6707ff5c2b0fb17fe0bc42d9af4cea" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/a835c/a835c491a1d08ff6ebd64daee923127c53f71e9e" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/82bc8/82bc83b400b2f99f6017ba8a60c6ba9ca9ec14a7" alt="在这里插入图片描述"
int main()
{
int* ptr = new int(3);
cout << ptr << endl;
delete ptr;
ptr = nullptr;
cout << ptr << endl;
delete ptr;
}
data:image/s3,"s3://crabby-images/ea960/ea960ec087dc83c2d47d6e9d76b647890ed47e62" alt="在这里插入图片描述"
智能指针
data:image/s3,"s3://crabby-images/c08f2/c08f271b15eec5c695db4a4f5d3f6a78b2f751ca" alt="在这里插入图片描述" new和delete是关系很紧密的两条语句,但是什么时候调用delete我们说不清,容易产生不销毁或者多销毁,而且还有内存所有权的问题。所以引入了智能指针的概念。
Shared_ptr——基于引用计数的共享内存解决方案
int main()
{
std::shared_ptr<int> x(new int(3));
cout << *x << endl;
}
share_ptr是一个智能指针,它的机制是引用计数,通过引用计数来维护内存什么时候进行销毁。我们不用担心内存泄漏,整个x在main结束,x的生命周期结束,x是一个share_ptr包含一个析构函数,x结束,析构函数销毁new int构造的内存。
#include <iostream>
#include <new>
#include <memory>
using namespace std;
int main()
{
std::shared_ptr<int> x(new int(3));
std::shared_ptr<int> y = x;
}
data:image/s3,"s3://crabby-images/3fa3b/3fa3b3d86ae8e58ea124d5ada68c4bdb86c56a50" alt="在这里插入图片描述" get()
#include <iostream>
#include <new>
#include <memory>
using namespace std;
std::shared_ptr<int> fun()
{
std::shared_ptr<int> res(new int(3));
return res;
}
int main()
{
auto y = fun();
cout << *y << endl;
cout << *(y.get()) << endl;
}
#include <iostream>
#include <new>
#include <memory>
using namespace std;
std::shared_ptr<int> fun()
{
std::shared_ptr<int> res(new int(3));
return res;
}
void fun2(int* x)
{
}
int main()
{
auto y = fun();
cout << *y << endl;
cout << *(y.get()) << endl;
fun2(y.get());
}
reset()
#include <iostream>
#include <new>
#include <memory>
using namespace std;
std::shared_ptr<int> fun()
{
std::shared_ptr<int> res(new int(3));
return res;
}
int main()
{
auto y = fun();
y.reset(new int(100));
cout << *y << endl;
}
指定内存回收逻辑:把内存放在一个池子里,一般从内存池里拿,不用的的时候放回池子里,不涉及new和delete操作,可以提升速度。 data:image/s3,"s3://crabby-images/46bba/46bbad49d063fd5d87cad6d9a56926163fcaa4e5" alt="在这里插入图片描述" make_shared
#include <iostream>
#include <new>
#include <memory>
using namespace std;
int main()
{
std::shared_ptr<int> ptr(new int(3));
std::shared_ptr<int> ptr2 = std::make_shared<int>(3);
auto ptr2 = std::make_shared<int>(3);
}
支持数组 data:image/s3,"s3://crabby-images/c3268/c32682ba4887b0a83bd5a38642408680170c024a" alt="在这里插入图片描述"
int main()
{
std::shared_ptr<int[]> ptr(new int[5]);
}
c++20可以用make_shared data:image/s3,"s3://crabby-images/4abdd/4abdd7a8f2db2f858cdd0305914850badf973672" alt="在这里插入图片描述" 注意: shared_ptr 管理的对象不要调用 delete 销毁 data:image/s3,"s3://crabby-images/34f4d/34f4d04b91954bc1f96dbe96a47fc39c448f8367" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/3ed6d/3ed6d2adcf2ff59685eda10d4ac023026ad6d91a" alt="在这里插入图片描述"
unique_ptr——独占内存的解决方案
data:image/s3,"s3://crabby-images/b8167/b8167ea381f5fa36a6482484e3bea14ecd3b7bca" alt="在这里插入图片描述" 一般意义指针就是共享的行为,但是不排除独占的时候。就要用unique_ptr
#include <iostream>
#include <new>
#include <memory>
using namespace std;
int main()
{
std::unique_ptr<int> x(new int(3));
}
int main()
{
std::unique_ptr<int> x(new int(3));
cout << x.get() << endl;
std::unique_ptr<int> y = std::move(x);
cout << x.get() << endl;
cout << y.get() << endl;
}
data:image/s3,"s3://crabby-images/15c79/15c79c465f13e3b5efae4bb146703b93f00665f1" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/b4139/b4139a6021bdb25f1042e80b03edef2337ccb98b" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/5f774/5f7745840befa6bb7318f6765aa2cbc735c7fad5" alt="在这里插入图片描述"
weak_ptr——防止循环引用而引入的智能指针
data:image/s3,"s3://crabby-images/739e1/739e133db07d0acac6024ea1cdc200e5183f129c" alt="在这里插入图片描述" shared_ptr通过引用计数来销毁。但是17-20行 data:image/s3,"s3://crabby-images/7790c/7790c76f15c5f614b105871c96f8a95483779ad0" alt="在这里插入图片描述" 所以要解决循环引用,使用weak_ptr来解决。 weak_ptr可以和shared_ptr共享一块内存,但是weak_ptr并不会增加引用计数。这样就解决了循环引用的问题。
动态内存的相关问题
data:image/s3,"s3://crabby-images/27137/27137912d4b5eedec86bd52f5eef6c7b27d5d673" alt="在这里插入图片描述"
sizeof不会返回动态内存的大小:
#include <iostream>
#include <new>
#include <memory>
using namespace std;
int main()
{
int* ptr = new int(3);
cout << sizeof(ptr) << endl;
int* ptr2 = new int[3];
cout << sizeof(ptr2) << endl;
}
使用分配器( allocator )来分配内存
对象的构造分2步:1分配内存;2构造对象 allocator成员函数: data:image/s3,"s3://crabby-images/454ae/454aef2c7bce9d44957281c3be54c5c5cf58516a" alt="在这里插入图片描述" allocate分配内存
int main()
{
std::allocator<int> al;
int* ptr = al.allocate(3);
}
deallocate回收内存
int main()
{
std::allocator<int> al;
int* ptr = al.allocate(3);
al.deallocate(ptr, 3);
}
使用 malloc / free 来管理内存
malloc / free另外一对内存管理的方法。对比new和delete。既然有了new和delete,为什么还要这个呢?malloc / free 继承自c语言。new和delete都包含2个操作,分配和对象。但是malloc和free只会分配内存,不会在内存上构造对象,因为c语言内部,不是很强调对象的构造。c++引入类之后,才强调对象的概念。 data:image/s3,"s3://crabby-images/a4e17/a4e175ac4d11479c8988f39f5ca3a1df4669e964" alt="在这里插入图片描述" 但是mallo不保证内存的对其,所以引入了aligned_alloc来分配内存对齐
使用 aligned_alloc 来分配对齐内存
如果有可能还是使用allocator。
动态内存与异常安全
#include <iostream>
#include <new>
#include <memory>
using namespace std;
int main()
{
int* ptr = new int(3);
delete ptr;
}
如果内存没有及时释放,内存在堆上不断的激增,最后导致系统崩溃。
#include <iostream>
#include <new>
#include <memory>
using namespace std;
void fun()
{
std::shared_ptr<int> ptr(new int(3));
}
int main()
{
}
C++ 对于垃圾回收的支持
垃圾回收:动态指定了内存,现在不用了,需要回收掉。python,java都支持垃圾回收。系统会判断内存没有使用,然后自动回收。c++对于垃圾回收可以说是欲拒还迎的状态。 data:image/s3,"s3://crabby-images/10a8c/10a8c4498eebd52200586cdc4be3af10ef3698b1" alt="在这里插入图片描述"
|