SQL Server笔记 -- 第68章:内存中 OLTP(Hekaton)

68.1 声明内存优化表变量

为获得更高性能,可将表变量改为内存优化。传统表变量示例:

sql 复制代码
DECLARE @tvp TABLE (
  col1 INT NOT NULL,
  col2 CHAR(10)
);

使用内存优化版本,需先创建内存优化表类型,再声明变量:

sql 复制代码
CREATE TYPE dbo.memTypeTable AS TABLE (
  Col1 INT NOT NULL INDEX ix1,
  Col2 CHAR(10)
) WITH (MEMORY_OPTIMIZED = ON);
GO

DECLARE @tvp memTypeTable;
INSERT INTO @tvp VALUES (1,'1'),(2,'2'),(3,'3'),(4,'4'),(5,'5'),(6,'6');
SELECT * FROM @tvp;

68.2 创建内存优化表

sql 复制代码
-- 创建示例数据库
CREATE DATABASE SQL2016_Demo
ON PRIMARY (
  NAME = N'SQL2016_Demo',
  FILENAME = N'C:\Dump\SQL2016_Demo.mdf',
  SIZE = 5120KB,
  FILEGROWTH = 1024KB
)
LOG ON (
  NAME = N'SQL2016_Demo_log',
  FILENAME = N'C:\Dump\SQL2016_Demo_log.ldf',
  SIZE = 1024KB,
  FILEGROWTH = 10%
);
GO

USE SQL2016_Demo;
GO

-- 添加内存优化文件组
ALTER DATABASE SQL2016_Demo
ADD FILEGROUP MemFG CONTAINS MEMORY_OPTIMIZED_DATA;
GO

-- 向文件组添加文件
ALTER DATABASE SQL2016_Demo
ADD FILE (
  NAME = MemFG_File1,
  FILENAME = N'C:\Dump\MemFG_File1'
) TO FILEGROUP MemFG;
GO

-- 创建内存优化表(持久性可选 SCHEMA_AND_DATA 或 SCHEMA_ONLY)
CREATE TABLE dbo.MemOptTable1 (
  Column1 INT NOT NULL,
  Column2 NVARCHAR(4000) NULL,
  SpidFilter SMALLINT NOT NULL DEFAULT (@@spid),
  INDEX ix_SpidFilter NONCLUSTERED (SpidFilter),
  INDEX ix_SpidFilter HASH (SpidFilter) WITH (BUCKET_COUNT = 64),
  CONSTRAINT CHK_soSessionC_SpidFilter CHECK (SpidFilter = @@spid)
) WITH (
  MEMORY_OPTIMIZED = ON,
  DURABILITY = SCHEMA_AND_DATA
);
GO

CREATE TABLE MemOptTable2 (
  ID INT NOT NULL PRIMARY KEY NONCLUSTERED HASH WITH (BUCKET_COUNT = 10000),
  FullName NVARCHAR(200) NOT NULL,
  DateAdded DATETIME NOT NULL
) WITH (MEMORY_OPTIMIZED = ON, DURABILITY = SCHEMA_AND_DATA);
GO

68.3 查看生成的 DLL 文件与内存优化表

sql 复制代码
SELECT OBJECT_ID('MemOptTable1') AS MemOptTable1_ObjectID,
       OBJECT_ID('MemOptTable2') AS MemOptTable2_ObjectID;
GO

SELECT name, description
FROM sys.dm_os_loaded_modules
WHERE name LIKE '%XTP%';
GO

-- 列出所有内存优化表
SELECT name, type_desc, durability_desc, is_memory_optimized
FROM sys.tables
WHERE is_memory_optimized = 1;
GO

68.4 创建内存优化的系统版本化时态表

sql 复制代码
CREATE TABLE dbo.MemOptimizedTemporalTable (
  BusinessDocNo BIGINT NOT NULL,
  ProductCode INT NOT NULL,
  UnitID TINYINT NOT NULL,
  PriceID TINYINT NOT NULL,
  SysStartTime DATETIME2(7) GENERATED ALWAYS AS ROW START NOT NULL,
  SysEndTime DATETIME2(7) GENERATED ALWAYS AS ROW END NOT NULL,
  PERIOD FOR SYSTEM_TIME (SysStartTime, SysEndTime),
  CONSTRAINT PK_MemOptimizedTemporalTable PRIMARY KEY NONCLUSTERED (
    BusinessDocNo ASC,
    ProductCode ASC
  )
) WITH (
  MEMORY_OPTIMIZED = ON,
  DURABILITY = SCHEMA_AND_DATA,
  SYSTEM_VERSIONING = ON (
    HISTORY_TABLE = History.MemOptimizedTemporalTable_History,
    DATA_CONSISTENCY_CHECK = ON
  )
);

68.5 内存优化表类型与临时表

传统表类型:

sql 复制代码
CREATE TYPE dbo.testTableType AS TABLE (
  col1 INT NOT NULL,
  col2 CHAR(10)
);

内存优化表类型(需加索引):

sql 复制代码
CREATE TYPE dbo.testTableType AS TABLE (
  col1 INT NOT NULL,
  col2 CHAR(10),
  INDEX ix NONCLUSTERED (col1)
) WITH (MEMORY_OPTIMIZED = ON);

全局临时表:

sql 复制代码
CREATE TABLE ##tempGlobalTable (
  Col1 INT NOT NULL,
  Col2 NVARCHAR(4000)
);

对应的内存优化版本(使用 SCHEMA_ONLY 持久性):

sql 复制代码
CREATE TABLE dbo.tempGlobalTable (
  Col1 INT NOT NULL INDEX ix NONCLUSTERED,
  Col2 NVARCHAR(4000)
) WITH (
  MEMORY_OPTIMIZED = ON,
  DURABILITY = SCHEMA_ONLY
);

迁移步骤:

  1. 按相同架构新建 SCHEMA_ONLY 内存优化表,并至少建一个索引
  2. 将代码中对 ##temp 的引用改为新表名
  3. 把 DROP TABLE ##temp 改为 DELETE FROM 新表 以清空数据
  4. 移除原 CREATE TABLE ##temp 语句
相关推荐
ouliten8 小时前
cuda编程笔记(37)--逐行量化的kernel分析
笔记
MimCyan8 小时前
面向开发者的 LLM 入门课程(个人笔记记录-2026.03.30)
笔记·ai
数据库小组8 小时前
2026 年,MySQL 到 SelectDB 同步为何更关注实时、可观测与可校验?
数据库·mysql·数据库管理工具·数据同步·ninedata·selectdb·迁移工具
华科易迅8 小时前
MybatisPlus增删改查操作
android·java·数据库
Hammer_Hans8 小时前
DFT笔记34
笔记
qcwl668 小时前
深入理解Linux进程与内存 学习笔记#4
笔记·学习
Kethy__8 小时前
计算机中级-数据库系统工程师-计算机体系结构与存储系统
大数据·数据库·数据库系统工程师·计算机中级
SHoM SSER9 小时前
MySQL 数据库连接池爆满问题排查与解决
android·数据库·mysql
蒸蒸yyyyzwd9 小时前
后端学习笔记 day4
linux·笔记·学习
熬夜的咕噜猫9 小时前
MySQL备份与恢复
数据库·oracle