sql-server – 选择Indexed View的Clustered Index有哪些因素?

简要地
他们查询优化器选择索引视图索引的因素是什么?

对我来说,索引视图似乎无法理解我对Optimizer如何选择索引的理解.我见过this asked before,但OP并不太受欢迎.我真的在寻找路标,但我会编写一个伪示例,然后使用大量的DDL,输出和示例发布真实示例.

Assume I’m using Enterprise 2008+,understand with(noexpand)

伪示例

以这个伪示例为例:我创建了一个包含22个连接,17个过滤器和一个跨越一堆1000万行表的马戏团小马的视图.这个观点是昂贵的(是的,有大写字母E)来实现.我将SCHEMABIND和索引视图.然后是SELECT a,b FROM AnIndexedView WHERE theClusterKeyField< 84.在优化器逻辑中,我无法执行基础连接. 结果:
>没有提示:4825读取720行,47 cpu超过76ms,和
估计子树成本为0.30523.
>使用提示:17个读取,720行,超过4ms的15个cpu,以及一个
估计子树成本为0.007253

那么这里发生了什么?我已经在Enterprise 2008,2008-R2和2012中尝试过了.我可以想到使用视图索引的每个指标都非常有效.我没有参数嗅探问题或倾斜数据,因为这是广告.

真实(长)的例子

除非你是一个自虐的人,否则你可能不需要或不想阅读这部分内容.

版本
是的,企业.

Microsoft sql Server 2012 – 11.0.2100.60 (X64) Feb 10 2012 19:39:15
copyright (c) Microsoft Corporation Enterprise Edition (64-bit) on
Windows NT 6.2 (Build 9200: ) (Hypervisor)

风景

CREATE VIEW dbo.TimelineMaterialized    WITH SCHEMABINDING
AS
SELECT  TM.TimelineID,TM.TimelineTypeID,TM.EmployeeID,TM.CreateUTC,CUL.CultureCode,CASE 
           WHEN TM.CustomerMessageID    > 0 THEN TM.CustomerMessageID
           WHEN TM.CustomerSessionID    > 0 THEN TM.CustomerSessionID
           WHEN TM.NewItemTagID         > 0 THEN TM.NewItemTagID
           WHEN TM.OutfitID             > 0 THEN TM.OutfitID
           WHEN TM.ProductTransactionID > 0 THEN TM.ProductTransactionID
           ELSE 0 END  As HrefId,CASE 
          WHEN TM.CustomerMessageID    > 0 THEN IsNull(C.Name,'N/A')   
          WHEN TM.CustomerSessionID    > 0 THEN IsNull(C.Name,'N/A')
          WHEN TM.NewItemTagID         > 0 THEN IsNull(NI.Title,'N/A')
          WHEN TM.OutfitID             > 0 THEN IsNull(O.Name,'N/A')
          WHEN TM.ProductTransactionID > 0 THEN IsNull(PT_PL.NameLocalized,'N/A')
                 END as HrefText

FROM       dbo.Timeline TM
INNER JOIN dbo.CustomerSession    CS    ON TM.CustomerSessionID    = CS.CustomerSessionID
INNER JOIN dbo.CustomerMessage    CM    ON TM.CustomerMessageID    = CM.CustomerMessageID
INNER JOIN dbo.Outfit             O     ON PO.OutfitID             = O.OutfitID
INNER JOIN dbo.ProductTransaction PT    ON TM.ProductTransactionID = PT.ProductTransactionID
INNER JOIN dbo.Product            PT_P  ON PT.ProductID            = PT_P.ProductID
INNER JOIN dbo.ProductLang        PT_PL ON PT_P.ProductID          = PT_PL.ProductID
INNER JOIN dbo.Culture            CUL   ON PT_PL.CultureID         = CUL.CultureID
INNER JOIN dbo.NewsItemTag        NIT   ON TM.NewsItemTagID        = NIT.NewsItemTagID
INNER JOIN dbo.NewsItem           NI    ON NIT.NewsItemID          = NI.NewsItemID
INNER JOIN dbo.Customer           C     ON  C.CustomerID = CASE 
                                             WHEN TM.TimelineTypeID = 1 THEN CM.CustomerID 
                                             WHEN TM.TimelineTypeID = 5 THEN CS.CustomerID
                                             ELSE 0 END

WHERE        CUL.IsActive = 1

聚集指数

CREATE UNIQUE CLUSTERED INDEX PK_TimelineMaterialized  ON 
                   TimelineMaterialized (EmployeeID,CreateUTC,CultureCode,TimelineID)

测试sql

-- NO HINT - - -  - - -  - - -  - - -  - - - 
SELECT  *                 --yes yes,star is bad ...just a test example
FROM    TimelineMaterialized TM 
WHERE 
            TM.EmployeeID   = 2
        AND TM.CultureCode  = 'en-US'
        AND TM.CreateUTC    > '9/10/2012'
        AND TM.CreateUTC    < '9/11/2012'

-- WITH HINT - - -  - - -  - - -  - - -  - - - 
SELECT  *               
FROM    TimelineMaterialized TM with(noexpand)
WHERE 
            TM.EmployeeID   = 2
        AND TM.CultureCode  = 'en-US'
        AND TM.CreateUTC    > '9/10/2012'
        AND TM.CreateUTC    < '9/11/2012'

结果= 11行输出

Profiler输出
前4行没有提示.底部的4行是使用提示.

执行计划
GitHub Gist for both Execution Plans in SQLPlan format

没有提示执行计划 – 为什么不使用我给你的sql先生的聚集索引?它集中在3个过滤器字段上.尝试一下,你可能会喜欢它.

使用提示时的简单计划.

解决方法

匹配索引视图是一个相对昂贵的操作*,因此优化器首先尝试其他快速简单的转换.如果那些碰巧产生了一个便宜的计划(在你的情况下为0.05个单位),优化会提前结束.赌注是持续优化将消耗比保存更多的时间.请记住,优化器的主要目标是快速实现“足够好”的计划.

在视图上使用聚簇索引本身并不昂贵,但是将逻辑查询树与潜在的索引视图匹配的过程可以是.正如我在另一个问题的评论中提到的,查询中的视图引用在优化之前被扩展,因此优化器不知道你首先在视图上编写了查询 – 它只看到了展开的树(就像视图已被列入内容).

“足够好的计划”意味着优化者找到了一个体面的计划,并在探索阶段提前停止. “TimeOut”意味着它超过了在当前阶段开始时将其自身设置为“预算”的优化步骤数.

预算是根据前一阶段中找到的最佳计划的成本设定的.使用如此低成本的查询(0.05),预算移动的数量将非常小,并且在给定样本查询中涉及的连接数量的情况下通过常规转换很快耗尽(例如,有很多方法可以重新排列内部连接) .

如果您有兴趣了解为什么索引视图匹配代价昂贵,因此留待优化的后期阶段和/或仅考虑更昂贵的查询,则有关于主题here(pdf)和here(citeseer)的两篇Microsoft研究论文).

一个相关因素是索引视图匹配在优化阶段0(事务处理)中不可用.

进一步阅读:

Indexed Views and Statistics

*仅适用于企业版(或同等版本)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


SELECT a.*,b.dp_name,c.pa_name,fm_name=(CASE WHEN a.fm_no=&#39;LJCG001H&#39; THEN dbo.ELTPNAME(a.fw_nu) ELSE d.fm_name END),e.fw_state_nm,f.fw_rmk_nm
if not exists(select name from syscolumns where name=&#39;tod_no&#39; and id=object_id(&#39;iebo09d12&#39;)) alter table iebo09d12 add tod_no varchar(
select a.*,pano=a.pa_no,b.pa_name,f.dp_name,e.fw_state_nm,g.fa_name from LJSS007H a (nolock) Left join LJPA002H b (nolock) On a.pa_no =b.pa_no Left jo
要在 SQL Server 2019 中设置定时自动重启,可以使用 Windows 任务计划程序。下面是详细的步骤: 步骤一:创建批处理文件 打开记事本。 输入以下内容: net stop &quot;SQL Server (MSSQLSERVER)&quot; net start &quot;SQ
您收到的错误消息表明数据库 &#39;EastRiver&#39; 的事务日志已满,导致数据库操作失败。要解决这个问题,可以按照以下步骤操作: 1. 备份事务日志首先,备份事务日志以释放空间: BACKUP LOG [EastRiver] TO DISK = N&#39;C:\Backup\East
首先我需要查询出需要使用SQL Server Profiler跟踪的数据库标识ID,若不知道怎么查询数据库的标识ID, 打开SQL Server management studio,点击工具。选择SQL Server Profiler。 登录,登录成功后,如果有个默认弹窗,先取消 新建追踪 命名
--最新的解决方法 --先创建用户帐户,不进行授权,然后通过下面的SQL语句将该用户帐户关联至对应的数据库用户。优点是避免了重新授权的操作。 USE tempdbEXEC sp_change_users_login &#39;Update_One&#39;, &#39;iemis&#39;, &#3
命令: ALTER TABLE 表名 add 列名 数据类型 default 默认值 not null 例如: ALTER TABLE LJEL005H add el_req int default 15 not null
declare @i int set @i=340 while @i&lt;415 begin set @i=@iʱ insert into LJWK007H select &#39;2024&#39;,&#39;28&#39;,&#39;9110&#39;,&#39;3PTSD621000000
alter table LJSU002H add default (0) for su_totalamt with values
命令: 有默认值时用:alter table 表名 DROP 约束 alter table 表名 DROP COLUMN 列名 例如: alter table LJEL005H DROP COLUMN el_req 注意: 如果有默认值先删除约束,否则报错: --消息 5074,级别 16,状态 1
mysql报错Unknown collation: utf8mb4_0900_ai_ci 解决方案: 将文件内的所有 utf8mb4_0900_ai_ci 换成 utf8_general_ci utf8mb4 换成 utf8
SQL SERVER根据数据表的某个栏位查询另一个数据表符合条件的某个栏位的值,如果多行则合并为一张字符串形式 要在 SQL Server 中根据一个数据表的某个列查询另一个数据表符合条件的某个列的值,并将多行结果合并为一个字符串,可以使用 FOR XML PATH 子句来执行此操作。以下是一个示例
ALTER TABLE LJPA001H DROP CONSTRAINT DF_LJPA001H_pa_sex_1 ALTER TABLE LJPA001H ALTER COLUMN pa_sex VARCHAR(1)
DATEDIFF和DATEADD函数。DATEDIFF函数计算两个日期之间的小时、天、周、月、年等时间间隔总数。DATEADD函数计算一个日期通过给时间间隔加减来获得一个新的日期。要了解更多的DATEDIFF和DATEADD函数以及时间间隔可以阅读微软联机帮助。使用DATEDIFF和DATEADD函
select top 100 substring(qr_code,8,8) ,* from [LiuJun_PKh_lcfc_hf] --where right(ri,8) or substring(qr_code,8,8)=. select top 10 left(right(one_code,1
1、发现事务日志备份突然停止了 2、查看维护计划中的事务日志设置 3、发现备份任务中,事务日志需要指向的数据库不在 4、进入数据库属性 5、在选项中将恢复模式改为“完整”
select DB_ID(&#39;SBTERPDB&#39;)
您收到的错误消息表明数据库 &#39;EastRiver&#39; 的事务日志已满,导致数据库操作失败。要解决这个问题,可以按照以下步骤操作: 1. 备份事务日志首先,备份事务日志以释放空间: BACKUP LOG [EastRiver] TO DISK = N&#39;C:\Backup\East