内存中 OLTP 内的初始领域调查

适用于:SQL ServerAzure SQL 数据库Azure SQL 托管实例

本文适用于急于了解 Microsoft SQL Server 和 Azure SQL 数据库的内存中 OLTP 性能功能基础知识的开发人员。

有关内存中 OLTP,本文提供以下内容:

  • 功能简介。
  • 实现功能的核心代码示例。

SQL Server 和 SQL 数据库在内存中技术支持方面只有细微的不同。

在现实中,一些博客主将内存中 OLTP 称为 Hekaton

内存中功能的优点

SQL Server 提供的内存中功能可极大提升许多应用程序系统的性能。 本部分以直接明了的方式介绍了相关注意事项。

OLTP(联机事务处理)的功能

必须同时处理大量 SQL INSERT 的系统最适合采用 OLTP 功能。

  • 我们的基准显示,通过采用内存中功能,速度可提升 5 倍到超过 20 倍不等。

处理 Transact-SQL 中的繁重计算的系统很适合采用此功能。

  • 专用于繁重计算的存储过程可提速高达 99 倍。

稍后你可以查看以下文章,这些文章提供从内存中 OLTP 获取性能的演示:

操作分析的功能

内存中分析是指 SQL SELECT,它聚合事务数据,通常通过包含一个 GROUP BY 子句来实现。 称为 columnstore 的索引类型是操作分析的核心。

有两种主要方案:

  • “批操作分析” 是指在工作时间后运行或在具有事务数据副本的辅助硬件上运行的聚合进程。
  • “实时操作分析”是指工作时间内在用于事务工作负载的主硬件上运行的聚合进程。

本文重点介绍 OLTP,而不是介绍相关分析。 有关列存储索引如何将分析引入 SQL 的信息,请参阅:

列存储

一系列精彩的博客文章从多个视角完美地介绍了列存储索引。 大部分文章都在深入介绍列存储支持的实时运营分析的概念。 这些文章由 Microsoft 项目经理 Sunil Agarwal 于 2016年 3 月创作。

实时运行分析

  1. Real-Time Operational Analytics Using In-Memory Technology
  2. Real-Time Operational Analytics - Overview nonclustered columnstore index (NCCI)
  3. Real-Time Operational Analytics: Simple example using nonclustered clustered columnstore index (NCCI) in SQL Server 2016
  4. Real-Time Operational Analytics: DML operations and nonclustered columnstore index (NCCI) in SQL Server 2016
  5. Real-Time Operational Analytics: Filtered nonclustered columnstore index (NCCI)
  6. Real-Time Operational Analytics: Compression Delay Option for Nonclustered columnstore index (NCCI)
  7. Real-Time Operational Analytics: Compression Delay option with NCCI and the performance
  8. Real-Time Operational Analytics: Memory-Optimized Tables and columnstore index

对列存储索引进行碎片整理

  1. 使用 REORGANIZE 命令的列存储索引碎片整理
  2. REORGANIZE 的列存储索引合并策略

批量导入数据

  1. Clustered Column Store: Bulk Load
  2. Clustered columnstore index: Data Load Optimizations - Minimal Logging
  3. Clustered columnstore index: Data Load Optimization - Parallel Bulk Import

内存中 OLTP 的功能

让我们来了解一下内存中 OLTP 的主要功能。

内存优化表

CREATE TABLE 语句上的 T-SQL 关键字 MEMORY_OPTIMIZED 是表格被创建的方式,目的是令表格存在于活动内存而非磁盘上。

内存优化表 在活动内存中有自身的一种表示形式,在磁盘上有辅助副本。

  • 磁盘副本用于在关闭然后重新启动服务器或数据库后进行常规恢复。 此内存加磁盘的双重性对你和你的代码处于完全隐藏的状态。

本机编译模块

CREATE PROCEDURE 语句上的 T-SQL 关键字 NATIVE_COMPILATION 是如何创建本地编译的存储过程。 每次数据库联机循环中首次使用本机进程时,会将 T-SQL 语句编译为机器代码。 T-SQL 指令不再接受对每个指令的慢速解释。

  • 我们看到本机编译结果所需的持续时间是解释持续时间的 1/100。

本机模块只能引用内存优化表,而不能引用基于磁盘的表。

有三种类型的本机编译模块:

Azure SQL 数据库中的可用性

内存中 OLTP 和列存储在 Azure SQL 数据库中可用。 有关详细信息,请参阅 Optimize Performance using In-Memory Technologies in SQL Database(在 SQL 数据库中使用内存中技术优化性能)。

1.确保兼容级别 >= 130

本部分一开始有编号的几节说明可用于实现内存中 OLTP 功能的 Transact-SQL 语法。

