【1】字符串类型递归报错
在CTE递归测试,遇到了小问题:
登时就迷糊了:不都是取的是Unit表中的同一个列,相加之后类型就变了么?
难道是因为,系统知道这是在进行递归运算,但又不确定递归的层次,以及字符串连接后的长度,于是把这个字符串的长度预设为很大很大的。
例子1:
;with t1(id ,num1 ) as ( select 1 ,1 union all select id+1,id+id*10 from t1 where id<=10 )
但是上面例子中的 num 1也进行了运算,为什么就不需要显示转换类型?
【2】错误情况
、
【2】基本分析与验证
这是因为递归也不是无限递归的,也是有层次限制的:
【在测试递归查询的结果时,可以通过在 INSERT、UPDATE、DELETE 或 SELECT 语句的 OPTION 子句中使用 MAXRECURSION 提示和 0 到 32,767 之间的值,来限制特定语句允许的递归级数。】
而int 足够了:(目测是int类型,因为直接写一个数字, 默认的类型就是int)
Int:从 -2^31 (-2,147,483,648) 到 2^31 - 1 (2,147,483,647) 的整型数据(所有数字)。存储大小为 4 个字节
Int类型猜测正确, 但为什么错误提示是:溢出,而不是更多字节的bigint?
前面不是说了么:数值的默认类型是int,一旦确定就不会再改变。(而且这个环境还没有智能到,根据所赋的值的大小,或者要进行的加减乘除的运算,自动匹配到最合适的类型。)
于是乎,我茅塞顿开:字符串进行运算的时候,也是有默认类型的(字符串运算,默认就会把字符串转换成 varchar(8000) )下面就是做图验证
所以这个问题就是CTE递归时,Union运算要求同结构,同类型;
而运算后的默认类型和自定义的类型不一致, 显示转换下就OK了, 选择哪种转换方式, 看具体需求。
【3】核心问题所在
但其实只要上下字符串类型一致即可;
【3.1】报错
注意,把字符串转换到下面也不行:
如下图,我们其实的 aa其实并不是 varchar(8000),那为什么【2】中我们把 上面的 cast('aa' varchar(8000)) 这样一改就可以了呢?
因为union all 下面递归,经过计算/拼接的字符串,它是一个可变的长度,无法确定它的长度,但字符串默认最多是8000,所以 union all 上半部只要定义为 varchar(8000),那么就可以解决这个问题。
【3.2】解决:上下类型一致
参考:
http://bbs.csdn.net/topics/290079093
http://technet.microsoft.com/zh-cn/library/ms186243(v=sql.105).aspx