use*_*056 3 sql sql-server merge sqlperformance
我有一个包含 500 万行的数据库表。聚集索引是自增标识列。PK 是生成 256 字节的代码VARCHAR,它是 URL 的 SHA256 哈希,这是表上的非聚集索引。
表格如下:
CREATE TABLE [dbo].[store_image](
[imageSHAID] [nvarchar](256) NOT NULL,
[imageGUID] [uniqueidentifier] NOT NULL,
[imageURL] [nvarchar](2000) NOT NULL,
[showCount] [bigint] NOT NULL,
[imageURLIndex] AS (CONVERT([nvarchar](450),[imageURL],(0))),
[autoIncID] [bigint] IDENTITY(1,1) NOT NULL,
CONSTRAINT [PK_imageSHAID] PRIMARY KEY NONCLUSTERED
(
[imageSHAID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO
CREATE CLUSTERED INDEX [autoIncPK] ON [dbo].[store_image]
(
[autoIncID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, IGNORE_DUP_KEY = OFF, DROP_EXISTING = OFF, ONLINE = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
GO
Run Code Online (Sandbox Code Playgroud)
imageSHAID是图像 URL 的 SHA256 哈希,例如“ http://blah.com/image1.jpg ”,它被哈希为 256 长度的 varchar。
imageGUID 是代码生成的 guid,我在其中标识图像(稍后将用作索引,但现在我已省略此列作为索引)
imageURL 是图像的完整 URL(最多 2000 个字符)
showCount 是图像显示的次数,每次显示此特定图像时都会增加。
imageURLIndex 是一个限制为 450 个字符的计算列,这允许我在我选择的 imageURL 上进行文本搜索,它是可索引的(再次为简洁起见省略了索引)
autoIncID 是聚集索引,应该允许更快地插入数据。
我定期从临时表合并到store_image表中。temp表结构如下(与store_image表非常相似):
CREATE TABLE [dbo].[store_image_temp](
[imageSHAID] [nvarchar](256) NULL,
[imageURL] [nvarchar](2000) NULL,
[showCount] [bigint] NULL,
) ON [PRIMARY]
GO
Run Code Online (Sandbox Code Playgroud)
运行合并过程时,我DataTable使用以下代码将 a 写入临时表:
using (SqlBulkCopy bulk = new SqlBulkCopy(storeConn, SqlBulkCopyOptions.KeepIdentity | SqlBulkCopyOptions.KeepNulls, null))
{
bulk.DestinationTableName = "[dbo].[store_image_temp]";
bulk.WriteToServer(imageTableUpsetDataTable);
}
Run Code Online (Sandbox Code Playgroud)
我然后运行合并命令来更新showCount在store_image通过基于该临时表合并表格imageSHAID。如果该图像当前不存在于store_image表中,我会创建它:
merge into store_image as Target using [dbo].[store_image_temp] as Source
on Target.imageSHAID=Source.imageSHAID
when matched then update set
Target.showCount=Target.showCount+Source.showCount
when not matched then insert values (Source.imageSHAID,NEWID(), Source.imageURL, Source.showCount);
Run Code Online (Sandbox Code Playgroud)
我通常尝试store_image在任何一个合并过程中将 2k-5k 行从临时表合并到表中。
我曾经在 SSD(仅连接 SATA 1)上运行此数据库,并且速度非常快(低于 200 毫秒)。我用完了 SSD 上的空间,所以我将数据库交换到 1TB 7200 缓存旋转磁盘,从那时起完成时间超过 6-100 秒(6000 - 100000MS)。当批量插入运行时,我可以看到大约 1MB-2MB/秒的磁盘活动,低 CPU 使用率。
这是这种数据量的典型写入时间吗?对我来说似乎有点慢,是什么导致了性能缓慢?当然,随着imageSHAID被索引,我们应该期待比这更快的搜索时间?
任何帮助,将不胜感激。
谢谢你的时间。
您UPDATE在MERGE更新中的条款showCount。这需要对聚集索引进行键查找。
但是,聚集索引也被声明为非唯一的。即使底层列是唯一的,这也会向优化器提供信息。
所以,我会做出这些改变
autoIncIDimageSHAID成为一个独立的唯一索引(不是约束)并为showCount. 唯一约束不能有 INCLUDE更多观察:
nvarchar哈希或 URL 列。这些不是 unicode。char(64)(对于 SHA2-512)。| 归档时间: |
|
| 查看次数: |
4188 次 |
| 最近记录: |