SQL SERVER2005数据类型(二)
时间:2010-10-10 来源:永不放弃1988
1、与字符相关的问题
<1>模式匹配
LIKE 关键字搜索与指定模式匹配的字符串、日期或时间值。有关详细信息,请参阅数据类型 (Transact-SQL)。LIKE 关键字使用常规表达式包含值所要匹配的模式。模式包含要搜索的字符串,字符串中可包含四种通配符的任意组合。
通配符 | 含义 |
---|---|
% |
包含零个或多个字符的任意字符串。 |
_ |
任何单个字符。 |
[ ] |
指定范围(例如 [a-f])或集合(例如 [abcdef])内的任何单个字符。 |
[^] |
不在指定范围(例如 [^a - f])或集合(例如 [^abcdef])内的任何单个字符。 |
请将通配符和字符串用单引号引起来,例如:
LIKE 'Mc%' 将搜索以字母 Mc 开头的所有字符串(如 McBadden)。
LIKE '%inger' 将搜索以字母 inger 结尾的所有字符串(如 Ringer 和 Stringer)。
LIKE '%en%' 将搜索任意位置包含字母 en 的所有字符串(如 Bennet、Green 和 McBadden)。
LIKE '_heryl' 将搜索以字母 heryl 结尾的所有六个字母的名称(如 Cheryl 和 Sheryl)。
LIKE '[CK]ars[eo]n' 将搜索 Carsen、Karsen、Carson 和 Karson(如 Carson)。
LIKE '[M-Z]inger' 将搜索以字母 inger 结尾、以 M 到 Z 中的任何单个字母开头的所有名称(如 Ringer)。
LIKE 'M[^c]%' 将搜索以字母 M 开头,并且第二个字母不是 c 的所有名称(如 MacFeather)。
可以将 NOT LIKE 与同样的通配符结合使用。若要查找 Contact 表中区号不是 415 的所有电话号码,请使用下列等价查询中的任意一个:
SELECT Phone FROM AdventureWorks.Person.Contact WHERE Phone NOT LIKE '415%' -- Or SELECT Phone FROM AdventureWorks.Person.Contact WHERE NOT Phone LIKE '415%' |
注意:当使用 not like谓词时,表达式将不再是一个SARG,优化器不再考虑使用有序的索引访问。
图:where 筛选器分别使用like与not like的执行计划
不与 LIKE 一同使用的通配符将解释为常量而非模式,换言之,这些通配符仅代表其本身的值。以下查询试图查找只由四个字符 415% 组成的电话号码。该查询将不会查找以 415 开头的电话号码。有关常量的详细信息,请参阅常
SELECT Phone FROM AdventureWorks.Person.Contact WHERE Phone = '415%' |
注意:使用通配符时应着重考虑的另一个问题是对性能的影响。如果表达式以通配符开头,就不能使用索引(就如同给定了姓名 "%mith" 而非 "Smith" 时,将无法知道应从电话簿的哪一页开始查找)。表达式中间或结尾处的通配符不妨碍索引的使用(就如同在电话簿中一样,如果姓名为 "Samuel%",则不论 Samuels 和 Samuelson 是否都在电话簿上,都知道该从何处开始查找。)
搜索通配符字符
可以搜索通配符字符。有两种方法可指定平常用作通配符的字符:
- 使用 ESCAPE 关键字定义转义符。在模式中,当转义符置于通配符之前时,该通配符就解释为普通字符。例如,要搜索在任意位置包含字符串 5% 的字符串,请使用:
WHERE ColumnA LIKE '%5/%%' ESCAPE '/'
- 在方括号 ([ ]) 中只包含通配符本身。若要搜索破折号 (-) 而不是用它指定搜索范围,请将破折号指定为方括号内的第一个字符:
WHERE ColumnA LIKE '9[-]5'
符号 含义 LIKE '5[%]'
5%
LIKE '5%'
5 后跟 0 个或多个字符的字符串
LIKE '[_]n'
_n
LIKE '_n'
an, in, on (and so on)
LIKE '[a-cdf]'
a、b、c、d 或 f
LIKE '[-acdf]'
-、a、c、d 或 f
LIKE '[ [ ]'
[
LIKE ']'
]
如果使用 LIKE 执行字符串比较,模式串中的所有字符(包括每个前导空格和尾随空格)都有意义。如果要求比较返回带有字符串 LIKE 'abc '(abc 后跟一个空格)的所有行,将不会返回列值为 abc(abc 后没有空格)的行。但是反过来,情况并非如此。可以忽略模式所要匹配的表达式中的尾随空格。如果要求比较返回带有字符串 LIKE 'abc'(abc 后没有空格)的所有行,将返回以 abc 开头且具有零个或多个尾随空格的所有行。
2、区分大小写的筛选器
字符数据的敏感性依赖于数据的排序规则属性。多数环境都使用SQL Server默认的不区分大小写的排序规则。当你使用区分大小写的筛选器返回信息时,应该注意性能问题:
可以从上图看出,在第一种方案中你对基列customerID进行了处理,筛选器不再是一个SARG,优化器不会考虑使用索引。但可以像第二种方案那样,再添加一个区分大小写的筛选器,以允许优化器使用索引。
知识补充:
COLLATE { <collation_name> | database_default }
<collation_name> :: =
{ Windows_collation_name } | { SQL_collation_name }
用途:一个子句,可应用于数据库定义或列定义以定义排序规则,或应用于字符串表达式以应用排序规则转换。
3、大型对象
max 说明符
当你使用可变长度的数据类型:varchar,nvarchar或varbinary定义列、变量或参数时,可以指定max说明符代替实际大小。通过使用max说明符,sql server存储的值可以达到LOB支持的最大容量,即2g。
4、隐式转换
<1>标量表达式
如果二元表达式的两个操作数具有不同的数据类型,在评估结果之前将先把低优先级的操作数隐式转换为高优先级操作数 的数据类型。
SQL_VARIANT是一种数据类型,用于存储 SQL Server 2005 支持的各种数据类型(不包括 text、ntext、image、timestamp 和 sql_variant)的值。
sql_variant 的最大长度可以是 8016 个字节。这包括基类型信息和基类型值。实际基类型值的最大长度是 8,000 个字节。
对于 sql_variant 数据类型,必须先将它转换为其基本数据类型值,然后才能参与诸如加减这类运算。
可以为 sql_variant 分配默认值。该数据类型还可以将 NULL 作为其基础值,但是 NULL 值没有关联的基类型。而且,sql_variant 不能以另一个 sql_variant 作为它的基类型。
唯一键、主键或外键可能包含类型为 sql_variant 的列,但是,组成指定行的键的数据值的总长度不应大于索引的最大长度。该最大长度是 900 个字节。
一个表可以包含任意多个 sql_variant 列。
不能在 CONTAINSTABLE 和 FREETEXTTABLE 中使用 sql_variant。
比较 sql_variant 值
sql_variant 数据类型在用于转换的数据类型层次结构列表中位于顶部。为了进行 sql_variant 比较,SQL Server 数据类型层次结构顺序划分为多个数据类型系列。
数据类型层次结构 | 数据类型系列 |
---|---|
sql_variant |
sql_variant |
datetime |
日期和时间 |
smalldatetime |
日期和时间 |
float |
近似数值 |
real |
近似数值 |
decimal |
精确数值 |
money |
精确数值 |
smallmoney |
精确数值 |
bigint |
精确数值 |
int |
精确数值 |
smallint |
精确数值 |
tinyint |
精确数值 |
bit |
精确数值 |
nvarchar |
Unicode |
nchar |
Unicode |
varchar |
Unicode |
char |
Unicode |
varbinary |
Binary |
binary |
Binary |
uniqueidentifier |
Uniqueidentifier |
下列规则适用于 sql_variant 比较:
- 当不同基本数据类型的 sql_variant 值进行比较,而且基本数据类型属于不同的数据类型系列时,则在层次结构图中数据类型系列较高的值被认为在两个值中较大。
- 当不同基本数据类型的 sql_variant 值进行比较,而且基本数据类型属于相同的数据类型系列时,则在层次结构图中基本数据类型较低的值先隐式转换为其他数据类型,然后再进行比较。
- 在比较 char、nvarchar、nchar 或 varchar 数据类型的 sql_variant 值时,将基于以下条件进行计算:LCID、LCID 版本、比较标志和排序 ID。各个条件按所列出的顺序作为整数值进行比较。