SELECT 赋值与ORDER BY冲突的问题

  问题描述:
  使用 SELECT 语句,轮询表中的数据,并且处理变量数据时,如果有ORDER BY语句,则得不到想要的结果,但去掉ORDER BY,结果正常。
  具体的问题表现参考下面的问题重现代码
 
  问题重现代码
  -- 测试数据
DECLARE @T TABLE(id int,value nvarchar(16))
INSERT INTO @T SELECT
1,   N'好人' UNION ALL SELECT
2,   N'坏人' UNION ALL SELECT
3,   N'吃饭' UNION ALL SELECT
4,   N'垃圾'
 
  -- 赋值处理
DECLARE @str nvarchar(4000)
SET @str = N'我不是一个好人,也不是垃圾'
SELECT @str = REPLACE(@str, value, N'<u>' + value + N'</u>')
FROM @T
WHERE CHARINDEX(value, @str) > 0
--ORDER BY CHARINDEX(value, @str) DESC
SELECT @str
 
/* -- 结果(当赋值处理语句注释掉ORDER BY 时)
我不是一个<u>好人</u>,也不是<u>垃圾</u>
-- */
 
/* -- 结果(当赋值处理语句加上ORDER BY 时)
我不是一个<u>好人</u>,也不是垃圾
-- */
 
  问题分析:
  两个处理语句的结果不同,通过查看它们的执行计划应该可以看出原因所在,为此,通过
SET SHOWPLAN_ALL ON
  输出了两种执行语句的执行计划(仅StmtText部分,有兴趣的读者在自己的电脑上测试的时候,可以去了解其他部分的信息)

StmtText

Step

DECLARE @str nvarchar(4000) SET @str =
N'我不是一个好人,也不是垃圾'

 

SELECT @str =
REPLACE(@str, value,
N'<u>'
+ value +
N'</u>') FROM @T   WHERE
CHARINDEX(value, @str)
> 0

4

 

|--

Compute Scalar(DEFINE:([Expr1002]=replace([@str], @T.[value], '<u>'+@T.[value]+'</u>')))

3

 

 

|--

Filter(WHERE:(charindex(@T.[value], [@str], NULL)>0))

2

 

 

 

|--

Table Scan(OBJECT:(@T))

1

 

 

 

 

 

 

 

DECLARE @str nvarchar(4000) SET @str =
N'我不是一个好人,也不是垃圾'

 

SELECT @str =
REPLACE(@str, value,
N'<u>'
+ value +
N'</u>') FROM @T   WHERE
CHARINDEX(value, @str)
> 0   ORDER
BY
CHARINDEX(value, @str)
DESC

5

 

|--

Sort(ORDER BY:([Expr1003] DESC))

4

 

 

|--

Compute Scalar(DEFINE:([Expr1002]=replace([@str], @T.[value], '<u>'+@T.[value]+'</u>'), [Expr1003]=charindex(@T.[value], [@str], NULL)))

3

 

 

 

|--

Filter(WHERE:(charindex(@T.[value], [@str], NULL)>0))

2

 

 

 

 

|--

Table Scan(OBJECT:(@T))

1

  从上面的列表可以看出,两种处理的最大差异,在于赋值前,是否有ORDER BY 子句,从一般的理解上,可能会认为是否排序并不重要,但换个角度来看问题,就比较容易理解为什么有ORDER BY子句后得不到我们想要的结果了:
当有ORDER BY子句时,对于SELECT @str = 这种赋值处理,SQL Server认为赋值处理肯定只会保留最后一条记录的处理结果,而ORDER BY子句确定了数据顺序,也就知道最后一条记录是那个,因此只会处理ORDER BY的最后一条记录。(读者可以自行去测试一下,调整ORDER BY顺序,看看结果是否与我的推论相符)
当没有ORDER BY子句时,因为无法确定数据顺序,所以SQL Server必须扫描满足条件的每条数据来得到结果,这样每扫描一条记录都会处理一次,所以结果是我们所预知的
 
  问题解决方法:
  修改处理语句,使查询优化器使用与我们需要结果一致的执行方法,可以解决这个问题。
对于示例中的处理语句,可以调整如下:
DECLARE @str nvarchar(4000)
SET @str = N'我不是一个好人,也不是垃圾'
SELECT @str = REPLACE(@str, value, N'<u>' + value + N'</u>')
FROM(
    SELECT TOP 100 PERCENT
        value
    FROM @T
    WHERE CHARINDEX(value, @str) > 0
    ORDER BY CHARINDEX(value, @str) DESC
)A
SELECT @str
 
  补充:
  此问题的结论只是笔者对于查询分析的一个推论,并无相应的官方文档可以证明,所以欢迎大家发表自己的看法

时间: 2024-12-31 19:22:30

SELECT 赋值与ORDER BY冲突的问题的相关文章

