SQLServer 参数化查询经验分享

本篇文章将介绍%ignore_a_1%。我将讨论如果一个查询可以被参数化,那么SQL Server优化器怎样尝试将其参数化,以及你可以怎样建立你自己的参数化查询。

什么是参数化查询?
  一个简单理解参数化查询的方式是把它看做只是一个T-SQL查询,它接受控制这个查询返回什么的参数。通过使用不同的参数,一个参数化查询返回不同的结果。要获得一个参数化查询,你需要以一种特定的方式来编写你的代码,或它需要满足一组特定的标准。

  有两种不同的方式来创建参数化查询。第一个方式是让查询优化器自动地参数化你的查询。另一个方式是通过以一个特定方式来编写你的T-SQL代码,并将它传递给sp_executesql系统存储过程,从而编程一个参数化查询。这篇文章的后面部分将介绍这个方法。

  参数化查询的关键是查询优化器将创建一个可以重用的缓存计划。通过自动地或编程使用参数化查询,SQL Server可以优化类似T-SQL语句的处理。这个优化消除了对使用高贵资源为这些类似T-SQL语句的每一次执行创建一个缓存计划的需求。而且通过创建一个可重用计划,SQL Server还减少了存放过程缓存中类似的执行计划所需的内存使用。

  现在让我们看看使得SQL Server创建参数化查询的不同方式。

  参数化查询是怎样自动创建的?

  微软编写查询优化器代码的人竭尽全力地优化SQL Server处理你的T-SQL命令的方式。我想这是查询优化器名称的由来。这些尽量减少资源和最大限度地提高查询优化器执行性能的方法之一是查看一个T-SQL语句并确定它们是否可以被参数化。要了解这是如何工作的,让我们看看下面的T-SQL语句:

在这里,你可以看到这个命令有两个特点。第一它简单,第二它在WHERE谓词中包含一个用于SalesOrderID值的指定值。查询优化器可以识别这个查询比较简单以及SalesOrderID有一个参数(“56000”)。因此,查询优化器可以自动地参数化这个查询。

  如果你使用下面的SELECT语句来查看一个只包含用于上面语句的缓存计划的、干净的缓冲池,那么你会看到查询优化器将T-SQL查询重写为一个参数化T-SQL语句:

当我在一个SQL Server 2008实例上运行这个命令时,我得到下面的输出,(注意,输出被重新格式化了,以便它更易读):

  cnt size plan_text

  — ——- ————————————————————–

  1 49152 (@1 int)SELECT * FROM [AdventureWorks].[Sales].[SalesOrderHeader]

  WHERE [SalesOrderID]=@1

  如果你看看上面输出中的plan_text字段,你会看到它不像原来的T-SQL文本。如前所述,查询优化器将这个查询重新编写为一个参数化T-SQL语句。在这里,你可以看到它现在有一个数据类型为(int)的变量(@1),它在之前的SELECT语句中被定义的。另外在plan_text的末尾, 值“56000”被替换为变量@1。既然这个T-SQL语句被重写了,而且被存储为一个缓存计划,那么如果未来一个T-SQL命令和它大致相同,只有SalesOrderID字段被赋的值不同的话,它就可以被用于重用。让我们在动作中看看它。

网龙b2b仿阿里巴巴电子商务平台 网龙b2b仿阿里巴巴电子商务平台

本系统经过多次升级改造,系统内核经过多次优化组合,已经具备相对比较方便快捷的个性化定制的特性,用户部署完毕以后,按照自己的运营要求,可实现快速定制会费管理,支持在线缴费和退费功能财富中心,管理会员的诚信度数据单客户多用户登录管理全部信息支持审批和排名不同的会员级别有不同的信息发布权限企业站单独生成,企业自主决定更新企业站信息留言、询价、报价统一管理,分系统查看分类信息参数化管理,支持多样分类信息,

网龙b2b仿阿里巴巴电子商务平台 0 查看详情 网龙b2b仿阿里巴巴电子商务平台 如果我在我的机器上运行下面的命令:

在这里,我首先释放过程缓存,然后我执行两个不同、但却类似的非参数化查询来看看查询优化器是会创建两个不同的缓存计划还是创建用于这两个查询的一个缓存计划。在这里,你可以看到查询优化器事实上很聪明,它参数化第一个查询并缓存了计划。然后当第二个类似、但有一个不同的SalesOrderID值的查询发送到SQL Server时,优化器可以识别已经缓存了一个计划,然后重用它来处理第二个查询。你可以这么说是因为“cnt”字段现在表明这个计划被用了两次。

  数据库配置选项PARAMETERIZATION可以影响T-SQL语句怎样被自动地参数化。对于这个选项有两种不同的设置,SIMPLE和FORCED。当PARAMETERIZATION设置被设置为SIMPLE时,只有简单的T-SQL语句才会被参数化。要介绍这个,看下下面的命令:

这个查询类似于我前面的示例,除了在这里我添加了一个额外的JOIN标准。当数据库AdventureWorks的PARAMETERIZATION选项被设置为SIMPLE时,这个查询不会被自动地参数化。SIMPLE PARAMETERIZATION设置告诉查询优化器只参数化简单的查询。但是当选项PARAMETERIZATION被设置为FORCED时,这个查询将被自动地参数化。

  当你设置数据库选项为使用FORCE PARAMETERIZATION时,查询优化器试图参数化所有的查询,而不仅仅是简单的查询。你可能会认为这很好。但是在某些情况下,当数据库设置PARAMETERIZATION为FORCED时,查询优化器将选择不是很理想的查询计划。当数据库设置PARAMETER为FORCED时,它改变查询中的字面常量。这可能导致当查询中涉及计算字段时索引和索引视图不被选中参与到执行计划中,从而导致一个无效的计划。FORCED PARAMETERIZATION选项可能是改进具有大量类似的、传递过来的参数稍有不同的查询的数据库性能的一个很好的解决方案。一个在线销售应用程序,它的客户对你的产品执行大量的类似搜索, 产品值不同,这可能是一个能够受益于FORCED PARAMETERIZATION的很好的应用程序类型。

不是所有的查询从句都会被参数化。例如查询的TOP、TABLESAMPLE、 HAVING、GROUP BY、ORDER BY、OUTPUT…INTO或FOR XML从句不会被参数化。

  使用sp_execute_sql来参数化你的T-SQL

  你不需要依赖于数据库的PARAMETERIZATION选项来使得查询优化器参数化一个查询。你可以参数化你自己的查询。你通过重新编写你的T-SQL语句并使用“sp_executesql”系统存储过程执行重写的语句来实现。正如已经看到的,上面包括一个“JOIN”从句的SELECT语句在数据库的PARAMETERIZATION设置为SIMPLE时没有被自动参数化。让我重新编写这个查询以便查询优化器将创建一个可重用的参数化查询执行计划。

  为了说明,让我们看两个类似的、不会被自动参数化的T-SQL语句,并创建两个不同的缓存执行计划。然后我将重新编写这两个查询使得它们都使用相同的缓存参数化执行计划。

  让我们看看这个代码:

在这里,我释放了过程缓存,然后运行这两个包含一个JOIN的、不同的非简单的T-SQL语句。然后我将检查缓存计划。这是这个使用DMV 的SELECT语句的输出(注意,输出被重新格式化了,以便它更易读): 正如你从这个输出看到的,这两个SELECT语句没有被查询优化器参数化。优化器创建了两个不同的缓存执行计划,每一个都只被执行了一次。我们可以通过使用sp_executesql系统存储过程来帮助优化器为这两个不同的SELECT语句创建一个参数化执行计划。
下面是上面的代码被重新编写来使用sp_executesql 系统存储过程: 如同你所看到的,我重新编写了这两个SELECT语句,使它们通过使用“EXEC sp_executesql”语句来执行。对这些EXEC语句中的每一个,我都传递三个不同的参数。第一个参数是基本的SELECT语句,但是我将SalesOrderID的值用一个变量(@SalesOrderID)替代。在第二个参数中,我确定了@SalesOrderID的数据类型,在这个例子中它是一个integer。然后在最后一个参数中,我传递了SalesOrderID的值。这个参数将控制我的SELECT根据SalesOrderID值所生成的结果。sp_executesql的每次执行中前两个参数都是一样的。但是第三个参数不同,因为每个都有不同的SalesOrderID值。

  现在当我运行上面的代码时,我从DMV SELECT语句得到下面的输出(注意,输出被重新格式化了,以便它更易读):

从这个输出,你可以看出,我有一个参数化缓存计划,它被执行了两次,为每个EXEC语句各执行了一次。

  使用参数化查询来节省资源和优化性能

  在语句可以被执行之前,每个T-SQL语句都需要被评估,而且需要建立一个执行计划。创建执行计划会占用宝贵的CPU资源。当执行计划被创建后,它使用内存空间将它存储在过程缓存中。降低CPU和内存使用的一个方法是利用参数化查询。尽管数据库可以被设置为对所有查询FORCE参数化,但是这不总是最好的选择。通过了解你的哪些T-SQL语句可以被参数化然后使用sp_executesql存储过程,你可以帮助SQL Server节省资源并优化你的查询的性能。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/814551.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
