- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有以下表结构:
AuditUserMethods
:
+---------------+---------------+----------+
| ColumnName | DataType | Nullable |
+---------------+---------------+----------+
| Id | INT | NOT NULL |
| CreatedDate | DATETIME | NOT NULL |
| ApiMethodName | NVARCHAR(MAX) | NOT NULL |
| Request | NVARCHAR(MAX) | NOT NULL |
| Result | NVARCHAR(MAX) | NOT NULL |
| Method_Id | INT | NOT NULL |
| User_Id | INT | NULL |
+---------------+---------------+----------+
AuditUserMethodErrorCodes
:
+--------------------+----------+----------+
| ColumnName | DataType | Nullable |
+--------------------+----------+----------+
| Id | INT | NOT NULL |
| AuditUserMethod_Id | INT | NOT NULL |
| ErrorCode | INT | NOT NULL |
+--------------------+----------+----------+
ID
是两个表中的 PK。存在一对多的关系。一个 AuditUserMethod
可以有多个 AuditUserMethodErrorCodes
。因此 FK AuditUserMethod_Id
。
AuditUserMethods
表中的 AuditUserMethod_Id
和 CreatedDate
都有两个非聚集索引。
该过程的目的是返回基于过滤器的分页结果集。 @PageSize
确定返回多少行,@PageIndex
确定返回哪个页面。所有其他变量用于过滤。
返回三个结果集。
AuditUserMethods
详细信息AuditUserMethodErrorCodes
详细信息存储过程:
CREATE PROCEDURE [api].[Audit_V1_GetAuditDetails]
(
@Users XML = NULL,
@Methods XML = NULL,
@ErrorCodes XML = NULL,
@FromDate DATETIME = NULL,
@ToDate DATETIME = NULL,
@PageSize INT = 5,
@PageIndex INT = 0
)
AS
BEGIN
DECLARE @UserIds TABLE (Id INT)
DECLARE @MethodNames TABLE (Name NVARCHAR(256))
DECLARE @ErrorCodeIds TABLE (Id INT)
DECLARE @FilterUsers BIT = 0
DECLARE @FilterMethods BIT = 0
DECLARE @FilterErrorCodes BIT = 0
INSERT @UserIds
SELECT
x.y.value('.', 'int')
FROM
@Users.nodes('Ids/x/@i') AS x (y)
INSERT @MethodNames
SELECT
x.y.value('.', 'NVARCHAR(256)')
FROM
@Methods.nodes('ArrayOfString/string') AS x (y)
INSERT @ErrorCodeIds
SELECT
x.y.value('.', 'int')
FROM
@ErrorCodes.nodes('Ids/x/@i') AS x (y)
IF EXISTS (SELECT TOP 1 0 FROM @UserIds)
SET @FilterUsers = 1
IF EXISTS (SELECT TOP 1 0 FROM @MethodNames)
SET @FilterMethods = 1
IF EXISTS (SELECT TOP 1 0 FROM @ErrorCodeIds)
SET @FilterErrorCodes = 1
DECLARE @StartRow INT = @PageIndex * @Pagesize
DECLARE @PageDataResults TABLE (Id INT,
CreatedDate DATETIME,
ApiMethodName NVARCHAR(256),
Request NVARCHAR(MAX),
Result NVARCHAR(MAX),
MethodId INT,
UserId INT,
TotalRows INT);
WITH PageData AS
(
SELECT
id AS id
, createddate AS createddate
, apimethodname AS apimethodname
, request AS request
, result AS result
, method_id AS method_id
, user_id AS user_id
, ROW_NUMBER() OVER (ORDER BY createddate DESC, id DESC) AS row_number
, COUNT(*) OVER() as TotalRows
FROM
dbo.AuditUserMethods AS aum
WHERE
(@FromDate IS NULL OR
(@FromDate IS NOT NULL AND aum.createddate > @FromDate))
AND (@ToDate IS NULL OR
(@ToDate IS NOT NULL AND aum.createddate < @ToDate))
AND (@FilterUsers = 0 OR
(@FilterUsers = 1 AND aum.user_id IN (SELECT Id FROM @UserIds)))
AND (@FilterMethods = 0 OR
(@FilterMethods = 1 AND aum.ApiMethodName IN (SELECT Name FROM @MethodNames)))
AND (@FiltererRorCodes = 0 OR
(@FiltererRorCodes = 1
AND EXISTS (SELECT 1
FROM AuditUserMethodErrorCodes e
WHERE e.AuditUserMethod_Id = aum.Id
AND e.ErrorCode IN (SELECT Id FROM @ErrorCodeIds)
)
)
)
)
INSERT @PageDataResults
SELECT TOP (@Pagesize)
PageData.id AS id
, PageData.createddate AS createddate
, PageData.apimethodname AS apimethodname
, PageData.request AS request
, PageData.result AS result
, PageData.method_id AS method_id
, PageData.user_id AS user_id
, PageData.TotalRows AS totalrows
FROM
PageData
WHERE
PageData.row_number > @StartRow
ORDER BY
PageData.createddate DESC
SELECT
Id, CreatedDate, ApiMethodName, Request, Result, MethodId, UserId
FROM
@PageDataResults
SELECT
aumec.AuditUserMethod_Id, aumec.ErrorCode
FROM
@PageDataResults ps
INNER JOIN
AuditUserMethodErrorCodes aumec ON ps.Id = aumec.AuditUserMethod_Id
SELECT TOP 1
TotalRowsNumberOfReturnedAuditEntries
FROM @PageDataResults
END
AuditUserMethods
表包含 500,000 行,AuditUserMethodErrorCodes
包含 67843 行。
我正在使用以下参数执行该过程:
EXEC [api].[Audit_V1_GetAuditDetails] @Users = N'<Ids><x i="1" /></Ids>'
,@Methods = NULL
,@ErrorCodes = N'<Ids />'
,@FromDate = '2015-02-15 07:18:59.613'
,@ToDate = '2015-07-02 08:18:59.613'
,@Pagesize = 5000
,@PageIndex = 0
存储过程仅用了 2 秒多一点的时间来执行并返回 5000 行。我需要这个存储过程运行得更快,但我不确定如何改进它。
以实际执行计划为准。相对于该批处理,CTE 占了 99%。在 CTE 中,排序占了 95% 的成本:
最佳答案
我将从声明几个表参数类型开始。
CREATE TYPE [api].[IdSet] AS TABLE
(
[Id] INT NOT NULL
);
和,
CREATE TYPE [api].[StringSet] AS TABLE
(
[Value] NVARCHAR(256) NOT NULL
);
然后我会更改存储过程的签名以使用它们。
注意我还会将总计数作为输出参数而不是单独的结果集返回。
CREATE PROCEDURE [api].[Audit_V2_GetAuditDetails]
(
@userIds [api].[IdSet] READONLY,
@methodNames [api].[StringSet] READONLY,
@errorCodeIds [api].[IdSet] READONLY,
@fromDate DATETIME = NULL,
@toDate DATETIME = NULL,
@pageSize INT = 5,
@pageIndex INT = 0,
@totalCount BIGINT OUTPUT
)
我知道您可能仍需要进行 XML 提取,但如果您在 SP 之外进行提取将有助于查询规划器。
现在,在 SP 中,我不会使用 @PageDataResults
我只会获取页面的 ID。我也不会使用 CTE,这在这种情况下没有帮助。
我将简化查询并运行一次以汇总总计数,然后如果它大于 0,则再次运行相同的查询以仅返回 ID 页面。查询的主体将由服务器内部缓存。
此外,Id' 使用 OFFSET
和 FETCH
扩展对 ORDER BY
进行分页,
我在下面概述了一些逻辑上的简化,
CREATE PROCEDURE [api].[Audit_V2_GetAuditDetails]
(
@userIds [api].[IdSet] READONLY,
@methodNames [api].[StringSet] READONLY,
@errorCodeIds [api].[IdSet] READONLY,
@fromDate DATETIME = NULL,
@toDate DATETIME = NULL,
@pageSize INT = 5,
@pageIndex INT = 0,
@totalCount BIGINT OUTPUT
)
AS
DECLARE @offset INT = @pageSize * @pageIndex;
DECLARE @filterUsers BIT = 0;
DECLARE @filterMethods BIT = 0;
DECLARE @filterErrorCodes BIT = 0;
IF EXISTS (SELECT 0 FROM @userIds)
SET @filterUsers = 1;
IF EXISTS (SELECT 0 FROM @methodNames)
SET @filterMethods = 1;
IF EXISTS (SELECT 0 FROM @errorCodeIds)
SET @filterErrorCodes = 1;
SELECT
@totalCount = COUNT_BIG(*)
FROM
[dbo].[AuditUserMethods] [aum]
LEFT JOIN
@userIds [U]
ON [U].[Id] = [aum].[user_id]
LEFT JOIN
@methodName [M]
ON [M].[Value] = [aum].[ApiMethodName]
WHERE
(
@fromDate IS NULL
OR
[aum].[createddate] > @fromDate
)
AND
(
@toDate IS NULL
OR
[aum].[createddate] < @toDate
)
AND
(
@filterUsers = 0
OR
[U].[Id] IS NOT NULL
(
AND
(
@filterMethods = 0
OR
[M].[Value] IS NOT NULL
(
AND
(
@filterErrorCodes = 0
OR
(
EXISTS(
SELECT
1
FROM
[dbo].[AuditUserMethodErrorCodes] [e]
JOIN
@errorCodeIds [ec]
ON [ec].[Id] = [e].[ErrorCode]
WHERE
[e].[AuditUserMethod_Id] = [aum].[Id])
);
DECLARE @pageIds [api].[IdSet];
IF @totalCount > 0
INSERT @pageIds
SELECT
[aum].[id]
FROM
[dbo].[AuditUserMethods] [aum]
LEFT JOIN
@userIds [U]
ON [U].[Id] = [aum].[user_id]
LEFT JOIN
@methodName [M]
ON [M].[Value] = [aum].[ApiMethodName]
WHERE
(
@fromDate IS NULL
OR
[aum].[createddate] > @fromDate
)
AND
(
@toDate IS NULL
OR
[aum].[createddate] < @toDate
)
AND
(
@filterUsers = 0
OR
[U].[Id] IS NOT NULL
(
AND
(
@filterMethods = 0
OR
[M].[Value] IS NOT NULL
(
AND
(
@filterErrorCodes = 0
OR
(
EXISTS(
SELECT
1
FROM
[dbo].[AuditUserMethodErrorCodes] [e]
JOIN
@errorCodeIds [ec]
ON [ec].[Id] = [e].[ErrorCode]
WHERE
[e].[AuditUserMethod_Id] = [aum].[Id])
)
ORDER BY
[aum].[createddate] DESC,
[aum].[id] DESC
OFFSET @offset ROWS
FETCH NEXT @pageSize ROWS ONLY;
SELECT
[aum].[Id],
[aum].[CreatedDate],
[aum].[ApiMethodName],
[aum].[Request],
[aum].[Result],
[aum].[MethodId],
[aum].[UserId]
FROM
[dbo].[AuditUserMethods] [aum]
JOIN
@pageIds [i]
ON [i].[Id] = [aum].[id]
ORDER BY
[aum].[createddate] DESC,
[aum].[id] DESC;
SELECT
[aumec].[AuditUserMethod_Id],
[aumec].[ErrorCode]
FROM
[dbo].[AuditUserMethodErrorCodes] [aumec]
JOIN
@pageIds [i]
ON [i].[Id] = [aumec].[AuditUserMethod_Id];
/* The total count is an output parameter */
RETURN 0;
如果这不足以改善情况,您将需要查看查询计划并考虑哪些索引是最佳的。
警告 所有代码都是即兴编写的,因此,虽然想法是正确的,但语法可能并不完美。
关于SQL 服务器 2014 : slow stored procedure execution time,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31201550/
有时我一直在努力理解为什么在尝试创建航路点任务时任务管理器会收到错误“无法执行执行”。我附上了我正在使用的工作流程,如果您能看一下,我将不胜感激。 1.Initialize FlightControl
我正在 Python 中使用 SQLAlchemy 核心,并且我已多次阅读文档,但仍然需要有关 engine.execute() 的说明。与 connection.execute() . 据我了解,e
在我的 Zend 框架项目中,我想检查是否设置了 cookie。如果是这种情况,我想使用 cookie 内容登录用户。 由于我必须在调用任何 Controller 之前执行此自动登录,因此我尝试将其放
我正在尝试为我创建的 2 个选择语句的 UNION 创建一个 View 。 UNION 在单独执行时工作正常 但问题是当我将它作为 View 执行时,只有 UNION 的第一部分被执行。 我正在使用的
下面我写了一个简单的例子来演示我遇到的问题。执行代码后,我得到一个 cygwin 异常 7200。我环顾四周并尝试了一些事情,但没有解决。有人可以解释为什么我得到它,我该如何解决?感谢您抽出宝贵时间,
从池中获取连接然后在连接上调用 execute 而不是直接在池对象上调用 execute 的用例是什么? 在 Pool 的文档中类,显示此示例: con = await pool.acquire()
我正在尝试通过 SQL 将变量中的 2 个值插入表中,代码完成时没有错误,但条目未显示在表中。 我尝试在即时窗口中执行代码,但这给了我一个关于括号的错误(我真的不知道如何在那里正确输入提示),所以我将
我对广播接收器有点困惑。我有一个广播接收器,它在 TIME_SET 和 TIMEZONE_CHANGED 操作时触发(代码在下面给出)。我想知道的是,当 TIME_SET 和 TIMEZONE_CHA
我必须与需要随每个请求发送访问 token 的外部服务集成。访问 token 的到期时间很短(只有几个小时)。我决定以乐观的方式使用访问 token 。我将使用当前 token 调用外部服务。如果出现
如果我在 swift 中运行以下代码,步骤 1.、2.、3. 和 4. 是否始终按此顺序执行(它们应该如此),或者如果循环存在异步执行的风险,排序等,花费的时间比预期的要长? // 1. fo
我在我的 C++ 应用程序中看到访问冲突错误。在发生违规并使用 !analyze 时将 windbg 附加到进程时,我发现访问违规是由于试图执行不可执行的地址。我知道导致此问题的正在执行的地址。什么可
在使用 Ubuntu 大约一年之后,这对我来说是第一次。 我接手了一个跟踪维修的汽车服务项目。我可以看到每个文件的完整源代码,但是有一个没有扩展名的文件,但在 Ubuntu 中,属性显示为可执行文件(
什么是 LinqPad“自动跟踪执行”和“跳转到执行点”?如何使用它们,如果你能给出一个详细的例子将不胜感激。 最佳答案 这不是一个详细的示例,但它说明了该功能。如果你有一个像 "1".Dump()
我使用 Q.js 来实现 promise 。在下面的代码中,每个方法都会进行 ajax 调用,然后返回一个 Promise。一切都按预期进行,每个方法在下一个方法开始之前执行并完成: function
我有一个类,它实现了 Runnable接口(interface),并且是一个一旦启动就会无限期运行的任务(长时间运行的线程)。 public class LongRunningTask impleme
PDOStatement::execute() [pdostatement.execute]: SQLSTATE[HY093]: 无效的参数数量:绑定(bind)变量的数量与标记数量不匹配 我收到此错
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
想要为执行的每个 linux 命令添加 aspect:executionTime 有什么方法可以添加默认方面环境,以便必须为执行的 linux 命令获取 executionTime 最佳答案 根据 m
我正在尝试安装一个名为 MFOC 的工具.我按照其网站中提到的说明进行操作,如下所示: ebrahim@ubuntu:~$ cd Desktop/mfoc-master/ ebrahim@ubuntu
我刚开始使用 numba 来提高我的程序的性能。我已经减少了我将要呈现的情况 import numba as nb import numpy as np from time import time d
我是一名优秀的程序员,十分优秀!