RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 733363
Accepted
Anatol
Anatol
Asked:2020-10-20 14:25:10 +0000 UTC2020-10-20 14:25:10 +0000 UTC 2020-10-20 14:25:10 +0000 UTC

在视图查询中应用 WHERE 子句的时间点是什么?

  • 772

查询中的表达式在什么时候应用于WHERE视图,在生成视图的记录集时,或者应用于已经生成的集合,它是否取决于任何因素?

有两种特殊情况感兴趣:

1.在视图主体中的简单请求的情况下
2.在形成视图的请求使用窗口的情况下

示例 1:

CREATE VIEW dbo.vw_StockRemnants
AS
     SELECT tt.PKID AS ИД,
            tt.Name AS Наименование,
            tt.priceIn AS Вход,
            tt.Price AS Цена
     FROM Item AS tt
     WHERE tt.iditem IS NOT NULL;

SELECT *
FROM vw_StockRemnants
WHERE dbo.vw_StockRemnants.[ИД] = 12345;   

示例 2:

CREATE VIEW dbo.vw_StockRemnants
AS
     WITH CTE([ИД],
              [Наименование],
              [Вход],
              [Цена],
              rnumber,
              rcount)
          AS (SELECT tt.PKID AS ИД,
                     tt.Name AS Наименование,
                     tt.priceIn AS Вход,
                     tt.Price AS Цена,
                     ROW_NUMBER() OVER(PARTITION BY tt.pkid ORDER BY tt.sn) AS rnumber,
                     COUNT(tt.sn) OVER(PARTITION BY tt.pkid) AS rcount
              FROM Item AS tt
              WHERE tt.iditem IS NOT NULL)
          SELECT *
          FROM CTE AS c
          WHERE c.rcount = 1;

