MySQL中函數列舉

一、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 值之間的天數。該函數接受兩個任何有效日期或日期時間值的參數。如果傳遞dateTimetimeStamp值,則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的結果,其中beginend是 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;

執行 sql 之后:

注: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)

  1. 假如字符串str在由N子鏈組成的字符串列表strlist中,則返回值的范圍在1到N之間。
  2. 一個字符串列表就是一個由一些被‘,’符號分開的自鏈組成的字符串。
  3. 如果第一個參數是一個常數字符串,而第二個是typeSET列,則FIND_IN_SET()函數被優化,使用比特計算。
  4. 如果str不在strlist或strlist為空字符串,則返回值為0。
  5. 如任意一個參數為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
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,527評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,687評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 178,640評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,957評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,682評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,011評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,009評論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,183評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,714評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,435評論 3 359
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,665評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,148評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,838評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,251評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,588評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,379評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,627評論 2 380