SQL Server中使用set和select赋值时一些比较区别

SQL Server推荐使用 SET 而不是 SELECT 对变量进行赋值. 当表达式返回一个值并对一个变量进行赋值时,推荐使用 SET 方法. 下表列出 SET 与 SELECT 的区别.请特别注意红色部分.  代码如下 复制代码 /****************** 场景布置 ******************/ DROP TABLE # CREATE TABLE #(OrderId INT) INSERT #(OrderId) VALUES(1) INSERT #(OrderId) V

select-存储过程中SELECT赋值报错什么原因?

问题描述 存储过程中SELECT赋值报错什么原因? CREATE OR REPLACE PROCEDURE PROC_DICTABLE_TBYSFL(p_ANetUser VARCHAR)ISvc_DICTABLEID varchar2(36);vc_DICTABLECOLID varchar2(36);vc_DICCOL1 VARCHAR2(20);vc_DICCOL2 VARCHAR2(20);vc_DICCOL3 VARCHAR2(20);vc_DICCOL4 VARCHAR2(20);v

MySQL中select语句使用order按行排序_Mysql

本文介绍MySQL数据库中执行select查询语句,并对查询的结果使用order by 子句进行排序. 再来回顾一下SQL语句中的select语句的语法: Select 语句的基本语法: Select <列的集合> from <表名> where <条件> order by <排序字段和方式> 如果要对查询结果按某个字段排序,则要使用order by 子句,如下: select * from <表名> order by <字段名称>

jquery 怎么给select赋值

问题描述 后台传了个参数过来${name},如何把这个值赋给select文本框里,只要显示在里面就行:用select的id这样写不行 $('#id').val(${id}); 解决方案 $('#id').text(${id})或者 $('#id').html(${id})解决方案二:再javascript脚本中使用${}需要加引号,比如$('#id').val('${id}')解决方案三:给楼主个代码示例(测试通过):<select id="single"> <opt

基于SQL中SET与SELECT赋值的区别详解_Mysql

最近的项目写的SQL比较多,经常会用到对变量赋值,而我使用SET和SELECT都会达到效果.那就有些迷惑,这两者有什么区别呢?什么时候哪该哪个呢?经过网上的查询,及个人练习,总结两者有以下几点主要区别:假定有设定变量: 复制代码 代码如下: DECLARE @VAR1 VARCHAR(1) DECLARE @VAR2 VARCHAR(2) 1.SELECT可以在一条语句里对多个变量同时赋值,而SET只能一次对一个变量赋值,如下: 复制代码 代码如下: SELECT @VAR1='Y',@VAR2

jquery 多个 select 赋值后 点击一个按钮跳转到赋值的页面上去

问题描述 如题 应该怎么写这个jquery如图:这里一共有四个select 应该如何写这个JQ 然后能在点击后搜索 进入到赋值后的页面里去? 解决方案 我不太理解你的问题是想问什么东西.1.是后台取不到前面select的值吗? 那你select缺少name.2.如果是后台查询结果之后,跳转页面上的下拉列表值不是之前选中的. 你得把后台查询条目带到前台界面,然后进行赋值,$("#uid_a").val(值);解决方案二:<div id="abc"> <

sql server存储过程中SELECT 与 SET 对变量赋值的区别

SQL Server 中对已经定义的变量赋值的方式用两种,分别是 SET 和 SELECT. 对于这两种方式的区别,SQL Server 联机丛书中已经有详细的说明,但很多时候我们 并没有注意,其实这两种方式还是有很多差别的. SQL Server推荐使用 SET 而不是 SELECT 对变量进行赋值. 当表达式返回一个值并对一个变量进行赋值时,推荐使用 SET 方法. 下表列出 SET 与 SELECT 的区别.请特别注意红色部分. set select 同时对多个变量同时赋值 不支持 支持

sql-SQL Select 执行顺序 Select Order by 谁先执行

问题描述 SQL Select 执行顺序 Select Order by 谁先执行 SQL Select 语句 Select 和 Order by执行顺序是怎样的? 网上查了些,都说先执行Select 那为什么 Order by 可以对非Select 字段进行排序???? 栗子:select id,name from tb_student order by age 解决方案 select 先执行没错,有结果了再来排序,个人觉得可以这样想,select 查询出来后,在显示id和name之前,先对a

js解析json数据并动态赋值到select列表

jSON(JavaScriptObject Notation)一种简单的数据格式,比xml更轻巧.JSON是JavaScript原生格式,这意味着在JavaScript中处理JSON数据不需要任何特殊的API或工具包. JSON的规则很简单:对象是一个无序的"'名称/值'对"集合.一个对象以"{"(左括号)开始,"}"(右括号)结束.每个"名称"后跟一个":"(冒号):"'名称/值'对"