运算符连接表达式中的各个操作数,其作用是指明对操作数所进行的运算。运用运算符可以更加灵活地使用表中的数据。本节将介绍MySQL中的运算符的特点和使用方法。
运算符是告诉MySQL执行特定算术或逻辑操作的符号。MySQL内部的运算符很丰富,主要有四大类,分别是算术运算符、比较运算符、逻辑运算符、位运算符。
算术运算符用于各类数值运算,包括加(+)、减(−)、乘(*)、除(/)和求余(或称模运算,%)。
比较运算符用于比较运算,包括大于(>)、小于(<)、等于(=)、大于或等于(>=)、小于或等于(<=)、不等于(!=),以及IN、BETWEEN AND、IS NULL、GREATEST、LEAST、LIKE、REGEXP等。
逻辑运算符的求值结果均为1(TRUE)或0(FALSE),这类运算符有逻辑非(NOT或者!)、逻辑与(AND或者&&)、逻辑或(OR或者||)、逻辑异或(XOR)。
位运算符包括位与(&)、位或(|)、位非(~)、位异或(^)、左移(<<)、右移(>>)6种,参与运算的操作数按二进制位进行运算。
接下来将对MySQL中的4种运算符的使用进行详细介绍。
算术运算符是MySQL中最基本的运算符,如表3.10所示。
表3.10 MySQL中的算术运算符
下面分别讨论不同算术运算符的使用方法。
【 例3.26 】创建表tmp14,定义数据类型为INT的字段num,插入值64,对num值进行加法和减法运算。
(1)创建表tmp14:
CREATE TABLE tmp14 ( num INT);
(2)向字段num插入数据64:
INSERT INTO tmp14 value(64);
(3)对num值进行加法和减法运算:
由计算结果可以看到,可以对num字段的值进行加法和减法的运算,而且“+”和“–”的优先级相同,先加后减或者先减后加的结果是相同的。
【 例3.27 】对表tmp14中的num值进行乘法和除法运算,SQL语句如下:
由计算结果可以看到,对num值进行除法运算的时候,由于64无法被3整除,MySQL对num/3求商的结果保存到了小数点后面4位,结果为21.3333;64除以3的余数为1,因此取余运算num%3的结果为1。
在进行数学运算时,除数为0的除法是没有意义的,因此除法运算中的除数不能为0,如果被0除,则返回结果为NULL。
【 例3.28 】用num除以0,SQL语句如下:
由计算结果可以看到,除数为0时,对num值进行求商或者求余运算的结果均为NULL。
一个比较运算符的结果总是1、0或者是NULL。比较运算符经常在SELECT的查询条件子句中使用,用来查询满足指定条件的记录。MySQL中的比较运算符如表3.11所示。
表3.11 MySQL中的比较运算符
下面分别讨论不同比较运算符的使用方法。
等号(=)用来判断数字、字符串和表达式是否相等:如果相等,则返回值为1;否则返回值为0。
【 例3.29 】使用“=”进行相等判断,SQL语句如下:
由结果可以看到,在进行判断时,2=2和'2'=2的返回值相同,都为1。因为在进行判断时,MySQL自动进行了转换,把字符'2'转换成了数字2;'b'='b'为相同的字符比较,因此返回值为1;表达式1+3和表达式2+2的结果都为4,因此结果相等,返回值为1;由于“=”不能用于NULL的判断,因此返回值为NULL。
数值比较时有如下规则:
(1)若有一个或两个参数为NULL,则比较运算的结果为NULL。
(2)若同一个比较运算中的两个参数都是字符串,则按照字符串进行比较。
(3)若两个参数均为整数,则按照整数进行比较。
(4)若用字符串和数字进行相等判断,则MySQL可以自动将字符串转换为数字。
这个操作符和“=”操作符执行相同的比较操作,不过“<=>”可以用来判断NULL。在两个操作数均为NULL时,其返回值为1,而不为NULL;当一个操作数为NULL时,其返回值为0,而不为NULL。
【 例3.30 】使用“<=>”进行相等判断,SQL语句如下:
由结果可以看到,“<=>”在执行比较操作时,其作用和“=”相似,唯一的区别是“<=>”可以用来对NULL进行判断,两者都为NULL时返回值为1。
“<>”或者“!=”用于进行数字、字符串、表达式不相等的判断:如果不相等,则返回值为1;否则返回值为0。这两个运算符不能用于判断NULL。
【 例3.31 】使用“<>”和“!=”进行不相等判断,SQL语句如下:
由结果可以看到,两个不等于运算符的作用相同,都可以进行数字、字符串、表达式的比较判断。
“<=”用来判断左边的操作数是否小于或等于右边的操作数:如果小于或等于,则返回值为1;否则返回值为0。“<=”不能用于判断NULL。
【 例3.32 】使用“<=”进行比较判断,SQL语句如下:
由结果可以看到,左边操作数小于或等于右边时,返回值为1,例如对于4<=4,其判断结果为真,所以返回值为1;当左边操作数大于右边时,返回值为0,例如对于'good'<='god','good'第3个位置的o字符在字母表中的顺序大于'god'中第3个位置的d字符,因此判断结果为假,返回值为0;比较NULL时,返回NULL。
“<”运算符用来判断左边的操作数是否小于右边的操作数:如果小于,则返回值为1;否则返回值为0。“<”不能用于判断NULL。
【 例3.33 】使用“<”进行比较判断,SQL语句如下:
从结果中可以看到,当左边操作数小于右边操作数时,返回值为1,例如1<2的返回值1;当左边操作数大于右边操作数时,返回值为0,例如'good'<'god'的返回值为0;比较NULL时返回NULL。
“>=”运算符用来判断左边的操作数是否大于或等于右边的操作数:如果大于或等于,则返回值为1;否则返回值为0。“>=”不能用于判断NULL。
【 例3.34 】使用“>=”进行比较判断,SQL语句如下:
由结果可以看到,左边操作数大于或等于右边操作数时,返回值为1,例如4>=4的返回值为1;当左边操作数小于右边操作数时,返回值为0,例如1>=2的返回值为0;比较NULL时返回NULL。
“>”运算符用来判断左边的操作数是否大于右边的操作数:如果大于,则返回值为1;否则返回值为0。“>”不能用于判断NULL。
【 例3.35 】使用“>”进行比较判断,SQL语句如下:
由结果可以看到,左边操作数大于右边操作数时,返回值为1,例如5.5>5的返回值为1;左边操作数小于右边操作数时,返回值为0,例如1>2的返回值为0;比较NULL时返回NULL。
IS NULL和ISNULL检验一个值是否为NULL:如果为NULL,则返回值为1;否则返回值为0。IS NOT NULL检验一个值是否为非NULL:如果是非NULL,则返回值为1;否则返回值为0。
【 例3.36 】使用IS NULL、ISNULL和IS NOT NULL判断NULL值和非NULL值,SQL语句如下:
由结果可以看到,IS NULL和ISNULL的作用相同,只是格式不同;ISNULL和IS NOT NULL的返回值正好相反。
BETWEEN AND运算符语法格式为:
expr BETWEEN min AND max
假如expr大于或等于min且小于或等于max,则BETWEEN的返回值为1,否则返回值为0。
【 例3.37 】使用BETWEEN AND运算符进行值区间判断,SQL语句如下:
由结果可以看到,当4在端点值区间内或者等于其中一个端点值时,BETWEEN AND表达式返回值为1;12并不在指定区间内,因此返回值为0;对于字符串类型的比较,按字母表中字母顺序进行比较,“x”不在指定的字母区间内,因此返回值为0,而“b”位于指定字母区间内,因此返回值为1。
LEAST运算符语法格式为:
LEAST(值1,值2,...,值n)
其中,“值n”表示参数列表中有n个值。在有两个或多个参数的情况下,返回最小值。假如任意一个自变量为NULL,则LEAST()的返回值为NULL。
【 例3.38 】使用LEAST运算符进行大小判断,SQL语句如下:
由结果可以看到,当参数是整数或者浮点数时,LEAST将返回其中最小的值;当参数为字符串时,返回字母表中顺序最靠前的字符;当比较值列表中有NULL时,不能判断大小,返回值为NULL。
GREATEST运算符语法格式为:
GREATEST(值1, 值2,...,值n)
其中,“值n”表示参数列表中有n个值。当有两个或多个参数时,返回值为最大值。假如任意一个自变量为NULL,则GREATEST()的返回值为NULL。
【 例3.39 】使用GREATEST运算符进行大小判断,SQL语句如下:
由结果可以看到,当参数是整数或者浮点数时,GREATEST将返回其中最大的值;当参数为字符串时,返回字母表中顺序最靠后的字符;当比较值列表中有NULL时,不能判断大小,返回值为NULL。
IN运算符用来判断操作数是否为IN列表中的一个值:如果是,则返回值为1;否则返回值为0。
NOT IN运算符用来判断表达式是否为IN列表中一个值:如果不是,则返回值为1;否则返回值为0。
【 例3.40 】使用IN、NOT IN运算符进行判断,SQL语句如下:
由结果可以看到,IN和NOT IN的返回值正好相反。
在左侧表达式为NULL,或者在IN列表中找不到匹配项并且表中一个表达式为NULL的情况下,IN的返回值均为NULL。
【 例3.41 】存在NULL时的IN查询,SQL语句如下:
IN()语法也可用于在SELECT语句中进行嵌套子查询,在后面的章节中会详细介绍。
LIKE运算符用来匹配字符串,语法格式为:
expr LIKE 匹配条件
如果expr满足匹配条件,则返回值为1;如果不匹配,则返回值为0。expr或匹配条件中任何一个为NULL,则返回值为NULL。
LIKE运算符在进行匹配时,可以使用下面两种通配符:
(1)“%”:匹配任何数目的字符,甚至包括零字符。
(2)“_”:只能匹配一个字符。
【 例3.42 】使用LIKE运算符进行字符串匹配运算,SQL语句如下:
由结果可以看到,指定匹配的字符串为“stud”。“stud”表示直接匹配“stud”字符串,满足匹配条件,返回1;“stu_”表示匹配以stu开头的字符长度为4个字符的字符串,“stud”正好是4个字符,满足匹配条件,因此返回1;“%d”表示匹配以字母“d”结尾的字符串,“stud”满足匹配条件,因此返回1;“t _ _ _”表示匹配以‘t’开头的字符长度为4的字符串,“stud”不满足匹配条件,因此返回0;当字符‘s’与NULL匹配时,结果为NULL。
REGEXP运算符用来匹配字符串,语法格式为:
expr REGEXP 匹配条件
如果expr满足匹配条件,则返回1;如果不满足,则返回0。若expr或匹配条件任意一个为NULL,则返回值为NULL。
REGEXP运算符在进行匹配时,常用的通配符有下面5种:
(1)“^”:匹配以该字符后面的字符开头的字符串。
(2)“$”:匹配以该字符后面的字符结尾的字符串。
(3)“.”:匹配任何一个单字符。
(4)“[...]”:匹配方括号内的任何字符。例如,“[abc]”匹配“a”“b”或“c”。为了命名字符的范围,使用一个“-”。“[a-z]”匹配任何字母,而“[0-9]”匹配任何数字。
(5)“*”:匹配0个或多个在它前面的字符。例如,“x*”匹配任何数量的“x”字符,“[0-9]*”匹配任何数量的数字,而“*”匹配任何数量的任何字符。
【 例3.43 】使用REGEXP运算符进行字符串匹配运算,SQL语句如下:
由结果可以看到,指定匹配的字符串为“ssky”。“^s”表示匹配任何以字母“s”开头的字符串,因此满足匹配条件,返回1;“y$”表示匹配任何以字母“y”结尾的字符串,因此满足匹配条件,返回1;“.sky”表示匹配任何以“sky”结尾、字符长度为4的字符串,因此满足匹配条件,返回1;“[ab]”匹配任何包含字母a或者b的字符串,指定字符串中既没有字母a也没有字母b,因此不满足匹配条件,返回0。
提示 正则表达式是一个可以进行复杂查询的强大工具。相对于LIKE字符串匹配,它可以使用更多的通配符类型,查询结果更加灵活。读者可以参考相关的书籍或资料,详细学习正则表达式的写法,这里就不再详细介绍了。在后面的章节中,将会介绍如何使用正则表达式查询表中的记录。
在SQL中,所有逻辑运算符的求值所得结果均为TRUE、FALSE或NULL。在MySQL中,它们体现为1(TRUE)、0(FALSE)和NULL。MySQL中的逻辑运算符如表3.12所示。
表3.12 MySQL中的逻辑运算符
接下来,分别讨论不同的逻辑运算符的使用方法。
逻辑非运算符“NOT”表示当操作数为0时,所得值为1;当操作数为非零值时,所得值为0;当操作数为NULL时,所得的返回值为NULL。
【 例3.44 】使用逻辑非运算符“NOT”进行逻辑判断,SQL语句如下:
mysql> SELECT NOT 10, NOT (1-1), NOT -5, NOT NULL, NOT 1 + 1; +--------+-----------+--------+----------+-----------+ | NOT 10 | NOT (1-1) | NOT -5 | NOT NULL | NOT 1 + 1 | +--------+-----------+--------+----------+-----------+ | 0 | 1 | 0 | NULL | 0 | +--------+-----------+--------+----------+-----------+
逻辑与运算符“AND”表示当所有操作数均为非零值并且不为NULL时,计算所得结果为1;当一个或多个操作数为0时,所得结果为0;其余情况返回值为NULL。
【 例3.45 】使用逻辑与运算符“AND”进行逻辑判断,SQL语句如下:
mysql> SELECT 1 AND -1,1 AND 0,1 AND NULL, 0 AND NULL; +----------+---------+------------+------------+ | 1 AND -1 | 1 AND 0 | 1 AND NULL | 0 AND NULL | +----------+---------+------------+------------+ | 1 | 0 | NULL | 0 | +----------+---------+------------+------------+
由结果可以看到, “1 AND -1”中没有0或者NULL,因此结果为1;“1 AND 0”中有操作数0,因此结果为0;“1 AND NULL”中虽然有NULL,但是没有操作数0,因此结果为NULL。
提示 “AND”运算符可以有多个操作数,需要注意的是:当有多个操作数进行运算时,AND两边一定要使用空格隔开,否则会影响结果的正确性。
逻辑或运算符“OR”表示当两个操作数均为非NULL值且任意一个操作数为非零值时,结果为1,否则结果为0;当有一个操作数为NULL,且另一个操作数为非零值时,结果为1,否则结果为NULL;当两个操作数均为NULL时,所得结果为NULL。
【 例3.46 】使用逻辑或运算符“OR”进行逻辑判断,SQL语句如下:
mysql> SELECT 1 OR -1 OR 0, 1 OR 2,1 OR NULL, 0 OR NULL, NULL OR NULL; +--------------+--------+-----------+-----------+--------------+ | 1 OR -1 OR 0 | 1 OR 2 | 1 OR NULL | 0 OR NULL | NULL OR NULL | +--------------+--------+-----------+-----------+--------------+ | 1 | 1 | 1 | NULL | NULL | +--------------+--------+-----------+-----------+--------------+
由结果可以看到,“1 OR -1 OR 0”中没有NULL,并且包含有非零的值1和−1,因此结果为1;“1 OR 2”中没有操作数0,因此结果为1;“1 OR NULL”中虽然有NULL,但是有操作数1,因此结果为1;“0 OR NULL”中没有非零值,并且有NULL,因此结果为NULL;“NULL OR NULL”中只有NULL,因此结果为NULL。
逻辑异或运算符“XOR”表示当任意一个操作数为NULL时,返回值为NULL;对于非NULL的操作数,如果两个操作数都是非零值或者都是零值,则返回结果为0,如果一个为零值、另一个为非零值,则返回结果为1。
【 例3.47 】使用逻辑异或运算符“XOR”进行逻辑判断,SQL语句如下:
mysql> SELECT 1 XOR 1, 0 XOR 0, 1 XOR 0, 1 XOR NULL, 1 XOR 1 XOR 1; +---------+---------+---------+------------+---------------+ | 1 XOR 1 | 0 XOR 0 | 1 XOR 0 | 1 XOR NULL | 1 XOR 1 XOR 1 | +---------+---------+---------+------------+---------------+ | 0 | 0 | 1 | NULL | 1 | +---------+---------+---------+------------+---------------+
由结果可以看到,在“1 XOR 1”和“0 XOR 0”中,运算符两边的操作数都为非零值或者都是零值,因此返回0;在“1 XOR 0”中,两边的操作数一个为零值、一个为非零值,因此结果为1;在“1 XOR NULL”中,有一个操作数为NULL,因此结果为NULL;在“1 XOR 1 XOR 1”中,有多个操作数且运算符相同,因此从左到右依次计算,“1 XOR 1”的结果为0,再与1进行异或运算,最终结果为1。
提示 a XOR b的计算等同于(a AND (NOT b))或者((NOT a)AND b)。
位运算符是在二进制数上进行计算的运算符。位运算符会先将操作数变成二进制数,然后进行位运算,最后将计算结果从二进制数变回十进制数。MySQL中提供的位运算符有位或(|)、位与(&)、位异或(^)、位左移(<<)、位右移(>>)和位取反(~)运算符,如表3.13所示。
表3.13 MySQL中的位运算符
接下来,分别讨论不同的位运算符的使用方法。
位或运算的实质是将参与运算的几个操作数按照对应的二进制数逐位进行逻辑或运算。对应的二进制位有一个或两个为1,则该位的运算结果为1,否则为0。
【 例3.48 】使用位或运算符进行运算,SQL语句如下:
mysql> SELECT 10 | 15, 9 | 4 | 2; +---------+-----------+ | 10 | 15 | 9 | 4 | 2 | +---------+-----------+ | 15 | 15 | +---------+-----------+
10的二进制值为1010,15的二进制值为1111,按位或运算之后,结果为1111,即十进制整数15;9的二进制值为1001,4的二进制值为0100,2的二进制值为0010,按位或运算之后,结果为1111,即十进制整数15。其结果为一个64位无符号整数。
位与运算的实质是将参与运算的几个操作数按照对应的二进制数逐位进行逻辑与运算。对应的二进制位都为1,则该位的运算结果为1,否则为0。
【 例3.49 】使用位与运算符进行运算,SQL语句如下:
10的二进制值为1010,15的二进制值为1111,按位与运算之后,结果为1010,即十进制整数10;9的二进制值为1001,4的二进制值为0100,2的二进制值为0010,按位与运算之后,结果为0000,即十进制整数0。其结果为一个64位无符号整数。
位异或运算的实质是将参与运算的两个操作数按照对应的二进制数逐位进行逻辑异或运算。对应位的二进制数不同时,对应位的结果才为1。如果两个对应位的数都为0或者都为1,则对应位的结果为0。
【 例3.50 】使用位异或运算符进行运算,SQL语句如下:
10的二进制值为1010,15的二进制值为1111,按位异或运算之后,结果为0101,即十进制整数5;1的二进制值为0001,0的二进制值为0000,按位异或运算之后,结果为0001,即十进制整数1;1和1本身二进制位完全相同,因此结果为0。
位左移运算符<<使指定的二进制值的所有位都左移指定的位数。左移指定位数之后,左边高位的数值将被移出并丢弃,右边低位空出的位置用0补齐。语法格式为:
expr<<n
其中,n指定expr要移位的位数。
【 例3.51 】使用位左移运算符进行运算,SQL语句如下:
1的二进制值为0000 0001,左移两位之后变成0000 0100,即十进制整数4;十进制4左移两位之后变成0001 0000,即整十进制数16。
位右移运算符使指定的二进制值的所有位都右移指定的位数。右移指定位数之后,右边低位的数值将被移出并丢弃,左边高位空出的位置用0补齐。语法格式为:
expr>>n
其中,n指定expr要移位的位数。
【 例3.52 】使用位右移运算符进行运算,SQL语句如下:
1的二进制值为0000 0001,右移1位之后变成0000 0000,即十进制整数0;16的二进制值为0001 0000,右移两位之后变成0000 0100,即十进制整数4。
位取反运算的实质是将参与运算的操作数按照对应的二进制数逐位反转,即1取反后变为0,0取反后变为1。
【 例3.53 】使用位取反运算符进行运算,SQL语句如下:
在逻辑运算“5&~1”中,由于位取反运算符(~)的级别高于位与运算符(&),因此先对1进行取反操作,取反之后,除了最低位为0外,其他位都为1,即1110;然后与十进制数值5进行与运算,结果为0100,即十进制整数4。
提示 MySQL经过位运算之后的数值是一个64位的无符号整数,1的二进制值表示为最右边位为1、其他位均为0,取反操作之后,除了最低位为0外,其他位均变为1。
可以使用BIN()函数查看1取反之后的结果,SQL语句如下:
mysql> SELECT BIN(~1); +------------------------------------------------------------------+ | BIN(~1) | +------------------------------------------------------------------+ | 1111111111111111111111111111111111111111111111111111111111111110 | +------------------------------------------------------------------+
这样,读者就可以明白【例3.53】是如何计算的了。
运算符的优先级决定了不同的运算符在表达式中计算的先后顺序。表3.14列出了MySQL中的各类运算符及其优先级。
表3.14 运算符按优先级由低到高排列
可以看到,不同运算符的优先级是不同的。一般情况下,级别高的运算符先进行计算,如果级别相同,MySQL按表达式的顺序从左到右依次计算。当然,在无法确定优先级的情况下,可以使用圆括号来改变优先级,这样会使计算过程更加清晰。