SELECT *
FROM vw_StockRemnants
WHERE dbo.vw_StockRemnants.[ИД] = 12345;
sql
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    i-one
    2020-10-24T00:39:00Z2020-10-24T00:39:00Z

    当使用视图执行查询时(也使用内联表值函数),优化器会将视图定义替换到查询中并尝试优化生成的复合查询。

    В первом случае запрос скорее всего выполнится как

    SELECT
        tt.PKID AS ИД,
        tt.Name AS Наименование,
        tt.priceIn AS Вход,
        tt.Price AS Цена
    FROM Item AS tt
    WHERE tt.iditem IS NOT NULL AND tt.PKID = 12345;
    

    А во втором, по-видимому, как

    WITH CTE([ИД],
        [Наименование],
        [Вход],
        [Цена],
        rnumber,
        rcount)
    AS (SELECT tt.PKID AS ИД,
            tt.Name AS Наименование,
            tt.priceIn AS Вход,
            tt.Price AS Цена,
            ROW_NUMBER() OVER(PARTITION BY tt.pkid ORDER BY tt.sn) AS rnumber,
            COUNT(tt.sn) OVER(PARTITION BY tt.pkid) AS rcount
        FROM Item AS tt
        WHERE tt.iditem IS NOT NULL AND tt.PKID = 12345)
    SELECT *
    FROM CTE AS c
    WHERE c.rcount = 1;
    

    т.к. во внутреннем запросе нумерация и подсчёт количества происходят в группах, разбитых по tt.PKID, а предикат tt.PKID = 12345, так же как и c.rcount = 1, делает фильтрацию группы.


    При построении плана запроса оптимизатор старается применять предикаты настолько рано, насколько это возможно (т.н. predicate pushdown или filter pushdown), т.к., во-первых, выгодно на ранних этапах плана отсечь ненужные данные (меньше работы на следующих этапах), и, во-вторых, раннее применение предиката положительно сказывается на возможности использования индексов.

    Рассмотрим пример. Пусть есть таблица

    DECLARE @item TABLE(A int, B int);
    INSERT INTO @item VALUES (1, 2), (2, NULL);
    

    Выполним запрос с подзапросом

    SELECT *
    FROM (
        SELECT *
        FROM @item
        WHERE B IS NOT NULL
        ) I
    WHERE A = 1;
    

    и посмотрим план выполнения (графический и текстовый)

    plan 1

    |--Table Scan(OBJECT:(@item), WHERE:([A]=(1) AND [B] IS NOT NULL))
    

    Как видим, оптимизатор объединил оба условия и поместил их в узел Table Scan (чтение данных таблицы).

    Если бы запрос исполнялся именно так как он записан, то мы бы увидели

    plan 2

    |--Filter(WHERE:([A]=(1))
      |--Table Scan(OBJECT:(@item), WHERE:([B] IS NOT NULL))
    

    т.е. чтение таблицы с условием [B] IS NOT NULL как первый этап плана, и фильтрацию набора строк с условием [A]=(1) как второй его этап. В данном случае вряд ли это заметно понизило бы производительность. Если бы, однако, на столбце [A] у нас был индекс, то такой план не позволил бы его использовать оптимально (или даже вовсе).

    Однако для применения predicate pushdown у оптимизатора должно быть достаточно оснований. Препятствовать этому виду оптимизации может использование в предикатах недетерминированных выражений (в т.ч. с участием пользовательских скалярных функций). Также ни один вид оптимизации не должен нарушать семантику запроса (корректность результата превыше скорости).

    Смоделируем второй случай. Пусть есть таблица с данными

    DECLARE @item TABLE (A int, B int, C char(1));
    INSERT INTO @item VALUES
        (1, NULL, NULL), (1, 1, 'X'),
        (2, 1, NULL),
        (3, NULL, NULL), (3, 1, NULL), (3, 1, 'Y'),
        (4, NULL, NULL), (4, 1, 'Z');
    

    Мы делаем запрос

    WITH R1 AS (
        SELECT
            A, B, C,
            Num = ROW_NUMBER() OVER(PARTITION BY A ORDER BY C),
            Cnt = COUNT(C) OVER(PARTITION BY A)
        FROM @item
        WHERE B IS NOT NULL
    ),
    R2 AS (
        SELECT *
        FROM R1
        WHERE Cnt = 1
    )
    SELECT *
    FROM R2
    WHERE A = 1;
    

    семантика которого такова:

    1. Выбираем данные, делаем нумерацию и подсчёт в группах по A
    2. Берём только группы с COUNT(C) = 1
    3. Оставляем только записи с A = 1

    План выполнения запроса

    plan 3

    |--Filter(WHERE:([Expr1004]=(1)))
      |--Nested Loops(Inner Join)
        |--Table Spool
          |  |--Segment
          |    |--Sequence Project(DEFINE:([Expr1003]=row_number))
          |      |--Segment
          |        |--Sort(ORDER BY:([C] ASC))
          |          |--Table Scan(OBJECT:(@item), WHERE:([A]=(1) AND [B] IS NOT NULL))
          |--Nested Loops(Inner Join, WHERE:((1)))
            |--Compute Scalar(DEFINE:([Expr1004]=CONVERT_IMPLICIT(int,[Expr1007],0)))
            |  |--Stream Aggregate(DEFINE:([Expr1007]=COUNT([C])))
            |    |--Table Spool
            |--Table Spool
    

    показывает, однако, что условие A = 1, заданное на последнем этапе, оптимизатор перенёс на первый этап, объединив его с [B] IS NOT NULL в узле Table Scan.

    Если изменить запрос, убрав PARTITION BY, например

    WITH R1 AS (
        SELECT
            A, B, C,
            Num = ROW_NUMBER() OVER(ORDER BY C),
            Cnt = COUNT(C) OVER()
        FROM @item
        WHERE B IS NOT NULL
    ),
    R2 AS (
        SELECT *
        FROM R1
        WHERE Cnt = 1
    )
    SELECT *
    FROM R2
    WHERE A = 1;
    

    то это меняет семантику запроса так, что оптимизатор уже не может применить условие [A] = 1 при чтении данных из таблицы, и в плане запроса

    |--Filter(WHERE:([A]=(1) AND [Expr1004]=(1)))
      |--Nested Loops(Inner Join)
        |--Table Spool
        |  |--Segment
        |    |--Sequence Project(DEFINE:([Expr1003]=row_number))
        |      |--Segment
        |        |--Sort(ORDER BY:([C] ASC))
        |          |--Table Scan(OBJECT:(@item), WHERE:([B] IS NOT NULL))
        |--Nested Loops(Inner Join, WHERE:((1)))
          |--Compute Scalar(DEFINE:([Expr1004]=CONVERT_IMPLICIT(int,[Expr1007],0)))
          |  |--Stream Aggregate(DEFINE:([Expr1007]=COUNT([C])))
          |    |--Table Spool
          |--Table Spool
    

    мы видим в свойствах Table Scan лишь условие [B] IS NOT NULL, а условие [A]=(1) применяется в Filter на самом последнем этапе.

    • 3

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    Python 3.6 - 安装 MySQL (Windows)

    • 1 个回答
  • Marko Smith

    C++ 编写程序“计算单个岛屿”。填充一个二维数组 12x12 0 和 1

    • 2 个回答
  • Marko Smith

    返回指针的函数

    • 1 个回答
  • Marko Smith

    我使用 django 管理面板添加图像,但它没有显示

    • 1 个回答
  • Marko Smith

    这些条目是什么意思,它们的完整等效项是什么样的

    • 2 个回答
  • Marko Smith

    浏览器仍然缓存文件数据

    • 1 个回答
  • Marko Smith

    在 Excel VBA 中激活工作表的问题

    • 3 个回答
  • Marko Smith

    为什么内置类型中包含复数而小数不包含?

    • 2 个回答
  • Marko Smith

    获得唯一途径

    • 3 个回答
  • Marko Smith

    告诉我一个像幻灯片一样创建滚动的库

    • 1 个回答
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Алексей Шиманский 如何以及通过什么方式来查找 Javascript 代码中的错误? 2020-08-03 00:21:37 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5