`

case when 遇到varchar转为int类型值失败的错误

阅读更多

 

 

问题描述:

在Sql Server 2005下,

使用如下语句报错:在将 varchar 值 '大' 转换成数据类型 int 时失败。

 

注:status 是整型字段

 

select ff=  case when status>'6' then '大'   when status='3' then '小'  else    status     end from member

 

问题解决:

 

select ff=  case when status>'6' then '大'   when status='3' then '小'  else   CAST(status AS varchar(11))   end from member

 

 

原因:在Sql Server 2005中,默认是以int优先级高,因此在返回的结果里有status这个字段,要将它转化了即可;

 

 

 

 

 

 

 

 

 

---------------------------------------------------------------------------

转自:

http://cache.baidu.com/c?m=9d78d513d99212ee0ffa940f484c803a0e54f1744ad1c7630cc3923884142d563416f4bb56664344829f2f2716af384b9cf42102301420c68cbe8d5daec885585f9f5744671df65662d20edec05124b137e129fedb6ff0ca8725e5d8c5d2af4322c84472789786894d7162dd6f814f6da5b19939022e63ad9d3372fe296058ec3433b35089e225187695b19d0243996c96230696dc28ad764af650b31f6c7410fd0ca60826776cfd&p=8b2a950796d811a059efca6d424b&user=baidu

 

公共新闻组中,一个经常出现的问题是“怎样才能根据传递给存储过程的参数返回一个排序的输出?”。一些高水平专家的帮助之下,我整理出了这个问题的几种解决方案。

一、用IF...ELSE执行预先编写好的查询


  对于大多数人来说,首先想到的做法也许是:通过IF...ELSE语句,执行几个预先编写好的查询中的一个。例如,假设要从Northwind数据库查询得到一个货主(Shipper)的排序列表,发出调用的代码以存储过程参数的形式指定一个列,存储过程根据这个列排序输出结果。Listing 1显示了这种存储过程的一个可能的实现(GetSortedShippers存储过程)。


【Listing 1: 用IF...ELSE执行多个预先编写好的查询中的一个】


CREATE PROC GetSortedShippers

@OrdSeq AS int

AS


IF @OrdSeq = 1

SELECT * FROM Shippers ORDER BY ShipperID

ELSE IF @OrdSeq = 2

SELECT * FROM Shippers ORDER BY CompanyName

ELSE IF @OrdSeq = 3

SELECT * FROM Shippers ORDER BY Phone



  这种方法的优点是代码很简单、很容易理解,SQL Server的查询优化器能够为每一个SELECT查询创建一个查询优化计划,确保代码具有最优的性能。这种方法最主要的缺点是,如果查询的要求发生了改变,你必须修改多个独立的SELECT查询——这里是三个。


二、用列名字作为参数

  另外一个选择是让查询以参数的形式接收一个列名字。Listing 2显示了修改后的GetSortedShippers存储过程。CASE表达式根据接收到的参数,确定SQL ServerORDER BY子句中使用哪一个列。注意,ORDER BY子句中的表达式并未SELECT清单中出现。ANSI SQL-92标准中,ORDER BY子句中不允许出现没有SELECT清单中指定的表达式,但ANSI SQL-99标准允许。SQL Server一直允许这种用法。


【Listing 2:用列名字作为参数,第一次尝试】


CREATE PROC GetSortedShippers

@ColName AS sysname

AS


SELECT *

FROM Shippers

ORDER BY

CASE @ColName

WHEN 'ShipperID' THEN ShipperID

WHEN 'CompanyName' THEN CompanyName

WHEN 'Phone' THEN Phone

ELSE NULL

END



  现,我们来试一下新的存储过程,以参数的形式指定ShipperID列:


EXEC GetSortedShippers 'ShipperID'



  此时一切正常。但是,当我们视图把CompanyName列作为参数调用存储过程时,它不再有效:


EXEC GetSortedShippers 'CompanyName'



  仔细看一下错误信息:


Server: Msg 245, Level 16, State 1, Procedure GetSortedShippers, Line 5

Syntax error converting the nvarchar value 'Speedy

Express' to a column of data type int.



  它显示出,SQL Server试图把“Speedy Express”(nvarchar数据类型转换成一个整数——当然,这个操作是不可能成功的。出现错误的原因于,按照“数据类型优先级”规则,CASE表示式中最高优先级的数据类型决定了表达式返回数据类型。“数据类型优先级”规则可以SQL Server Books Online(BOL)找到,它规定了int数据类型的优先级要比nvarchar数据类型高。前面的代码要求SQL Server按照CompanyName排序输出,CompanyName是nvarchar数据类型。这个CASE表达式的返回可能是ShipperID(int类型),可能是CompanyName(nvarchar类型),或Phone(nvarchar类型)。由于int类型具有较高的优先级,因此CASE表达式返回数据类型应该是int。


为了避免出现这种转换错误,我们可以尝试把ShipperID转换成varchar数据类型。采用这种方法之后,nvarchar作为最高优先级的数据类型被返回。Listing 3显示了修改后的GetSortedShippers存储过程。

【Listing 3:用列名字作为参数,第二次尝试】


ALTER PROC GetSortedShippers

@ColName AS sysname

AS


SELECT *

FROM Shippers

ORDER BY

CASE @ColName

WHEN 'ShipperID'

THEN CAST(ShipperID AS varchar(11))

WHEN 'CompanyName'

THEN CompanyName

WHEN 'Phone'

THEN Phone

ELSE NULL

END



  现,假设我们再把三个列名字中的任意一个作为参数调用存储过程,输出结果看起来正确。看起来就象指定的列正确地为查询输出提供了排序标准。但这个表只有三个货主,它们的ID分别是1、2、3。假设我们把更多的货主加入到表,如Listing 4所示(ShipperID列有IDENTITY属性,SQL Server自动为该列生成)。


【Listing 4:向Shippers表插入一些记录】


INSERT INTO Shippers VALUES('Shipper4', '(111) 222-9999')

INSERT INTO Shippers VALUES('Shipper5', '(111) 222-8888')

INSERT INTO Shippers VALUES('Shipper6', '(111) 222-7777')

INSERT INTO Shippers VALUES('Shipper7', '(111) 222-6666')

INSERT INTO Shippers VALUES('Shipper8', '(111) 222-5555')

INSERT INTO Shippers VALUES('Shipper9', '(111) 222-4444')

INSERT INTO Shippers VALUES('Shipper10', '(111) 222-3333')



  现调用存储过程,指定ShipperID作为排序列:


EXEC GetSortedShippers 'ShipperID'



  表一显示了存储过程的输出。ShipperID等于10的记录位置错误,因为这个存储过程的排序输出是字符排序,而不是整数排序。按照字符排序时,10排列2的前面,因为10的开始字符是1。


表一:记录排序错误的查询结果


ShipperID CompanyName Phone

1 Speedy Express (503) 555-9831

10 Shipper10 (111) 222-3333

2 United Package (503) 555-3199

3 Federal Shipping (503) 555-9931

4 Shipper4 (111) 222-9999

5 Shipper5 (111) 222-8888

6 Shipper6 (111) 222-7777

7 Shipper7 (111) 222-6666

8 Shipper8 (111) 222-5555

9 Shipper9 (111) 222-4444


为了解决这个问题,我们可以用前置的0补足ShipperID,使得ShipperID都有同样的长度。按照这种方法,基于字符的排序具有和整数排序同样的输出结果。修改后的存储过程如Listing 5所示。十个0被置于ShipperID的绝对之前,而结果中,代码只是使用最右边的10个字符。SIGN函数确定正数的前面加上加号(+)前缀,还是负数的前面加上负号(-)前缀。按照这种方法,输出结果总是有11个字符,包含一个“+”或“-”字符、前导的字符0以及ShipperID的绝对

【Listing 5:用列名字作为参数,第三次尝试】


ALTER PROC GetSortedShippers

@ColName AS sysname

AS


SELECT *

FROM Shippers

ORDER BY

CASE @ColName

WHEN 'ShipperID' THEN CASE SIGN(ShipperID)

WHEN -1 THEN '-'

WHEN 0 THEN '+'

WHEN 1 THEN '+'

ELSE NULL

END +

RIGHT(REPLICATE('0', 10) +

CAST(ABS(ShipperID) AS varchar(10)), 10)

WHEN 'CompanyName' THEN CompanyName

WHEN 'Phone' THEN Phone

ELSE NULL

END



  如果ShipperID的都是正数,加上符号前缀就没有必要,但为了让方案适用于尽可能多的范围,本例加上了符号前缀。排序时“-”“+”的前面,所以它可以用于正、负数混杂排序的情况。


  现,如果我们用任意三个列名字之一作为参数调用存储过程,存储过程都能够正确地返回结果。Richard Romley提出了一种巧妙的处理方法,如Listing 6所示。它不再要求我们搞清楚可能涉及的列数据类型。这种方法把ORDER BY子句分成三个独立的CASE表达式,每一个表达式处理一个不同的列,避免了由于CASE只返回一种特定数据类型的能力而导致的问题。


【Listing 6:用列名字作为参数,Romley提出的方法】


ALTER PROC GetSortedShippers

@ColName AS sysname

AS


SELECT *

FROM Shippers

ORDER BY

CASE @ColName WHEN 'ShipperID'

THEN ShipperID ELSE NULL END,

CASE @ColName WHEN 'CompanyName'

THEN CompanyName ELSE NULL END,

CASE @ColName WHEN 'Phone'

THEN Phone ELSE NULL END



  按照这种方法编写代码,SQL Server能够为每一个CASE表达式返回恰当的数据类型,而且无需进行数据类型转换。但应该注意的是,只有当指定的列不需要进行计算时,索引才能够优化排序操作。


三、用列号作为参数

  就象第一个方案所显示地那样,你也许更喜欢用列的编号作为参数,而不是使用列的名字(列的编号即一个代表你想要作为排序依据的列的数字)。这种方法的基本思想与使用列名字作为参数的思想一样:CASE表达式根据指定的列号确定使用哪一个列进行排序。Listing 7显示了修改后的GetSortedShippers存储过程。


【Listing 7:用列号作为参数】


ALTER PROC GetSortedShippers

@ColNumber AS int

AS


SELECT *

FROM Shippers

ORDER BY

CASE @ColNumber

WHEN 1 THEN CASE SIGN(ShipperID)

WHEN -1 THEN '-'

WHEN 0 THEN '+'

WHEN 1 THEN '+'

ELSE NULL

END +

RIGHT(REPLICATE('0', 10) +

CAST(ABS(ShipperID) AS varchar(10)), 10)

WHEN 2 THEN CompanyName

WHEN 3 THEN Phone

ELSE NULL

END



  当然,这里你也可以使用Richard的方法,避免ORDER BY子句中列数据类型带来的问题。如果要根据ShipperID排序输出,你可以按照下面的方式调用修改后的GetSortedShippers存储过程:


EXEC GetSortedShippers 1


四、动态执行

  使用动态执行技术,我们能够更轻松地编写出GetSortedShippers存储过程。使用这种方法时,我们只需动态地构造出SELECT语句,然后用EXEC()命令执行这个SELECT语句。假设传递给存储过程的参数是列的名字,存储过程可以大大缩短:


ALTER PROC GetSortedShippers

@ColName AS sysname

AS

EXEC('SELECT * FROM Shippers ORDER BY ' +

@ColName)



  SQL Server 2000和7.0中,你可以用系统存储过程sp_ExecuteSQL替代Exec()命令。BOL说明了使用sp_ExecuteSQL比使用Exec()命令更有利的地方。一般地,如果满足以下三个条件,你能够不授予存储过程所涉及对象权限的情况下,授予执行存储过程的权限:首先,只使用Data Manipulation Language(DML)语言(即SELECT,INSERT,UPDATE,DELETE);其次,所有被引用的对象都有与存储过程同样的所有者;第三,没有使用动态命令。


  上面的存储过程不能满足第三个条件。这种情况下,你必须为所有需要使用存储过程的用户和组显式地授予Shippers表的SELECT权限。如果这一点可以接受的话,一切不存问题。类似地,你可以修改存储过程,使它接受一个列号参数,如Listing 8所示。


【Listing 8:用列号作为参数,动态执行(代码较长的方法)】


ALTER PROC GetSortedShippers

@ColNumber AS int

AS


DECLARE @cmd AS varchar(8000)


SET @cmd = 'SELECT * FROM Shippers ORDER BY ' +

CASE @ColNumber

WHEN 1 THEN 'ShipperID'

WHEN 2 THEN 'CompanyName'

WHEN 3 THEN 'Phone'

ELSE 'NULL'

END


EXEC(@cmd)



  注意,当你使用了函数时,你应该一个变量而不是EXEC()命令内构造SELECT语句。此时,CASE表达式动态地确定使用哪一个列。还有一种更简短的格式,T-SQL允许ORDER BY子句中指定SELECT清单中列的位置,如Listing 9所示。这种格式遵从了SQL-92标准,但ANSI SQL-99标准不支持这种格式,所以最好不要使用这种格式。


【Listing 9:列号作为参数,动态执行(代码较短的方法)】


ALTER PROC GetSortedShippers

@ColNumber AS int

AS

DECLARE @cmd AS varchar(8000)

SET @cmd = 'SELECT * FROM Shippers ORDER BY ' + CAST(@ColNumber AS varchar(4))


EXEC(@cmd)


五、用户定义函数

  如果你使用的是SQL Server 2000,想要编写一个用户定义的函数(UDF),这个用户定义函数接受列的名字或编号为参数、返回排序的结果集,Listing 10显示了大多数程序员当成第一选择的方法。


【Listing 10:列名字作为参数,使用UDF】


CREATE FUNCTION ufn_GetSortedShippers

(

@ColName AS sysname

)

RETURNS TABLE

AS


RETURN

SELECT *

FROM Shippers

ORDER BY

CASE @ColName

WHEN 'ShipperID' THEN CASE SIGN(ShipperID)

WHEN -1 THEN '-'

WHEN 0 THEN '+'

WHEN 1 THEN '+'

ELSE NULL

END +

RIGHT(REPLICATE('0', 10) +

CAST(ABS(ShipperID) AS

varchar(10)), 10)

WHEN 'CompanyName' THEN CompanyName

WHEN 'Phone' THEN Phone

ELSE NULL

END



  但是,SQL Server不接受这个函数,它返回如下错误信息:


Server: Msg 1033, Level 15, State 1, Procedure ufn_GetSortedShippers,

Line 24

The ORDER BY clause is invalid in views, inline functions, and

subqueries, unless TOP is also specified.



  注意错误信息中的“unless”。SQL Server 2000不允许视图、嵌入式UDF、子查询中出现ORDER BY子句,因为它们都应该返回一个表,表不能指定行的次序。然而,如果使用了TOP关键词,ORDER BY子句帮助确定查询所返回的行。因此,如果指定了TOP,你还可以同时指定ORDER BY。由于带有TOP的UDF中允许使用ORDER BY子句,你可以使用一个技巧:把“SELECT *”替换成“SELECT TOP 100 PERCENT *”。这样,你就能够成功地构造出一个接受列名字或编号为参数、返回排序结果的函数。


  新构造的函数可以按照如下方式调用:


SELECT * FROM ufn_GetSortedShippers('ShipperID')



  现,你已经了解了几种用参数确定查询输出中记录次序的方法。编写那些允许用户指定查询结果排序标准的列的应用程序时,你可以使用本文介绍的各种技术,用列名字或编号作为参数,构造出使用CASE表达式和动态执行能力的各种方案。

分享到:
评论

相关推荐

    mysql存储过程之case语句用法实例详解

    简单CASE语句主要用于比较一个表达式(case_expression)与一系列预定义的值(when_expression)。基本语法如下: ```sql CASE case_expression WHEN when_expression_1 THEN commands WHEN when_expression_2 ...

    MySQL case when使用方法实例解析

    `name` varchar(255) DEFAULT NULL, `age` int(2) DEFAULT NULL, `num` int(3) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8; 插入数据: INSERT INTO `t_demo` VALUES (‘1’, ...

    MySql的SQL语言入门教程

    在 MySql 中,常用的数据类型有 INT、BIGINT、FLOAT、DOUBLE、NUMERIC、DATE、DATETIME、CHAR、TIMESTAMP、VARCHAR、BLOB、TEXT 等。 * INT:整数类型,用于存储整数值。 * BIGINT:大整数类型,用于存储大整数值。...

    SQL Server 中的数据类型隐式转换问题

    在上述例子中,我们遇到了一个涉及`CASE WHEN`语句和数据类型隐式转换的问题。 首先,我们创建了一个名为`TEST`的表,包含`ID`(整数),`GOOD_TYPE`(字符串)和`GOOD_WEIGHT`(数值)三列,并插入了一条记录。然后,...

    SQL数据库经典面试题(笔试题).pdf

    可以使用CASE WHEN结合GREATEST函数(如果支持的话)或者嵌套的CASE WHEN语句。 ```sql SELECT GREATEST(A, COALESCE(B, A), COALESCE(C, A)) AS 'MaxCol' FROM table_name; ``` 或者 ```sql SELECT CASE...

    行转列问题

    - `SUM(CASE WHEN cc = 'A' THEN _value END)`:如果`cc`字段的值为`'A'`,则计算`_value`的总和并将其作为新的列`A`的值。 - 同样的逻辑应用于`B`和`C`两列。 #### 知识点扩展 - **聚合函数**:本例中使用了`SUM...

    2022年SQL语句面试题范文.doc

    create table #tmp(语文 int,数学 int ,英语 int) insert into #tmp values(70, 80, 58) select case when 语文>=80 then '优秀' when 语文>=60 then '及格' else '不及格' end as 语文, case when 数学>=...

    SQL经典面试题及答案.pdf

    - 使用`CASE WHEN`语句根据条件返回相应的列值。 **示例代码**: ```sql SELECT CASE WHEN A > B THEN A ELSE B END AS 列1, CASE WHEN B > C THEN B ELSE C END AS 列2 FROM table_name; ``` #### 三、日期筛选...

    汉字转全拼sql语句

    DECLARE @strlen int SELECT @strlen = LEN(@str), @re = '' WHILE @strlen > 0 BEGIN SET @crs = SUBSTRING(@str, @strlen, 1) SELECT @re = CASE WHEN @crs 吖' THEN @crs WHEN @crs 厑' THEN 'a...

    sql交叉表中是使用

    ### SQL Server 中交叉表的使用及遇到的小问题 在 SQL Server 中,交叉表(Pivot Table)是一种将一列中的数据转换为多列的方式,从而实现数据的汇总、分析等功能。这种方式非常适合处理需要进行维度变换的数据场景...

    SQL Server高级内容之case语法函数概述及使用

    SQL Server中的CASE函数是一种非常强大的条件判断工具,它允许我们在SQL查询中实现类似于C#或Java中的switch-case或if-else逻辑。本文将深入探讨CASE函数的用法和应用场景。 CASE函数的基本语法如下: ```sql CASE...

    SQL脚本函数 10进制转换成36进制

    1. **定义**:该函数名为`fn_IntTo36`,接受一个整数类型的参数`@i`,返回一个`varchar(15)`类型的结果。 2. **初始化**:创建一个空字符串`@r`用于保存最终的三十六进制字符串。 3. **循环处理**: - 在每次循环中...

    数据库动态生成列数据库纵向列转横向列

    上述SQL语句首先选择所有原始数据,然后使用`PIVOT`操作符将`subject`列的值转化为新的列名,并将对应的`scores`作为这些新列的值。 - **动态SQL**: 对于不确定列名的情况,可以使用动态SQL来生成所需的查询语句...

    练习8T-SQL编程答案.doc

    @m='n 的值为' print @m+@n go -- 改正后的代码 declare @n decimal(5,2),@m varchar(10) set @n=99.995 set @m='n 的值为' PRINT @m+str(@n) go ``` ### 2. 用户定义类型 (User-Defined Types) 用户定义类型是一...

    SQL脚本函数 10进制转换为34进制

    - `varchar(15)`:返回类型为变长字符类型,最大长度为15个字符。 3. **内部变量**: - `@r varchar(10)`:用于存储转换后的34进制字符串结果。 4. **转换逻辑**: - 使用`while`循环处理每一位数字的转换。 ...

    SQL查询动态字段

    首先创建了一个名为`tb`的表,其中包含了三个字段:一个`varchar(10)`类型的字段(用于存储字符),一个同样为`varchar(10)`类型的字段(用于存储标记或类别),以及一个`int`类型的字段(用于存储数字)。...

    SQL的相关知识---行列转换

    通过遍历所有可能的科目,构造出对应的`CASE WHEN`语句,最终生成的查询语句能够自动适应不同科目的情况,使得查询结果能够包含所有已知科目的成绩。 #### 扩展功能:添加平均分和总分 除了简单的行列转换外,我们...

    Mssql 用bit来定义 bool True False .docx

    这种数据类型非常适合用来定义布尔(bool)值,因为它只需要一个字节的存储空间,而且在处理逻辑条件时非常高效。 1. **定义布尔字段**: 当我们需要在数据库表中表示一个逻辑状态,例如某个特征是否启用、事件...

Global site tag (gtag.js) - Google Analytics