首先,应将数据库的兼容级别至少设置为 130。 下面是 T-SQL 代码,用于查看当前数据库所设置的当前兼容级别。

SELECT d.compatibility_level
    FROM sys.databases as d
    WHERE d.name = Db_Name();

下面是用于更新级别的 T-SQL 代码(如有必要)。

ALTER DATABASE CURRENT
    SET COMPATIBILITY_LEVEL = 130;

2.提升为快照

当事务同时涉及基于磁盘的表和内存优化表时,我们将该事务称为跨容器事务。 在此类事务中,事务的内存优化部分必须以名为“快照”的事务隔离级别运行。

若要可靠地对跨容器事务中的内存优化表强制执行此级别,请通过执行以下 T-SQL 来变更数据库设置

ALTER DATABASE CURRENT
    SET MEMORY_OPTIMIZED_ELEVATE_TO_SNAPSHOT = ON;

3.创建优化文件组

在 Microsoft SQL Server 中,你必须先创建声明 CONTAINS MEMORY_OPTIMIZED_DATA 的文件组,然后才能创建内存优化表。 将该文件组分配给你的数据库。 有关详细信息,请参阅:

在 Azure SQL 数据库中,无需且不能创建此类文件组。

下面的示例 T-SQL 脚本可为内存中 OLTP 启用数据库,并配置所有推荐设置。 它适用于 SQL Server 和 Azure SQL 数据库:enable-in-memory-oltp.sql

请注意,具有 MEMORY_OPTIMIZED_DATA 文件组的数据库并非支持所有的 SQL Server 功能。 有关限制的详细信息,请参阅:内存中 OLTP 不支持的 SQL Server 功能

4.创建内存优化表

重要的 Transact-SQL 关键字为关键字 MEMORY_OPTIMIZED。

CREATE TABLE dbo.SalesOrder
    (
        SalesOrderId   integer   not null   IDENTITY
            PRIMARY KEY NONCLUSTERED,
        CustomerId   integer    not null,
        OrderDate    datetime   not null
    )
        WITH
            (MEMORY_OPTIMIZED = ON,
            DURABILITY = SCHEMA_AND_DATA);

针对内存优化表的 Transact-SQL INSERT 和 SELECT 语句与针对常规表的这些语句相同。

针对内存优化表的 ALTER TABLE

ALTER TABLE...ADD/DROP 可以在内存优化表中添加或删除列或索引。

  • 不能对内存优化表运行 CREATE INDEX 和 DROP INDEX,但可使用 ALTER TABLE...ADD/DROP INDEX。
  • 有关详细信息,请参阅 变更内存优化表

规划内存优化表和索引

5.创建本机编译的存储过程(本机过程)

重要的关键字为 NATIVE_COMPILATION。

CREATE PROCEDURE ncspRetrieveLatestSalesOrderIdForCustomerId  
        @_CustomerId   INT  
        WITH  
            NATIVE_COMPILATION,  
            SCHEMABINDING  
    AS  
    BEGIN ATOMIC  
        WITH  
            (TRANSACTION ISOLATION LEVEL = SNAPSHOT,
            LANGUAGE = N'us_english')  
      
        DECLARE @SalesOrderId int, @OrderDate datetime;
      
        SELECT TOP 1  
                @SalesOrderId = s.SalesOrderId,  
                @OrderDate    = s.OrderDate  
            FROM dbo.SalesOrder AS s  
            WHERE s.CustomerId = @_CustomerId  
            ORDER BY s.OrderDate DESC;  
      
        RETURN @SalesOrderId;  
    END;  

关键字 SCHEMABINDING 表示,除非先删除本机过程,否则不能删除本机过程中引用的表。 有关详细信息,请参阅创建本机编译的存储过程

请注意,无需创建本机编译存储过程即可访问内存优化表。 也可从传统的存储过程和即席批处理引用内存优化表。

6.执行本机过程

使用两行数据填充表。

INSERT into dbo.SalesOrder  
        ( CustomerId, OrderDate )  
    VALUES  
        ( 42, '2013-01-13 03:35:59' ),
        ( 42, '2015-01-15 15:35:59' );

然后,对本机编译的存储过程调用 EXECUTE。

DECLARE @LatestSalesOrderId int, @mesg nvarchar(128);
      
EXECUTE @LatestSalesOrderId =  
    ncspRetrieveLatestSalesOrderIdForCustomerId 42;
      
SET @mesg = CONCAT(@LatestSalesOrderId,  
    ' = Latest SalesOrderId, for CustomerId = ', 42);
PRINT @mesg;  

以下实际的 PRINT 输出:

-- 2 = Latest SalesOrderId, for CustomerId = 42  

文档指南和后续步骤

前面的简单示例为你学习内存中 OLTP 的更高级功能打下基础。 以下各节是可能需要知道的特殊注意事项,以及在哪里可以查看每个注意事项的相关详细信息的指南。

内存中 OLTP 功能如何执行得这样快

以下各小节简要介绍内存中 OLTP 功能如何在内部工作以提高性能。

内存优化表如何执行速度更快

双重特性: 内存优化表具有双重特性:在活动内存中是一种表示形式,在硬盘上是另一种表示形式。 每个事务将提交到该表的这两种表示形式。 事务针对更快的活动内存表示形式运行。 内存优化表从活动内存比磁盘速度更快受益。 此外,活动内存的更大灵活性使得实现针对速度进行优化的更高级表结构切实可行。 高级结构也是无页的,因此它可以避免闩锁和自旋锁的开销和争用。

无锁: 内存优化表依赖于 乐观 方法来实现数据完整性与并发性和高吞吐量这两个对立的目标。 在事务处理期间,该表不在任何版本的已更新数据行上放置锁。 在某些大容量系统中,这可以大大减少争用。

行版本: 内存优化表不使用锁,而是在表本身中(不是在 tempdb 中)添加新版本的已更新行。 原始行将一直保留,直到提交事务之后。 在事务处理期间,其他进程可以读取行的原始版本。

  • 为基于磁盘的表创建某一行的多个版本时,行版本将暂时存储在 tempdb 中。

更少的日志记录: 已更新行的之前和之后版本将保存在内存优化表中。 行对提供了很多通常写入到日志文件的信息。 这使得系统将更少信息更少次数地写入到日志。 还确保事务完整性。

本机过程如何执行速度更快

将常规解释型存储过程转换为本机编译的存储过程可大大减少要在运行时执行的指令数。

内存中功能的权衡

正如在计算机科学中所常见的,内存中功能提供的性能提升是一种折中方案。 相比功能的额外成本,更好的功能带来的好处要更有价值得多。 可在此处找到关于权衡的全面指南:

本部分的剩余部分列出了一些主要规划和权衡注意事项。

内存优化表的权衡

估计内存: 你必须估计内存优化表要使用的活动内存量。 计算机系统必须具有足够的内存容量,才能托管内存优化表。 有关详细信息,请参阅:

对大型表分区: 满足大量活动内存需求的一种方法是将大型表划分为几部分,内存中的部分存储 热度高的最近 数据行,磁盘上的其他部分存储 冷旧 数据行(如已完全发货和已完成的销售订单)。 此分区是一个需要设计和实现的手动过程。 请参阅:

本机过程的权衡

  • 本机编译的存储过程不能访问基于磁盘的表。 本机过程只能访问内存优化表。
  • 如果本机过程在服务器或数据库最近一次重新联机后第一次运行,则本机过程必须重新编译一次。 这会导致在本机过程开始运行前出现延迟。

内存优化表的高级注意事项

内存优化表的索引 在某些方面与传统的磁盘上表中的索引有所不同。 仅内存优化表可使用哈希索引。

你必须计划以确保有足够的活动内存可用于计划的内存优化表及其索引。 请参阅:

可使用 DURABILITY = SCHEMA_ONLY 声明内存优化表:

  • 此语法指示系统,在数据库已脱机时,丢弃内存优化表中的所有数据。 仅保留表定义。
  • 当数据库重新联机时,将内存优化表重新加载到活动内存中,数据为空。
  • 当涉及到成千上万个行时,SCHEMA_ONLY 表可作为 tempdb 中 #temporary 表 的上级替代项。

也可以将表变量声明为内存优化变量。 请参阅:

本机编译模块的高级注意事项

通过 Transact-SQL 提供的本机编译模块的类型包括:

本机编译的用户定义函数 (UDF) 比解释型 UDF 运行速度更快。 下面是一些关于 UDF 的注意事项:

  • 当 T-SQL SELECT 使用 UDF 时,始终会在返回每行后调用 UDF。
    • UDF 从不以内联方式运行,而是始终被调用。
    • 编译的区别与所有 UDF 固有的重复调用的开销相比,并没有那么重要。
    • 尽管如此,UDF 调用的开销在实用级别上通常是可接受的。

有关本机 UDF 性能的测试数据和说明,请参阅:

内存优化表的文档指南

请参阅讨论内存优化表的特殊注意事项的其他文章:

本机过程的文档指南

以下文章及其在目录 (TOC) 中的子文章详细介绍了本机编译的存储过程。

以下文章介绍了某些代码,演示了通过使用内存中 OLTP 可实现的性能提升: