SQL Server 中的树结构查询

如何解决SQL Server 中的树结构查询

使用 Azure SQL Server,我有一个表将组织的树结构存储在一个表中,如下所示:

    CREATE TABLE [dbo].[UserManagement_GroupDef]
    (
        [ID] [bigint] IDENTITY(1,1) NOT NULL,[PortalId] [bigint] NOT NULL,[GroupType] [int] NOT NULL,[ParentGroupId] [bigint] NULL,[GroupName] [nvarchar](2048) NOT NULL,[IsDefault] [bit] NOT NULL,[Email] [nvarchar](256) NULL,[PhoneNumber] [varchar](10) NULL,[UserManagementId] [nvarchar](450) NULL,[Address] [nvarchar](450) NULL,[Suite] [nvarchar](450) NULL,[City] [nvarchar](450) NULL,[State] [nvarchar](450) NULL,[Position] [nvarchar](450) NULL,[Zip] [nvarchar](450) NULL,[IsDeleted] [bit] NOT NULL,CONSTRAINT [PK_UserManagement_GroupDef] PRIMARY KEY CLUSTERED     
   (
    [ID] ASC
   )WITH (STATISTICS_NORECOMPUTE = OFF,IGNORE_DUP_KEY = OFF) ON [PRIMARY]
   ) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]

所有列都被编入索引。

[ParentGroupId] 引用同一表中的另一行。总而言之,该表代表了一个树结构。

在下面的查询中,我试图找到组 ID 155 的所有后代。

WITH tblChild AS
    (
        SELECT 
        ID,ParentGroupId,[IsDeleted],[PortalId],[GroupType],[GroupName],[IsDefault],[Email],[PhoneNumber],[UserManagementId],[Address],[Suite],[City],[State],[Position],[Zip]
            FROM UserManagement_GroupDef    
            WHERE ID = 155 AND IsDeleted = 0
        UNION ALL
        SELECT 
             UserManagement_GroupDef.ID,UserManagement_GroupDef.ParentGroupId,UserManagement_GroupDef.[IsDeleted],UserManagement_GroupDef.[PortalId],UserManagement_GroupDef.[GroupType],UserManagement_GroupDef.[GroupName],UserManagement_GroupDef.[IsDefault],UserManagement_GroupDef.[Email],UserManagement_GroupDef.[PhoneNumber],UserManagement_GroupDef.[UserManagementId],UserManagement_GroupDef.[Address],UserManagement_GroupDef.[Suite],UserManagement_GroupDef.[City],UserManagement_GroupDef.[State],UserManagement_GroupDef.[Position],UserManagement_GroupDef.[Zip]

        FROM UserManagement_GroupDef  
        JOIN 
            tblChild  
                ON 
                    UserManagement_GroupDef.ParentGroupId = tblChild.ID
                WHERE tblChild.IsDeleted = 0        
    )

    SELECT 
            tblChild.ID,tblChild.ParentGroupId,tblChild.[IsDeleted],tblChild.[PortalId],tblChild.[GroupType],tblChild.[GroupName],tblChild.[IsDefault],tblChild.[Email],tblChild.[PhoneNumber],tblChild.[UserManagementId],tblChild.[Address],tblChild.[Suite],tblChild.[City],tblChild.[State],tblChild.[Position],tblChild.[Zip]
        FROM tblChild       
    

该表目前只有不到 7000 条记录。此特定查询返回的结果不到 2000 行。结果最终是正确的,但执行需要大约 20 秒。

有没有办法加快这个查询或达到相同的结果,只是更快?

执行计划:

https://www.brentozar.com/pastetheplan/?id=S1k0Qdrzd

表格索引:

CREATE NONCLUSTERED INDEX [IX_UserManagement_GroupDef] ON [dbo].[UserManagement_GroupDef]
(
    [IsDeleted] ASC
)WITH (STATISTICS_NORECOMPUTE = OFF,DROP_EXISTING = OFF,ONLINE = OFF) ON [PRIMARY]

CREATE NONCLUSTERED INDEX [IX_UserManagement_GroupDef_2] ON [dbo].[UserManagement_GroupDef]
(
    [ParentGroupId] ASC
)WITH (STATISTICS_NORECOMPUTE = OFF,ONLINE = OFF) ON [PRIMARY]

CREATE NONCLUSTERED INDEX [IX_UserManagement_GroupDef_3] ON [dbo].[UserManagement_GroupDef]
(
    [IsDeleted] ASC
)WITH (STATISTICS_NORECOMPUTE = OFF,ONLINE = OFF) ON [PRIMARY]


CREATE NONCLUSTERED INDEX [umg_query_index] ON [dbo].[UserManagement_GroupDef]
(
    [ParentGroupId] ASC
)
INCLUDE(

[ID],[Zip]
)

WHERE ([IsDeleted]=(0))

WITH (STATISTICS_NORECOMPUTE = OFF,ONLINE = OFF) ON [PRIMARY]

解决方法

看起来正在发生的事情是编译器无法将外部谓词IsDeleted = 0下推到CTE中。因此它不能使用 O. Jones 推荐的 umg_query_index

为什么会这样,是因为递归部分的每次运行都可能返回 IsDeleted = 1 行,这些行需要反馈到下一次运行中。虽然它们确实不会出现在最终结果中,但这样的行仍然可能具有确实需要出现在最终结果集中的子行。所以没有办法从递归连接中消除它们。

您有两个选择:

  1. 更改您的 IX_UserManagement_GroupDef_2 索引,使其也包含 IsDeleted 列,它不需要在键中,它可以是 INCLUDE
CREATE NONCLUSTERED INDEX [IX_UserManagement_GroupDef_2] ON [dbo].[UserManagement_GroupDef]
([ParentGroupId] ASC) INCLUDE (IsDeleted)
WITH (DROP_EXISTING = ON,ONLINE = ON) ON [PRIMARY]
  1. 可能是更好的选择,自己下推谓词。然后它将在 CTE 的两个部分使用 O. Jones 索引。
WITH tblChild AS
(
    SELECT *
    FROM UserManagement_GroupDef 
    WHERE ID = 155 AND IsDeleted = 0                

    UNION ALL

    SELECT 
        u.* 
    FROM 
        UserManagement_GroupDef u
    JOIN 
        tblChild ON u.ParentGroupId = tblChild.ID
    WHERE u.IsDeleted = 0
)
SELECT *
FROM tblChild 

此查询与原始查询具有不同的语义。它不会返回父项为 IsDeleted = 1 的任何行。

此时您还可以将新索引更改为过滤索引,因为这意味着不会存储任何 IsDeleted 行。

CREATE NONCLUSTERED INDEX umg_query_index ON UserManagement_GroupDef
(ParentGroupId)  INCLUDE (IsDeleted) WHERE IsDeleted = 0
WITH (DROP_EXISTING = ON,ONLINE = ON) ON [PRIMARY];

您必须在索引中包含 IsDeleted 列,否则由于当前实现的逻辑错误,优化器可能无法使用它。


进一步说明:单列索引通常没有那么有用,一般应该避免。服务器尝试将两个索引合并在一起是不值得的,它会扫描聚集索引。

我建议您删除索引 IX_UserManagement_GroupDefIX_UserManagement_GroupDef_3,因为它们现在完全是多余的。

,

试试这个:

CREATE INDEX umg_query_index ON UserManagement_GroupDef (IsDeleted,ParentGroupId);

更好的是,在 SSMS 中运行查询。在运行之前,右键单击查询窗口并启用 Show Actual Execution Plan。

接着看执行计划。它可能会为您推荐合适的索引。

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

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams['font.sans-serif'] = ['SimHei'] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -> systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping("/hires") public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate<String
使用vite构建项目报错 C:\Users\ychen\work>npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)> insert overwrite table dwd_trade_cart_add_inc > select data.id, > data.user_id, > data.course_id, > date_format(
错误1 hive (edu)> insert into huanhuan values(1,'haoge'); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive> show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 <configuration> <property> <name>yarn.nodemanager.res