马上加入IBC程序猿 各种源码随意下,各种教程随便看! 注册 每日签到 加入编程讨论群

C#教程 ASP.NET教程 C#视频教程程序源码享受不尽 C#技术求助 ASP.NET技术求助

【源码下载】 社群合作 申请版主 程序开发 【远程协助】 每天乐一乐 每日签到 【承接外包项目】 面试-葵花宝典下载

官方一群:

官方二群:

缺失索引自动创建语句

[复制链接]
查看3664 | 回复0 | 2016-4-18 14:42:05 | 显示全部楼层 |阅读模式
【编者按】 本文主要介绍使用系统 SQL 实体自动创建非聚集(non-clustered)索引。作者为意大利软件工程师 GhostHost(笔名)。
本文系 OneAPM 工程师编译呈现,以下为正文。

引言
一直以来,关于索引的常见问题是:判断哪部分索引对保证数据库的良好性能是必需的。在本文中,笔者将提供针对该问题的解决方案。本文用例中的所有代码都基于名为dm_db_missing_index_details 的 SQL Server 系统视图。

背景
在开始安装前,进一步了解 dm_db_missing_index_details 会更有益处。
dm_db_missing_index_details 会返回缺失索引的细节信息。在本文中,我们将更关注以下几列:

  • index_handle:它是一个独特的跨服务器标识符,并且标志一个特定的缺失索引。
  • equality_columns:包含用于相等谓词的所有列
  • inequality_columns:包含用于其他比较的所有列
  • included columns索引中所包含的查询必要出现列
  • statement: 补充完整索引缺失的表名

实现
本系统的实现基于以下三个实体:

  • 一个可以计算出待创建索引名称的简单函数
  • 一个用于简化 dm_db_missing_index_details的用户视图
  • 一个为每个索引创建声明的进程
笔者选择将这个系统分为三段进程,但实际上合并存储过程和视图也是可行的。笔者之所以没有选择后一种做法是因为想在创建索引之前先从业务逻辑检查一下存在哪些索引。

完整代码

  •   
    [SQL] 纯文本查看 复制代码
      -- CREATE FUNCTION fn_Index_CreateIndexName
        CREATE FUNCTION [dbo].[fn_Index_CreateIndexName] (@equality_columns NVARCHAR(4000), _
        @Inequality_columns NVARCHAR(4000), @index_handlE INT) RETURNS VARCHAR(128)
        AS
        BEGIN
                DECLAR
        E @IndexName NVARCHAR(MAX)
            SET @IndexName = ISNULL(@equali
        ty_columns,@Inequality_columns)
        SET @IndexName = LTRIM(REPLACE(@IndexName,'[','_'))
        SET @IndexName = RTRIM(REPLACE(@IndexName,']','_'))
        SET @IndexName = REPLACE(@IndexName,',','')
        SET @IndexName = REPLACE(@IndexName,'_ _','_')
            IF LEN(@IndexName) > 120
            BEGIN
                SET @IndexName = SUBSTRING(@IndexName,0,120)
            END  
            SET @IndexName = @IndexName + CAST(@index_handlE AS NVARCHAR(15))
            RETURN @IndexName 
        END
        GO
        -- CREATE FUNCTION vw_Index_MissingIndex
        CREATE VIEW [dbo].[vw_Index_MissingIndex] 
        AS
        SELECT  '[' + d.name + ']' as DBName,
                [dbo].[fn_Index_CreateIndexName]_
                (mid.equality_columns,mid.Inequality_columns,mid.index_handlE) AS ID,
                REPLACE(mid.equality_columns,',',' ASC,') AS equality_columns,
                REPLACE(mid.Inequality_columns,',',' ASC,') AS Inequality_columns,
                mid.Included_columns,
                mid.[statement]
        FROM sys.dm_db_missing_index_details mid
        INNER JOIN sys.databases d
        on d.database_id = mid.database_id
        GO
        CREATE PROCEDURE [dbo].[usp_Index_MissingIndexCreationStatements]
        AS
        DECLARE @IndexCreationPlaceholder_Start  AS NVARCHAR(MAX)
        DECLARE @IndexCreationPlaceholder_End  AS NVARCHAR(MAX)
        -- PREPARE PLACEHOLDER
        SET @IndexCreationPlaceholder_Start = 'IF NOT EXISTS_
        (SELECT * FROM {2}.sys.indexes WHERE [name] = ''IX_{0}'' )
                        BEGIN
                        CREATE NONCLUSTERED INDEX [IX_{0}] ON {1}'
        SET @IndexCreationPlaceholder_End = ' WITH (PAD_INDEX = OFF, _
        STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, DROP_EXISTING = OFF, _
        ONLINE = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
                        END;' + char(13) + char(10)
        -- STATEMENT CREATION
        SELECT
            DBName,
            CASE
            WHEN NOT mid.equality_columns IS NULL AND NOT mid.Inequality_columns IS NULL THEN
                        REPLACE(REPLACE(REPLACE(@IndexCreationPlaceholder_Start,'{0}', _
                        mid.ID),'{1}',mid.[statement]),'{2}',mid.DBName)
                        + '
                           ( ' +
                           COALESCE(mid.equality_columns,'') +
                           ' ASC,' + 
                           COALESCE(mid.Inequality_columns,'') +
                           ' ASC
                        )' +
                        COALESCE('INCLUDE ( ' + mid.Included_columns + ' ) ','')
                        + @IndexCreationPlaceholder_End
                WHEN mid.equality_columns IS NULL AND NOT mid.Inequality_columns IS NULL THEN
                        REPLACE(REPLACE(REPLACE(@IndexCreationPlaceholder_Start,'{0}', _
        mid.ID),'{1}',mid.[statement]),'{2}',mid.DBName)
                    + '
                       ( ' +
                       COALESCE(mid.Inequality_columns,'') +
                       ' ASC
                    ) ' +
                    COALESCE('INCLUDE ( ' + mid.Included_columns + ' ) ','')
                    + @IndexCreationPlaceholder_End
            WHEN NOT mid.equality_columns IS NULL AND mid.Inequality_columns IS NULL THEN
                    REPLACE(REPLACE(REPLACE(@IndexCreationPlaceholder_Start,'{0}', _
                    mid.ID),'{1}',mid.[statement]),'{2}',mid.DBName)
                    + '
                       ( ' +
                   COALESCE(mid.equality_columns,'') +  ' ASC
                        ) ' +
                    COALESCE('INCLUDE ( ' + mid.Included_columns + ' ) ','')
                    + @IndexCreationPlaceholder_End
            ELSE NULL
        END AS Index_Creation_Statement,
        ' DROP INDEX [IX_' + mid.ID  + '] ON ' + mid.[statement]  _
            +  + char(13) + char(10) AS Index_Drop_Statement
        FROM [dbo].[vw_Index_MissingIndex] AS mid
        GO

OneAPM 助您轻松锁定 .NET 应用性能瓶颈,通过强大的 Trace 记录逐层分析,直至锁定行级问题代码。以用户角度展示系统响应速度,以地域和浏览器维度统计用户使用情况。想阅读更多技术文章,请访问 OneAPM 官方博客

+





*滑块验证:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则