内存中OLTP与内存不足

我已经写了好几回内存中OLTP的文章和”为何我还不推荐内存中OLTP给用户”。今天我想进一步谈下内存中OLTP背后的内存需求,还有若是你内存不够的话会发生什么。sql

一切都与内存有关!

咱们都知道好久以前有个名人说过对于任何人,640K的内存应该足够了。他错了!对于内存中OLTP,内存需求很是高:数据库

  • 哈希索引的每一个哈希桶由64位长的指针组成
  • 每次你修改/删除一条记录,新版本的写入在内存中存储。

微软建议内存至少是你内存优化表的2倍。当你修改或删除记录时,这个两倍数量的空间是用作可能的行版本存储。浏览器

几个星期前,有人问我一个很是有趣的问题:当你没有足够的内存,在数据库启动期间内存中OLTP不能重建哈希索引会发生什么?这哥听起来像很是简单的问题,但在这个特定场景里知道内存中OLTP如何反应很是重要。less

假设你在虚拟机里运行内存中OLTP,在某个时候你的虚拟机管理员给你的虚拟机比以前更少的内存。在虚拟化结合中,我常常看到这个。优化

让咱们玩坏内存中OLTP!

咱们来模拟这样的情景。在第一步,我想向你展现下,当你建立了内存优化表,你没有足够的可用物理内存,会发生什么。下列代码建立有4个哈希索引的新的内存优化表,每一个哈希索引包含250百万的哈希桶。所以对这个整个表须要近7.4GB的内存,但我运行的虚拟机只有8G的内存。
this

-- 250 000 000 x 4 =  1 000 000 000 Hash Buckets of 8 bytes: 8 000 000 000 = 7.4 GB of memory overhead.
-- The following query will fail, because there is too less memory available.
CREATE TABLE Foo
(
    Col1 INT NOT NULL PRIMARY KEY NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000),
    Col2 INT NOT NULL INDEX idx_Col2 NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000),
    Col3 INT NOT NULL INDEX idx_Col3 NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000),
    Col4 INT NOT NULL INDEX idx_Col4 NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000)
)
WITH
(
    MEMORY_OPTIMIZED = ON, 
    DURABILITY = SCHEMA_AND_DATA
)
GO

几秒后,CREATE TABLE语句失败,内存中OLTP给你一个漂亮的错误信息:你有太少的可用内存。spa

Msg 701, Level 17, State 137, Line 43 There is insufficient system memory in resource pool ‘default’ to run this query.设计

到目前还好。让咱们从新设计表,只须要3.7G内存:指针

 1 -- 250 000 000 x 2 = 500 000 000 Hash Buckets of 8 bytes: 4 000 000 000 = 3.7 GB of memory overhead.
 2 -- The following query will fail, because there is too less memory available.
 3 CREATE TABLE Foo
 4 (
 5     Col1 INT NOT NULL PRIMARY KEY NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000),
 6     Col2 INT NOT NULL INDEX idx_Col2 NONCLUSTERED HASH WITH (BUCKET_COUNT = 250000000)
 7 )
 8 WITH
 9 (
10     MEMORY_OPTIMIZED = ON, 
11     DURABILITY = SCHEMA_AND_DATA
12 )
13 GO

此次表建立成功,由于我有足够的可用内存。如今我使坏,关掉虚拟机。我配置它只有3G的内存:日志

如今当咱们重启虚拟机和SQL Server,你认为会发什么?你以为SQL Server能够把咱们数据库恢复在线么?或者你认为只有主文件组(PRIMARY file group)恢复在线,内存中文件组(In-Memory file group)仍是离线?咱们来试下!

重启后,当你在SSMS里查看对象浏览器,你能够看到咱们“整个”数据库在恢复待定状态(Recovery Pending)!

这真的真的太糟糕了,由于你不能访问你的任何数据库!即便基于传统硬盘的表也不能访问!当你查看SQL Server日志,你也会看到SQL Server给你有太少可用内存的错误信息:

偶滴神哪,咱们已经玩坏内存中OLTP……

小结

我知道模拟的状况很是少见,但我说过,当虚拟机管理员只从虚拟机里拿走内存时,这个状况很常见。与内存中OPTP结合,这就意味着你的整个数据库不可访问!

当你在基于内存中OLTP部署数据库时,请记住这个。你要认真考虑你的内存需求,你也要按需调整你的将来可用内存。

感谢关注!

原文连接:

http://www.sqlpassion.at/archive/2016/05/31/in-memory-oltp-and-too-less-memory/

相关文章
相关标签/搜索