中端显卡诚意之作?NVIDIA RTX 4060 Ti首发评测
上一篇 2025年11月27日 00:26:17
财务管理系统的优势与价值全面解析
下一篇 2025年11月27日 00:26:20

相关推荐

  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • 三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布

    三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布三星不再独享,消息称搭载骁龙 8 Gen 3 领先版处理器新机即将发布

    6 月 15 日消息,据博主@肥威 今日爆料,搭载骁龙 8 Gen 3 领先版%ign%ignore_a_1%re_a_1%的新机即将发布,把之前的 for Galaxy 改成“for Everybody”。 Pic Copilot AI时代的顶级电商设计师,轻松打造爆款产品图片 158 查看详情 …

    2026年5月10日 用户投稿
    000
  • 高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行

    高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行高通预热 2023 骁龙峰会:以AI为主题,10 月 25-26 日举行

    【环球网科技综合报道】10月17日消息,高通今日对 2023 骁龙峰会进行了预热,本次大会将以 %ign%ignore_a_1%re_a_1% 为主题,届时骁龙 8 gen 3 处理器也很大可能在本届峰会亮相。 在临近活动召开之日,相关业内人士也透露了高通骁龙8Gen3跑分及规格。据悉,高通骁龙8 …

    2026年5月10日 用户投稿
    000
  • JavaScript中的标签模板字面量(Tagged Templates)有哪些高级用法?

    标签模板通过自定义函数实现复杂逻辑,如html函数转义防止XSS,css函数生成唯一类名封装样式,结合哈希值隔离组件样式,确保安全与模块化。 标签模板字面量不只是字符串拼接工具,它能结合函数实现更复杂的逻辑处理。通过自定义标签函数,你可以解析模板中的表达式和静态部分,从而实现如国际化、样式封装、安全…

    2026年5月10日
    000
  • Go语言集成SQLite3数据库:使用go-sqlite3库的实践指南

    本文旨在为Go语言开发者提供一套完整的SQLite3数据库集成指南。我们将重点介绍如何使用广受欢迎的github.com/mattn/go-sqlite3库,涵盖其安装、数据库连接、表创建、数据插入、查询、更新及删除等核心操作,并提供实用的代码示例和注意事项,助您高效地在Go应用中实现SQLite3…

    2026年5月10日
    000
  • 一加手机因5G专利纠纷在德国再次被停售

    一加手机因5G专利纠纷在德国再次被停售一加手机因5G专利纠纷在德国再次被停售一加手机因5G专利纠纷在德国再次被停售一加手机因5G专利纠纷在德国再次被停售

    it之家 10 月 2 日消息,科技媒体 android headline 昨日(10 月 1 日)发布博文,报道称一加(oneplus)由于和 interdigital 之间的 5g 技术专利纠纷,其品牌手机再次在德国停售。it之家曾于今年 1 月报道,oppo 和诺基亚达成协议,一加手机重返德国…

    2026年5月10日 用户投稿
    000
  • php数据整理怎么按日期字段分组汇总_php按日期分组统计与时间段合并技巧

    可使用SQL或PHP对数据按日期分组汇总。1、通过MySQL的DATE()、YEAR()、MONTH()函数在查询时按日、月、年分组统计;2、在PHP中遍历数组,以date(‘Y-m-d’)等格式化日期作为键进行归类;3、按周可使用date(‘o-W’…

    2026年5月10日
    000
  • Shiny 应用中实现可滚动 Sortable 列表的实践指南

    本文详细介绍了如何在 Shiny 应用中创建具有滚动功能的 sortable 列表。通过应用 CSS 样式 max-height 和 overflow-y: auto 到 rank_list 容器,用户可以有效管理内容过多的列表,确保界面整洁且用户体验良好。教程将提供完整的代码示例和详细解释,帮助开…

    2026年5月10日
    000
  • 使用MySQL和PHP高效获取最热门数据条目:统计与排序实践

    本教程详细阐述如何利用mysql的聚合函数和php的mysqli扩展,高效地从数据库中查询并排序出最常出现的数据条目。文章将通过一个具体的案例,指导读者构建正确的sql查询,并结合php进行数据处理和调试,避免常见的sql语法错误和php运行时问题,从而准确获取按频率降序排列的热门数据。 在Web开…

    2026年5月10日
    000
  • Golang反射与标签解析结合使用实例

    Golang反射结合结构体标签的核心优势在于提供运行时动态解析和操作结构体元数据的能力,实现高度灵活、解耦的系统设计。通过reflect.TypeOf(obj).Field(i).Tag.Get(“tag_name”)模式,可在不修改结构体的前提下集中管理JSON序列化、数据…

    2026年5月10日
    300
  • 优化字符串查找:内存映射 vs. 数据库查询

    在Go服务器应用开发中,经常会遇到需要对接收到的字符串进行验证的场景,例如验证字符串是否存在于数据库中。针对高并发的HTTP请求,如何高效地进行字符串查找是一个关键问题。通常有两种策略:一是每次请求都执行SQL查询;二是将所有字符串预先加载到内存中的Map,然后通过Map进行快速查找。选择哪种策略取…

    2026年5月10日
    000
  • 如何用C#实现数据库的跨平台迁移?使用EF Core工具?

    使用EF Core实现跨平台数据库迁移,需定义实体与DbContext,通过动态配置不同数据库提供程序,利用EF Core CLI生成并应用迁移,结合Fluent API处理数据库差异,确保结构与数据兼容。 要实现数据库的跨平台迁移,C# 中最常用且高效的方式是使用 Entity Framework…

    2026年5月10日
    000
  • CRM的定制开发需要注意什么?3个开发必知事项

    CRM的定制开发需要注意以下几点:明确业务需求、选择合适的技术架构、确保数据安全。 其中,明确业务需求尤为重要。准确了解企业的业务流程、客户管理需求和目标是定制开发CRM系统的第一步。只有在深入分析和理解业务需求的基础上,才能开发出真正符合企业特定需求的CRM系统,从而提高工作效率和客户满意度。 一…

    2026年5月10日
    000
  • 全局数据库连接变量会影响性能吗?

    全局数据库连接变量:性能考量 项目中使用全局数据库连接变量是否会影响性能?答案取决于多种因素。让我们深入探讨: Java与Go数据库连接池的对比 Java使用数据源管理数据库连接池,可配置最大空闲连接数(maxIdle)和最大活跃连接数(maxActive)。Go的连接池设置类似。Java项目通常共…

    2026年5月10日
    000
  • 加密货币期权交易入门:比合约更灵活的风险对冲工具

    %ignore_a_1%期权通过权利与义务分离,提供精细化风险管理。首先选择BTC或ETH等标的资产,根据市场预期买入看涨或看跌期权,并选定到期日、行权价与权利金完成交易。作为卖方,可在高波动率时卖出虚值期权获取权利金,需评估隐含波动率、设置安全边际、准备保证金并监控持仓以控制风险。投资者还可构建组…

    2026年5月10日
    000
  • ASP.NET Core 中的健康检查 UI 如何配置?

    首先安装HealthChecks.UI和UI.InMemory.Storage包,然后在Program.cs中添加健康检查服务并配置数据库、Redis等检查项,接着注册健康检查UI服务并设置评估时间与存储方式,最后启用健康检查中间件和UI路由,启动后通过/health-ui访问可视化界面。 在 AS…

    2026年5月10日
    000
  • 如何设置php网站内容关联推荐_相关内容自动推荐配置方法

    基于标签匹配、关键词提取、分类体系、用户行为协同过滤及外部推荐引擎接口五种方法,可实现PHP网站的内容关联推荐功能。一、通过文章标签查找相似标签内容并按匹配数量排序,返回最多5条推荐;二、利用分词技术提取标题和正文关键词,计算与其他文章的关键词重合率,按阈值筛选高相关性内容;三、依据文章所属分类,在…

    2026年5月10日
    000
  • 怎么用php搜索_PHP站内搜索功能实现与优化方法教程

    1、通过PHP%ignore_a_1%关键词并用LIKE模糊查询实现基础搜索;2、使用预处理语句防止SQL注入,提升安全性;3、拆分关键词并多字段匹配以提高准确性;4、添加FULLTEXT全文索引优化大数量下的查询性能;5、利用Redis等缓存常见结果减少数据库压力。 如果您希望在自己的网站中实现搜…

    2026年5月10日
    100
  • 如何在Golang中处理数据库事务错误

    答案:在Golang中处理数据库事务需确保每个Begin都有对应的Commit或Rollback。使用db.Begin()开启事务后,应通过defer注册回滚逻辑,即使出错也能自动清理;成功则手动调用tx.Commit(),之后Rollback无效。注意区分错误类型:sql.ErrTxDone表示事…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信