一、ifNull(expr1,expr2)
ifNull(expr1,expr2)
類似于 Oracle 中的 NVL 函數。含義是:如果第一個參數不為空,則返回第一個參數,否則返回第二個參數。
二、if(expr1,expr2,expr3)
if(expr1,expr2,expr3)
類似于 Oracle 中的NVL2 函數。含義是:如果第一個表達式的值為 true(不為 0 或不為 null),則返回第二個參數的值,否則返回第三個參數的值。
三、dateDiff函數(begin減去end)
dateDiff(begin,end);
計算兩個 date,dateTime 或 timeStamp 值之間的天數。該函數接受兩個任何有效日期或日期時間值的參數。如果傳遞dateTime
或timeStamp
值,則dateDiff
函數僅將日期部分用于計算,并忽略時間部分。
SELECT
NOW() a,
-- NOW() 函數返回當前的日期和時間
DATEDIFF(NOW(),'2008-08-08') b,
-- begin>end,返回正數
DATEDIFF('2008-08-08',NOW()) c,
-- end<begin,返回負數
DATEDIFF(NOW(),'2008-08-08') / 7 as weeks1,
ROUND(DATEDIFF(NOW(),'2008-08-08') / 7, 2) as weeks2,
DATEDIFF(NOW(),'2008-08-08') / 30 as months1,
ROUND(DATEDIFF(NOW(),'2008-08-08') / 30,2) as months2
-- 將dateDiff函數的返回值除以7或30,可以計算周或月作為間隔時間
四、 timeStampDiff函數
1??區別于dateDiff(begin,end);
,timeStampDiff(unit,begin,end);
返回end-begin
的結果,其中begin
和end
是 date 或 dateTime 表達式。
2??該函數允許其參數具有混合類型,例如,begin
是 date 值,end
可以是 dateTime 值。 如果使用 date 值,則timeStampDiff
函數將其視為時間部分為“00:00:00”
的 dateTime 值。
3??unit
參數是確定(end-begin
)的結果的單位,表示為整數。 以下是有效單位:
- MICROSECOND
- SECOND
- MINUTE
- HOUR
- DAY
- WEEK
- MONTH
- QUARTER
- YEAR
SELECT
TIMESTAMPDIFF(YEAR, '2010-01-01', '2000-12-31') YEAR,
-- 不足10年,為9。而不是(2000-2010)
TIMESTAMPDIFF(MONTH, '2000-01-31', '2000-08-08') MONTH,
-- 不足7個月,為6。而不是(8-1)
TIMESTAMPDIFF(DAY, '2010-01-01', '2000-01-01') DAY,
TIMESTAMPDIFF(MINUTE, '2000-01-01 10:00:00', '2000-01-01 10:45:59') MINUTE,
-- 差值應該是45分59秒。 但是,將unit參數傳遞為MINUTE,此時,函數返回45分鐘
TIMESTAMPDIFF(SECOND, '2000-01-01 00:00:00', '2000-01-01 00:45:59') SECOND
-- 注:45分59秒= 45×60 + 59(秒)= 2759秒
五、截取字符串subString(str, pos)
subString(str, pos)
等同于subStr(str, pos)
select
name a,
SUBSTRING(name FROM 1 FOR 1) b,
SUBSTRING(name ,1,1) c,
SUBSTRING(name ,3,1) d,
substring(name ,3,2) e
from tab;
select substr('abcde',2,3),substr('abcde',2),substr('abcde',-4,5) from dual;
六、trim( )、rTrim( )、lTrim( )
SELECT
TRIM(' solos ') a,
TRIM(LEADING 's' FROM 'solos') b,
TRIM(LEADING 'S' FROM 'solos') c,
TRIM(TRAILING 's' FROM 'solos') d,
TRIM(BOTH 's' FROM 'solos') e,
LTRIM(' solos ') f,
RTRIM(' solos ') g,
TRIM('a' from 'abacde') h;
七、concat()
1??concat(str1, str2,...)
將多個字符串連接成一個字符串。返回結果為連接參數產生的字符串,如果有任何一個參數為 null,則返回值為 null。
2??concat(str1, seperator,str2,seperator,...)
返回結果為連接參數產生的字符串并且有分隔符,如果有任何一個參數為 null,則返回值為 null。
select * from ext_area; -- 3193
select * from ext_area where area like concat('%'); -- 3193
select * from ext_area where area like concat('%%'); -- 3193
select * from ext_area where area like concat('%''%'); -- null
select * from ext_area where area like concat('%' '%'); -- 3193
select concat('hello',',word!') str; -- 結果為:'hello,word!'
八、concat_ws()
區別于concat()
,concat_ws(separator, str1, str2, ...)
雖然也是將多個字符串連接成一個字符串,但是可以一次性指定分隔符【concat_ws 就是 concat with separator】
說明:第一個參數指定分隔符。需要注意的是分隔符不能為 null,如果為 null,則返回結果為 null。
九、group_concat()
以 deptId 分組,把 name 的[去重]值打印在一行,逗號分隔(默認)
select deptId,group_concat(distinct name) from tab group by deptId;
注:group_concat 只有與 group by 同時使用才能產生效果。
十、length: 獲取字節個數(utf-8 一個漢字為3個字節,gbk為2個字節)
SELECT
LENGTH( 'crud' ) a,
LENGTH( '哈crud' ) b,
length( 'a bc' ) c
FROM
DUAL;
十一、大小寫轉換
SELECT
lower( 'NAME' ) a,
upper( 'NaMe' ) b,
CONCAT( UCASE( LEFT ( 'name', 1 ) ), SUBSTRING( 'name', 2 ) ) c
FROM
DUAL;
十二、instr:返回子串第一次出現的索引,如果沒有則返回0
SELECT
INSTR( '哈嘍啊', '哈' ) a,
INSTR( '哈嘍啊', '哦' ) b;
-- a為1;b為0 (mysql是從1開始算位數)
十三、用指定字符實現左/右填充指定長度
SELECT
LPAD( 'crud', 6, '*' ) a,
LPAD( 'crud', 2, '*' ) b,
RPAD( 'crud', 7, '*' ) c,
RPAD( 'crud', 2, '*' ) d
十四、替換函數replace()
select
REPLACE('6166666','1','ac') a,
REPLACE('666A666','A','ac') b
from dual;
/四舍五入/
select round(23.44),round(23.44,1),round(53.44,-2.2) from dual;
/向上取整,向下取整/
select ceil(23.44),floor(23.44) from dual;
/絕對值/
select abs(-33.42),abs(33.42),abs(0) from dual;
/求余函數/
select mod(5,2) from dual;
/m的n次方/
select power(2,3),power(null,2) from dual;
/取平方根/
select sqrt(16) from dual;
/數學函數/
select tan(6) from dual;
/拼接字符串/
select 'ab'||'cd',concat('ab','cd') from dual;
/獲取系統時間/
select sysdate from dual;
/當前日期加整數月/
select add_months(sysdate,3),add_months(sysdate,-3) from dual;
/返回下一個指定天的日期/
select next_day(sysdate,'星期一') from dual;
/返回date所在月的最后一天/
select last_day(sysdate) from dual;
/計算兩個日期之間間隔的月數/
select months_between('18-12月-15','18-1月-15') from dual;
/獲取年月日時分秒/
select extract(year from sysdate) YEAR,extract(month from sysdate) month,extract(day from sysdate) DAY from dual;
FIND_IN_SET(str,strlist)
- 假如字符串str在由N子鏈組成的字符串列表strlist中,則返回值的范圍在1到N之間。
- 一個字符串列表就是一個由一些被‘,’符號分開的自鏈組成的字符串。
- 如果第一個參數是一個常數字符串,而第二個是typeSET列,則FIND_IN_SET()函數被優化,使用比特計算。
- 如果str不在strlist或strlist為空字符串,則返回值為0。
- 如任意一個參數為NULL,則返回值為NULL。這個函數在第一個參數包含一個逗號(‘,’)時將無法正常運行。
strlist:一個由英文逗號“,”鏈接的字符串,例如:"a,b,c,d",該字符串形式上類似于SET類型的值被逗號給鏈接起來。
示例:SELECT FIND_IN_SET('b','a,b,c,d')
//返回值為2,即第2個值。
MySQL中的 IN和FIND_IN_SET的查詢問題
原來以為mysql可以進行這樣的查詢
1??select id, list, name from table where 'Uzi' IN (list)
注:1. table含有三個字段id:int, list:varchar(255), name:varchar(255)。實際上這樣是不行的。
測試代碼:
CREATE TABLE test (
id int(8) NOT NULL auto_increment,
name varchar(255) NOT NULL,
list varchar(255) NOT NULL,
PRIMARY KEY (`id`)
)
insert into test values (1, 'name', 'Uzi,xiaohu,letme');
insert into test values (2, 'name2', 'xiaohu,Uzi,letme');
insert into test values (3, 'name3', 'letme,Uzi,xiaohu');
test1:sql =select * from test where 'Uzi' IN ('list');
得到結果空值。
test2:sql =select * from test where FIND_IN_SET('Uzi','list');
得到三條數據:
1 name Uzi,xiaohu,letme
2 name2 xiaohu,Uzi,letme
3 name3 letme,Uzi,xiaohu
修改表數據:
update test set list='Uzi' where id='1';
然后執行test1的sql,可以返回一條結果。
再來看看這個:
2??select id, list, name from table where 'Uzi' IN ('libk', 'zyfon', 'Uzi');
1??2??到底有什么區別呢?為什么第一條不能取得正確的結果,而第二條卻能取得結果?
原因其實是1??中(list),list是變量, 而2??中('libk', 'zyfon', 'Uzi')是常量。所以如果要讓1??能正確工作,需要用find_in_set():
select id, list, name from table where FIND_IN_SET( 'Uzi' , list)
1??的改進版。
總結:所以如果list是常量,則可以直接用IN,否則要用FIND_IN_SET()函數。
十、CAST()函數和CONVERT()函數
MySQL 的CAST()和CONVERT()函數可用來獲取一個類型的值,并產生另一個類型的值。兩者具體的語法如下:
CAST(value as type);
CONVERT(value, type);
就是CAST(xxx AS 類型)
CONVERT(xxx,類型)
。可以轉換的類型是有限制的。這個類型可以是以下值其中的一個:
- 二進制,同帶binary前綴的效果 : BINARY
- 字符型,可帶參數 : CHAR()
- 日期 : DATE
- 時間: TIME
- 日期時間型 : DATETIME
- 浮點數 : DECIMAL
- 整數 : SIGNED(字段取值范圍是-128 - 127)
- 無符號整數 : UNSIGNED(字段取值范圍是0-255)
例:
SELECT CONVERT('23',SIGNED) a,CAST('125e342.83' AS signed) b,CAST('3.35' AS signed) c;
結果如下:
應用
在MySQL中,進行中文排序和查找的時候,對漢字的排序和查找結果往往都是錯誤的。 這種情況在MySQL的很多版本中都存在。之所以這樣,是因為MySQL在查詢字符串時是大小寫不敏感的,在編繹MySQL時一般以ISO-8859字符集作為默認的字符集,因此在比較過程中中文編碼字符大小寫轉換造成了這種現象。
解決方法:
1??對于包含中文的字段加上”binary”屬性,使之作為二進制比較,例如將name char(10)
改成name char(10) binary
。
2??如果使用源碼編譯MySQL,可以編譯MySQL時使用 –with–charset=gbk
參數,這樣MySQL就會直接支持中文查找和排序了(默認的是latin1)。也可以用extra-charsets=gb2312,gbk
來加入多個字符集。
3??如果不想對表結構進行修改或者重新編譯MySQL,也可以在查詢語句的 order by 部分使用 CONVERT 函數。例如:
select * from tab order by CONVERT(ChineseColumnName USING gbk);
UTF8 默認校對集是utf8_general_ci
,它不是按照中文來的。需要強制讓MySQL按中文來排序。
十一、DATE_FORMAT(date,format)
DATE_FORMAT() 函數用于以不同的格式顯示日期/時間數據。date 參數是合法的日期。format 規定日期/時間的輸出格式。
可以使用的格式有:
格式 描述
%a 縮寫星期名
%b 縮寫月名
%c 月,數值
%D 帶有英文前綴的月中的天
%d 月的天,數值(00-31)
%e 月的天,數值(0-31)
%f 微秒
%H 小時 (00-23)
%h 小時 (01-12)
%I 小時 (01-12)
%i 分鐘,數值(00-59)
%j 年的天 (001-366)
%k 小時 (0-23)
%l 小時 (1-12)
%M 月名
%m 月,數值(00-12)
%p AM 或 PM
%r 時間,12-小時(hh:mm:ss AM 或 PM)
%S 秒(00-59)
%s 秒(00-59)
%T 時間, 24-小時 (hh:mm:ss)
%U 周 (00-53) 星期日是一周的第一天
%u 周 (00-53) 星期一是一周的第一天
%V 周 (01-53) 星期日是一周的第一天,與 %X 使用
%v 周 (01-53) 星期一是一周的第一天,與 %x 使用
%W 星期名
%w 周的天 (0=星期日, 6=星期六)
%X 年,其中的星期日是周的第一天,4 位,與 %V 使用
%x 年,其中的星期一是周的第一天,4 位,與 %v 使用
%Y 年,4 位
%y 年,2 位
應用
select DATE_FORMAT(NOW(),'%b %d %Y %h %i %p');
select DATE_FORMAT(NOW(),'%m-%d-%Y');
select DATE_FORMAT(NOW(),'%d %b %y');
select DATE_FORMAT(NOW(),'%d %b %Y %T:%f');
結果如下:
Jan 11 2020 01 17 PM
01-11-2020
11 Jan 20
11 Jan 2020 13:17:40:000000