Mysql系列的目標是:經過這個系列從入門到全面掌握一個高級開發所須要的所有技能。html
這是Mysql系列第18篇。java
環境:mysql5.7.25,cmd命令中進行演示。mysql
代碼中被[]包含的表示可選,|符號分開的表示可選其一。sql
上一篇存儲過程&自定義函數,對存儲過程和自定義函數作了一個簡單的介紹,可是如何可以寫出複雜的存儲過程和函數呢?數據庫
這須要咱們熟練掌握流程控制語句才能夠,本文主要介紹mysql中流程控制語句的使用,上乾貨。微信
/*建庫javacode2018*/ drop database if exists javacode2018; create database javacode2018; /*切換到javacode2018庫*/ use javacode2018; /*建立表:t_user*/ DROP TABLE IF EXISTS t_user; CREATE TABLE t_user( id int PRIMARY KEY COMMENT '編號', sex TINYINT not null DEFAULT 1 COMMENT '性別,1:男,2:女', name VARCHAR(16) not NULL DEFAULT '' COMMENT '姓名' )COMMENT '用戶表'; /*插入數據*/ INSERT INTO t_user VALUES (1,1,'路人甲Java'),(2,1,'張學友'),(3,2,'王祖賢'),(4,1,'郭富城'),(5,2,'李嘉欣'); SELECT * FROM t_user; DROP TABLE IF EXISTS test1; CREATE TABLE test1 (a int not null); DROP TABLE IF EXISTS test2; CREATE TABLE test2 (a int not null,b int NOT NULL );
if(條件表達式,值1,值2);
if函數有3個參數。函數
當參數1爲true的時候,返回
值1
,不然返回值2
。oop
需求:查詢
t_user
表數據,返回:編號、性別(男、女)、姓名。spa分析一下:數據庫中性別用數字表示的,咱們須要將其轉換爲(男、女),可使用if函數。3d
mysql> SELECT id 編號,if(sex=1,'男','女') 性別,name 姓名 FROM t_user; +--------+--------+---------------+ | 編號 | 性別 | 姓名 | +--------+--------+---------------+ | 1 | 男 | 路人甲Java | | 2 | 男 | 張學友 | | 3 | 女 | 王祖賢 | | 4 | 男 | 郭富城 | | 5 | 女 | 李嘉欣 | +--------+--------+---------------+ 5 rows in set (0.00 sec)
2種用法。
相似於java中的switch語句。
case 表達式 when 值1 then 結果1或者語句1(若是是語句須要加分號) when 值2 then 結果2或者語句2 ... else 結果n或者語句n end [case] (若是是放在begin end之間須要加case,若是在select後則不須要)
查詢
t_user
表數據,返回:編號、性別(男、女)、姓名。
/*寫法1:相似於java中的if else*/ SELECT id 編號,(CASE sex WHEN 1 THEN '男' ELSE '女' END) 性別,name 姓名 FROM t_user; /*寫法2:相似於java中的if else if*/ SELECT id 編號,(CASE sex WHEN 1 then '男' WHEN 2 then '女' END) 性別,name 姓名 FROM t_user;
寫一個存儲過程,接受3個參數:id,性別(男、女),姓名,而後插入到t_user表
建立存儲過程:
/*刪除存儲過程proc1*/ DROP PROCEDURE IF EXISTS proc1; /*s刪除id=6的記錄*/ DELETE FROM t_user WHERE id=6; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程proc1*/ CREATE PROCEDURE proc1(id int,sex_str varchar(8),name varchar(16)) BEGIN /*聲明變量v_sex用於存放性別*/ DECLARE v_sex TINYINT UNSIGNED; /*根據sex_str的值來設置性別*/ CASE sex_str when '男' THEN SET v_sex = 1; WHEN '女' THEN SET v_sex = 2; END CASE ; /*插入數據*/ INSERT INTO t_user VALUES (id,v_sex,name); END $ /*結束符置爲;*/ DELIMITER ;
調用存儲過程:
CALL proc1(6,'男','郭富城');
查看效果:
mysql> select * from t_user; +----+-----+---------------+ | id | sex | name | +----+-----+---------------+ | 1 | 1 | 路人甲Java | | 2 | 1 | 張學友 | | 3 | 2 | 王祖賢 | | 4 | 1 | 郭富城 | | 5 | 2 | 李嘉欣 | | 6 | 1 | 郭富城 | +----+-----+---------------+ 6 rows in set (0.00 sec)
需求:寫一個函數,根據t_user表sex的值,返回男女
建立函數:
/*刪除存儲過程proc1*/ DROP FUNCTION IF EXISTS fun1; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程proc1*/ CREATE FUNCTION fun1(sex TINYINT UNSIGNED) RETURNS varchar(8) BEGIN /*聲明變量v_sex用於存放性別*/ DECLARE v_sex VARCHAR(8); CASE sex WHEN 1 THEN SET v_sex:='男'; ELSE SET v_sex:='女'; END CASE; RETURN v_sex; END $ /*結束符置爲;*/ DELIMITER ;
看一下效果:
mysql> select sex, fun1(sex) 性別,name FROM t_user; +-----+--------+---------------+ | sex | 性別 | name | +-----+--------+---------------+ | 1 | 男 | 路人甲Java | | 1 | 男 | 張學友 | | 2 | 女 | 王祖賢 | | 1 | 男 | 郭富城 | | 2 | 女 | 李嘉欣 | | 1 | 男 | 郭富城 | +-----+--------+---------------+ 6 rows in set (0.00 sec)
相似於java中多重if語句。
case when 條件1 then 結果1或者語句1(若是是語句須要加分號) when 條件2 then 結果2或者語句2 ... else 結果n或者語句n end [case] (若是是放在begin end之間須要加case,若是是在select後面case能夠省略)
這種寫法和1中的相似,你們用上面這種語法實現第1中用法中的3個示例,貼在留言中。
if結構相似於java中的 if..else if...else的語法,以下:
if 條件語句1 then 語句1; elseif 條件語句2 then 語句2; ... else 語句n; end if;
只能使用在begin end之間。
寫一個存儲過程,實現用戶數據的插入和新增,若是id存在,則修改,不存在則新增,並返回結果
/*刪除id=7的記錄*/ DELETE FROM t_user WHERE id=7; /*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc2; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc2(v_id int,v_sex varchar(8),v_name varchar(16),OUT result TINYINT) BEGIN DECLARE v_count TINYINT DEFAULT 0;/*用來保存user記錄的數量*/ /*根據v_id查詢數據放入v_count中*/ select count(id) into v_count from t_user where id = v_id; /*v_count>0表示數據存在,則修改,不然新增*/ if v_count>0 THEN BEGIN DECLARE lsex TINYINT; select if(lsex='男',1,2) into lsex; update t_user set sex = lsex,name = v_name where id = v_id; /*獲取update影響行數*/ select ROW_COUNT() INTO result; END; else BEGIN DECLARE lsex TINYINT; select if(lsex='男',1,2) into lsex; insert into t_user VALUES (v_id,lsex,v_name); select 0 into result; END; END IF; END $ /*結束符置爲;*/ DELIMITER ;
看效果:
mysql> SELECT * FROM t_user; +----+-----+---------------+ | id | sex | name | +----+-----+---------------+ | 1 | 1 | 路人甲Java | | 2 | 1 | 張學友 | | 3 | 2 | 王祖賢 | | 4 | 1 | 郭富城 | | 5 | 2 | 李嘉欣 | | 6 | 1 | 郭富城 | +----+-----+---------------+ 6 rows in set (0.00 sec) mysql> CALL proc2(7,'男','黎明',@result); Query OK, 1 row affected (0.00 sec) mysql> SELECT @result; +---------+ | @result | +---------+ | 0 | +---------+ 1 row in set (0.00 sec) mysql> SELECT * FROM t_user; +----+-----+---------------+ | id | sex | name | +----+-----+---------------+ | 1 | 1 | 路人甲Java | | 2 | 1 | 張學友 | | 3 | 2 | 王祖賢 | | 4 | 1 | 郭富城 | | 5 | 2 | 李嘉欣 | | 6 | 1 | 郭富城 | | 7 | 2 | 黎明 | +----+-----+---------------+ 7 rows in set (0.00 sec) mysql> CALL proc2(7,'男','梁朝偉',@result); Query OK, 1 row affected (0.00 sec) mysql> SELECT @result; +---------+ | @result | +---------+ | 1 | +---------+ 1 row in set (0.00 sec) mysql> SELECT * FROM t_user; +----+-----+---------------+ | id | sex | name | +----+-----+---------------+ | 1 | 1 | 路人甲Java | | 2 | 1 | 張學友 | | 3 | 2 | 王祖賢 | | 4 | 1 | 郭富城 | | 5 | 2 | 李嘉欣 | | 6 | 1 | 郭富城 | | 7 | 2 | 梁朝偉 | +----+-----+---------------+ 7 rows in set (0.00 sec)
對循環內部的流程進行控制,如:
相似於java中的
continue
iterate 循環標籤;
相似於java中的
break
leave 循環標籤;
下面咱們分別介紹3種循環的使用。
相似於java中的while循環。
[標籤:]while 循環條件 do 循環體 end while [標籤];
標籤:是給while取個名字,標籤和
iterate
、leave
結合用於在循環內部對循環進行控制:如:跳出循環、結束本次循環。注意:這個循環先判斷條件,條件成立以後,纔會執行循環體,每次執行都會先進行判斷。
根據傳入的參數v_count向test1表插入指定數量的數據。
/*刪除test1表記錄*/ DELETE FROM test1; /*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc3; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc3(v_count int) BEGIN DECLARE i int DEFAULT 1; a:WHILE i<=v_count DO INSERT into test1 values (i); SET i=i+1; END WHILE; END $ /*結束符置爲;*/ DELIMITER ;
見效果:
mysql> CALL proc3(5); Query OK, 1 row affected (0.01 sec) mysql> SELECT * from test1; +---+ | a | +---+ | 1 | | 2 | | 3 | | 4 | | 5 | +---+ 5 rows in set (0.00 sec)
根據傳入的參數v_count向test1表插入指定數量的數據,當插入超過10條,結束。
/*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc4; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc4(v_count int) BEGIN DECLARE i int DEFAULT 1; a:WHILE i<=v_count DO INSERT into test1 values (i); /*判斷i=10,離開循環a*/ IF i=10 THEN LEAVE a; END IF; SET i=i+1; END WHILE; END $ /*結束符置爲;*/ DELIMITER ;
見效果:
mysql> DELETE FROM test1; Query OK, 20 rows affected (0.00 sec) mysql> CALL proc4(20); Query OK, 1 row affected (0.02 sec) mysql> SELECT * from test1; +----+ | a | +----+ | 1 | | 2 | | 3 | | 4 | | 5 | | 6 | | 7 | | 8 | | 9 | | 10 | +----+ 10 rows in set (0.00 sec)
根據傳入的參數v_count向test1表插入指定數量的數據,只插入偶數數據。
/*刪除test1表記錄*/ DELETE FROM test1; /*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc5; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc5(v_count int) BEGIN DECLARE i int DEFAULT 0; a:WHILE i<=v_count DO SET i=i+1; /*若是i不爲偶數,跳過本次循環*/ IF i%2!=0 THEN ITERATE a; END IF; /*插入數據*/ INSERT into test1 values (i); END WHILE; END $ /*結束符置爲;*/ DELIMITER ;
見效果:
mysql> DELETE FROM test1; Query OK, 5 rows affected (0.00 sec) mysql> CALL proc5(10); Query OK, 1 row affected (0.01 sec) mysql> SELECT * from test1; +----+ | a | +----+ | 2 | | 4 | | 6 | | 8 | | 10 | +----+ 5 rows in set (0.00 sec)
test2表有2個字段(a,b),寫一個存儲過程(2個參數:v_a_count,v_b_count),使用雙重循環插入數據,數據條件:a的範圍[1,v_a_count]、b的範圍[1,v_b_count]全部偶數的組合。
/*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc8; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc8(v_a_count int,v_b_count int) BEGIN DECLARE v_a int DEFAULT 0; DECLARE v_b int DEFAULT 0; a:WHILE v_a<=v_a_count DO SET v_a=v_a+1; SET v_b=0; b:WHILE v_b<=v_b_count DO SET v_b=v_b+1; IF v_a%2!=0 THEN ITERATE a; END IF; IF v_b%2!=0 THEN ITERATE b; END IF; INSERT INTO test2 VALUES (v_a,v_b); END WHILE b; END WHILE a; END $ /*結束符置爲;*/ DELIMITER ;
代碼中故意將
ITERATE a;
放在內層循環中,主要讓你們看一下效果。
見效果:
mysql> DELETE FROM test2; Query OK, 6 rows affected (0.00 sec) mysql> CALL proc8(4,6); Query OK, 1 row affected (0.01 sec) mysql> SELECT * from test2; +---+---+ | a | b | +---+---+ | 2 | 2 | | 2 | 4 | | 2 | 6 | | 4 | 2 | | 4 | 4 | | 4 | 6 | +---+---+ 6 rows in set (0.00 sec)
[標籤:]repeat 循環體; until 結束循環的條件 end repeat [標籤];
repeat循環相似於java中的do...while循環,無論如何,循環都會先執行一次,而後再判斷結束循環的條件,不知足結束條件,循環體繼續執行。這塊和while不一樣,while是先判斷條件是否成立再執行循環體。
根據傳入的參數v_count向test1表插入指定數量的數據。
/*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc6; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc6(v_count int) BEGIN DECLARE i int DEFAULT 1; a:REPEAT INSERT into test1 values (i); SET i=i+1; UNTIL i>v_count END REPEAT; END $ /*結束符置爲;*/ DELIMITER ;
見效果:
mysql> DELETE FROM test1; Query OK, 1 row affected (0.00 sec) mysql> CALL proc6(5); Query OK, 1 row affected (0.01 sec) mysql> SELECT * from test1; +---+ | a | +---+ | 1 | | 2 | | 3 | | 4 | | 5 | +---+ 5 rows in set (0.00 sec)
repeat中iterate
和leave
用法和while中相似,這塊的示例算是給你們留的做業,寫好的發在留言區,謝謝。
[標籤:]loop 循環體; end loop [標籤];
loop至關於一個死循環,須要在循環體中使用
iterate
或者leave
來控制循環的執行。
根據傳入的參數v_count向test1表插入指定數量的數據。
/*刪除存儲過程*/ DROP PROCEDURE IF EXISTS proc7; /*聲明結束符爲$*/ DELIMITER $ /*建立存儲過程*/ CREATE PROCEDURE proc7(v_count int) BEGIN DECLARE i int DEFAULT 0; a:LOOP SET i=i+1; /*當i>v_count的時候退出循環*/ IF i>v_count THEN LEAVE a; END IF; INSERT into test1 values (i); END LOOP a; END $ /*結束符置爲;*/ DELIMITER ;
見效果:
mysql> DELETE FROM test1; Query OK, 5 rows affected (0.00 sec) mysql> CALL proc7(5); Query OK, 1 row affected (0.01 sec) mysql> SELECT * from test1; +---+ | a | +---+ | 1 | | 2 | | 3 | | 4 | | 5 | +---+ 5 rows in set (0.00 sec)
loop中iterate
和leave
用法和while中相似,這塊的示例算是給你們留的做業,寫好的發在留言區,謝謝。
leave
和iterate
,leave
相似於java中的break
能夠退出循環,iterate
相似於java中的continue能夠結束本次循環mysql系列大概有20多篇,喜歡的請關注一下,歡迎你們加我微信itsoku或者留言交流mysql相關技術!
原文出處:https://www.cnblogs.com/itsoku123/p/11640021.html