目录
一,存储过程介绍
二,存储过程的优缺点
优点
缺点
三,存储过程的创建与调用
3.1,存储过程中的常用语法及参数
3.2,存储过程的使用
(1),创建存储函数,向数据表中插入50条数据
(2),in输入参数的使用
(3),out参数的使用
(4),inout参数的使用
四,存储过程中的变量及使用细则
4.1,变量定义
4.2,变量赋值
4.3,用户变量的使用
4.4,存储过程的一些常用查看命令
五,常用的存储过程的控制语句
5.1,条件语句
5.2,循环语句
六,其他相关知识点
6.1,存储过程体
6.2,MySQL AlTER命令对表的灵活操作
6.2.1,删除,添加表字段及默认值
6.2.2,CHANGE与MODIFY对修改字段的作用
6.2.3,其他修改的使用
自MySQL 5.0 版本开始支持存储过程。存储过程(Stored Procedure)是一种在数据库中存储复杂程序,以便外部程序调用的一种数据库对象。
存储过程是为了完成特定功能的SQL语句集,经编译创建并保存在数据库中,用户可通过指定存储过程的名字并给定参数(需要时)来调用执行。
存储过程思想就是数据库 SQL 语言层面的代码封装与重用。类似于Java开发中封装工具类方便以后直接调用的作用。
声明语句结束符,可以自定义:
DELIMITER $$
声明存储过程:
CREATE PROCEDURE pro_name(IN num int)
存储过程开始和结束符号:
BEGIN .... END
变量赋值:
SET num=1
变量定义:
DECLARE num int unsigned default 100;
创建mysql存储过程、存储函数:
create procedure 存储过程名(参数)
存储过程体:
create function 存储函数名(参数)
参数解析:
MySQL存储过程的参数用在存储过程的定义,共有三种参数类型,IN,OUT,INOUT,形式如:
CREATEPROCEDURE 存储过程名([[IN |OUT |INOUT ] 参数名 数据类形...])
需要注意的是,这里的参数根据需求而定,如果不需要参数,亦可不填写!
下面依次根据实例对不同的情况进行演示:
首先准备一张my_datas表:
mysql> create table if not exists `my_datas`(
`id` int(20) not null auto_increment comment '数据id',
`name` varchar(30) default null comment '姓名',
`address` varchar(45) default null comment '地址',
`time` datetime default null comment '创建时间',
primary key(`id`)
)engine=innodb auto_increment=1 default charset=utf8mb4;
Query OK, 0 rows affected, 1 warning (0.33 sec)
查看结构是否正确:
mysql> show columns from `my_datas`;
+---------+-------------+------+-----+-------------------+-------------------+
| Field | Type | Null | Key | Default | Extra |
+---------+-------------+------+-----+-------------------+-------------------+
| id | int | NO | PRI | NULL | auto_increment |
| name | varchar(30) | YES | | NULL | |
| address | varchar(45) | YES | | NULL | |
| time | datetime | YES | | NULL | DEFAULT_GENERATED |
+---------+-------------+------+-----+-------------------+-------------------+
4 rows in set (0.00 sec)
mysql> delimiter // #定义结束符
mysql> drop procedure if exists addMyData;
-> create procedure addMyData() #创建一个存储过程,名为:addMyData
-> begin
-> declare num int;
-> set num =1;
-> while num <=50 #插入50条数据
-> do
-> insert into `my_datas`(id,name,address,time)
-> values(null,concat('数据_',num,'号'),concat('北京四 合院',round(rand()*10),'号'),concat(current_timestamp())); #concat函数拼接信息
-> set num =num +1;
-> end
-> while;
-> end //;
Query OK, 0 rows affected, 1 warning (0.03 sec)
mysql> delimiter; #将语句的结束符号恢复为分号
默认情况下,存储过程和默认数据库相关联,如果想指定存储过程创建在某个特定的数据库下,那么在过程名前面加数据库名做前缀。 在定义过程时,使用 DELIMITER //命令将语句的结束符号从分号 ; 临时改为两个//,使得过程体中使用的分号被直接传递到服务器,而不会被客户端解释。
调用存储函数,并查询插入结果
mysql> call addMyData();
Query OK, 1 row affected (0.58 sec)
mysql> select * from `my_datas`;
+----+--------------+----------------------+---------------------+
| id | name | address | time |
+----+--------------+----------------------+---------------------+
| 1 | 数据_1号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 2 | 数据_2号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 3 | 数据_3号 | 北京四合院4号 | 2022-08-24 14:21:17 |
| 4 | 数据_4号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 5 | 数据_5号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 6 | 数据_6号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 7 | 数据_7号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 8 | 数据_8号 | 北京四合院5号 | 2022-08-24 14:21:17 |
| 9 | 数据_9号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 10 | 数据_10号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 11 | 数据_11号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 12 | 数据_12号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 13 | 数据_13号 | 北京四合院6号 | 2022-08-24 14:21:17 |
| 14 | 数据_14号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 15 | 数据_15号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 16 | 数据_16号 | 北京四合院9号 | 2022-08-24 14:21:17 |
| 17 | 数据_17号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 18 | 数据_18号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 19 | 数据_19号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 20 | 数据_20号 | 北京四合院9号 | 2022-08-24 14:21:17 |
| 21 | 数据_21号 | 北京四合院2号 | 2022-08-24 14:21:17 |
| 22 | 数据_22号 | 北京四合院2号 | 2022-08-24 14:21:17 |
| 23 | 数据_23号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 24 | 数据_24号 | 北京四合院10号 | 2022-08-24 14:21:17 |
| 25 | 数据_25号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 26 | 数据_26号 | 北京四合院5号 | 2022-08-24 14:21:17 |
| 27 | 数据_27号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 28 | 数据_28号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 29 | 数据_29号 | 北京四合院10号 | 2022-08-24 14:21:17 |
| 30 | 数据_30号 | 北京四合院10号 | 2022-08-24 14:21:17 |
| 31 | 数据_31号 | 北京四合院0号 | 2022-08-24 14:21:17 |
| 32 | 数据_32号 | 北京四合院2号 | 2022-08-24 14:21:17 |
| 33 | 数据_33号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 34 | 数据_34号 | 北京四合院6号 | 2022-08-24 14:21:17 |
| 35 | 数据_35号 | 北京四合院5号 | 2022-08-24 14:21:17 |
| 36 | 数据_36号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 37 | 数据_37号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 38 | 数据_38号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 39 | 数据_39号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 40 | 数据_40号 | 北京四合院6号 | 2022-08-24 14:21:17 |
| 41 | 数据_41号 | 北京四合院9号 | 2022-08-24 14:21:17 |
| 42 | 数据_42号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 43 | 数据_43号 | 北京四合院9号 | 2022-08-24 14:21:17 |
| 44 | 数据_44号 | 北京四合院5号 | 2022-08-24 14:21:17 |
| 45 | 数据_45号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 46 | 数据_46号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 47 | 数据_47号 | 北京四合院1号 | 2022-08-24 14:21:17 |
| 48 | 数据_48号 | 北京四合院7号 | 2022-08-24 14:21:17 |
| 49 | 数据_49号 | 北京四合院10号 | 2022-08-24 14:21:17 |
| 50 | 数据_50号 | 北京四合院9号 | 2022-08-24 14:21:17 |
+----+--------------+----------------------+---------------------+
50 rows in set (0.00 sec)
这样,50条需要的数据就能快速插入完毕。但是局限性在于插入的数据是在定义存储过程中写死的,不够灵活。
为了方便灵活的插入/修改/删除/查询我们需要的数据,我们可以定义in 来输入参数,如下:
创建一个用域删除指定id信息的存储过程
mysql> delimiter !!
mysql> create procedure delete_data(in ids int) #定义一个输入的参数
-> begin
-> delete from `my_datas` where id=ids;
-> end !!
Query OK, 0 rows affected (0.05 sec)
mysql> delimiter ;
执行存储过程并查看数据库信息情况:
mysql> call delete_data(3); #输入指定参数
Query OK, 1 row affected (0.04 sec)
mysql> select * from `my_datas` limit 3;
+----+-------------+---------------------+---------------------+
| id | name | address | time |
+----+-------------+---------------------+---------------------+
| 1 | 数据_1号 | 北京四合院3号 | 2022-08-24 14:21:17 |
| 2 | 数据_2号 | 北京四合院8号 | 2022-08-24 14:21:17 |
| 4 | 数据_4号 | 北京四合院3号 | 2022-08-24 14:21:17 |
+----+-------------+---------------------+---------------------+
3 rows in set (0.00 sec)
在存储过程中设置了需要传参的变量ids,调用存储过程的时候,通过传参将3赋值给ids,然后进行存储过程里的SQL操作。
构建一个简单的存储过程:
mysql> create procedure sums(a int ,b int ,out c int)
-> begin
-> set c = a+b;
-> end //
Query OK, 0 rows affected (0.03 sec)
mysql> delimiter ;
进行验证:
mysql> call sums(11,25,@s);
Query OK, 0 rows affected (0.01 sec)
mysql> select @s; #正常输出c的值
+------+
| @s |
+------+
| 36 |
+------+
构建一个inout型的存储过程:
mysql> delimiter //
mysql> create procedure inout_test(inout test int)
-> begin
-> select test;
-> set test=100;
-> select test;
-> end
-> //
Query OK, 0 rows affected (0.07 sec)
mysql> delimiter ;
测试执行过程:
mysql> set @test=52
-> ;
Query OK, 0 rows affected (0.00 sec)
mysql> call inout_test(@test);
+------+
| test |
+------+
| 52 |
+------+
1 row in set (0.00 sec)
+------+
| test |
+------+
| 100 |
+------+
1 row in set (0.00 sec)
Query OK, 0 rows affected (0.00 sec)
#调用了inout_param存储过程,接受了输入的参数,也输出参数,改变了变量
mysql> select @test;
+-------+
| @test |
+-------+
| 100 |
+-------+
1 row in set (0.00 sec)
注意事项:参数的名字不要和列名相同,不然在过程体中,参数名会当作列名来处理,并且,存储过程命名尽量不要跟一些常用函数命名一样,否则sql检查会出错。
局部变量声明一定要放在存储过程体的开始:
DECLARE 变量名 数据类型(int,float,date……)
例如:
DECLARE numbers int
DECLARE l_datetime datetime DEFAULT '2022-09-31 16:12:35';
DECLARE l_varchar varchar(255) DEFAULT '黎治跃在内卷!';
SET 变量名 = 表达式值 [,variable_name = expression ...]
mysql > SELECT '黎治跃在内卷' into @l;
Query OK, 0 rows affected (0.00 sec)
mysql > SELECT @l;
+-------------+
| @l |
+-------------+
| 黎治跃在内卷 |
+-------------+
1 row in set (0.00 sec)
mysql> set @z='做个人吧,黎治跃';
Query OK, 0 rows affected (0.00 sec)
mysql> select @z;
+--------------------------+
| @z |
+--------------------------+
| 做个人吧,黎治跃 |
+--------------------------+
1 row in set (0.00 sec)
mysql> SET @y=5+2+52;
Query OK, 0 rows affected (0.00 sec)
mysql> select @y;
+------+
| @y |
+------+
| 59 |
+------+
1 row in set (0.00 sec)
存储过程中使用用户变量:
mysql> create procedure see() select concat(@lzy,'Java全能选手');
Query OK, 0 rows affected (0.02 sec)
mysql> set @lzy ='黎治跃:';
Query OK, 0 rows affected (0.00 sec)
mysql> call see();
+---------------------------------+
| concat(@lzy,'Java全能选手') |
+---------------------------------+
| 黎治跃:Java全能选手 |
+---------------------------------+
1 row in set (0.00 sec)
Query OK, 0 rows affected (0.00 sec)
在存储过程间传递全局范围的用户变量:
mysql> CREATE PROCEDURE p1() SET @last_procedure='l1';
Query OK, 0 rows affected (0.02 sec)
mysql> CREATE PROCEDURE p2() SELECT CONCAT('Last procedure was ',@last_procedure);
Query OK, 0 rows affected (0.03 sec)
mysql> CALL p1( );
mysql> CALL p2( );
+-----------------------------------------------+
| CONCAT('Last procedure was ',@last_proc |
+-----------------------------------------------+
| Last procedure was l1 |
+-----------------------------------------------+
1 row in set (0.00 sec)
Query OK, 0 rows affected (0.00 sec)
注意:
查看具体存储过程信息
show create procedure 存储过程名 \G
查看所有存储过程
show procedure status \G
模糊查询对应的存储过程信息
show procedure status like “模糊查询名” \G
查看指定表中的存储信息
select * from 表名 where routine_name = '存储过程名' \G
5.1.1,if-then-else
mysql > delimiter //
mysql > create prcedure test2(in s int)
-> begin
-> declare num int;
-> set num=s+1;
-> if num=0 then
-> insert into `new_table` values(555);
-> end if;
-> if s=0 then
-> update `new_table` set s1=s1+1;
-> else
-> update `new_table` set s1=s1+2;
-> end if;
-> end;
-> //
Query OK, 0 rows affected (0.07 sec)
mysql > delimiter ;
5.1.2,case语句
mysql > delimiter //
mysql > create procedure test(in sb int)
-> begin
-> declare num int;
-> set num=sb+1;
-> case num
-> when 0 then
-> insert into `new_table` values(23);
-> when 1 then
-> insert into `new_table` values(24);
-> else
-> insert into `new_table` values(25);
-> end case;
-> end;
-> //
Query OK, 0 rows affected (0.06 sec)
mysql > delimiter ;
5.2.1,while ……end while语句
mysql > delimiter //
mysql > create procedure test()
-> begin
-> declare num int;
-> set num=0;
-> while num<6 do
-> insert into `new_tables` values(num);
-> set num=num+1;
-> end while;
-> end;
-> //
Query OK, 0 rows affected (0.03 sec)
mysql > delimiter ;
5.2.2,repeat……end repeat语句
这个语句与while语句的不同之处在于while是先检查再执行,而repeat语句是执行操作后检查结果。
mysql > delimiter //
mysql > create procedure test()
-> begin
-> declare num int;
-> set num=0;
-> repeat
-> insert into `new_table` values(num);
-> set num=num+1;
-> until num>=5 #循环条件
-> end repeat;
-> end;
-> //
Query OK, 0 rows affected (0.04 sec)
mysql > delimiter ;
5.2.3,loop……end loop语句
loop循环相当于一个while True ...if ... break 循环,与repeat一循环不同,loop可以在循环体的任何位置通过leave离开循环,而repeat只能在循环体最后进行until判断 。此外loop还提供了循环标签,用于在嵌套·循环中标识不同层次的循环。
mysql > delimiter //
mysql > create procedure test5()
-> begin
-> declare num int;
-> set num=0;
-> LOOP1:loop
-> insert into `new_table` values(num);
-> set num=num+1;
-> if num >=5 then
-> leave LOOP1;
-> end if;
-> end loop;
-> end;
-> //
Query OK, 0 rows affected (0.04 sec)
mysql > delimiter ;
BEGIN BEGIN BEGIN statements; END END END
每个嵌套块及其中的每条语句,必须以分号结束,表示过程体结束的begin-end块(又叫做复合语句compound statement),则不需要分号。
为语句块贴标签:
label1: BEGIN
label2: BEGIN
label3: BEGIN
statements;
END label3 ;
END label2;
END label1
标签有两个作用:
删除表字段
ALTER TABLE 表名 DROP 字段名;添加表字段
ALTER TABLE 表名 ADD 字段名 字段数据类型;添加表字段默认值
ALTER TABLE 表名 ALTER 字段名 SET DEFAULT 默认值;删除表字段默认值
ALTER TABLE 表名 ALTER 字段名 DROP DEFAULT;
另外,如果需要将表字段插入指定的位置,可以使用MySQL提供的关键字 FIRST (设定位第一列), AFTER 字段名(设定位于某个字段之后)。使用show columns查看表结构变化:
mysql> show columns from user;
+----------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+----------+-------------+------+-----+---------+----------------+
| id | int | NO | PRI | NULL | auto_increment |
| username | varchar(30) | YES | | NULL | |
| address | varchar(45) | YES | | NULL | |
+----------+-------------+------+-----+---------+----------------+
3 rows in set (0.04 sec)
mysql> alter table user add time datetime; #添加一个time字段
Query OK, 0 rows affected (0.17 sec)
Records: 0 Duplicates: 0 Warnings: 0
#在address后面添加一个sex字段
mysql> alter table user add sex tinyint(1) after address;Query OK, 0 rows affected, 1 warning (0.41 sec)
Records: 0 Duplicates: 0 Warnings: 1
mysql> show columns from user;
+----------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+----------+-------------+------+-----+---------+----------------+
| id | int | NO | PRI | NULL | auto_increment |
| username | varchar(30) | YES | | NULL | |
| address | varchar(45) | YES | | NULL | |
| sex | tinyint(1) | YES | | NULL | |
| time | datetime | YES | | NULL | |
+----------+-------------+------+-----+---------+----------------+
5 rows in set (0.00 sec)
当需要修改字段类型或者字段名时,常常会使用到change与modify关键字
modify使用
alter table 表名 modify 字段名 字段属性(更改后)
modify主要用于更改数据字段范围,当遇到在数据库构建时,范围数据定义过小,或者范围数据定义过大浪费内存空间时,对字段属性的更改。
change使用
alter table 表名 change old字段名 new字段名 对应的字段属性
change关键字主要用于对字段名的更改,在语法上CHANGE 关键字之后,紧跟着的是你要修改的字段名,然后指定新字段名及类型。但是CHANGE又不仅仅可以更改字段名,它也可以同时修改指明后的字段属性,同时对字段名和字段属性进行修改。
#使用change仅修改字段名
mysql> alter table user change address address varchar(40);
Query OK, 499 rows affected (0.38 sec)
Records: 499 Duplicates: 0 Warnings: 0
#使用modify仅修改字段属性
mysql> alter table user modify address varchar(45);
Query OK, 0 rows affected (0.06 sec)
Records: 0 Duplicates: 0 Warnings: 0
#使用change同时修改字段名和字段属性
mysql> alter table user change address u_address varchar(40));
Query OK, 499 rows affected (0.26 sec)
Records: 499 Duplicates: 0 Warnings: 0
#查看更改后的表结构
mysql> show columns from user;
+-----------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-----------+-------------+------+-----+---------+----------------+
| id | int | NO | PRI | NULL | auto_increment |
| username | varchar(30) | YES | | NULL | |
| u_address | varchar(40) | YES | | NULL | |
| sex | tinyint(1) | YES | | NULL | |
| time | datetime | YES | | NULL | |
+-----------+-------------+------+-----+---------+----------------+
5 rows in set (0.01 sec)
修改指定表的数据引擎
alter table user engine=指定数据引擎
例如:ALTER TABLE testalter_tbl ENGINE = MYISAM;
如果对当前数据表信息不清楚的话可以通过SHOW TABLE STATUS命令进行查看。
例如:查看当前user表的信息
mysql> show table status like 'user' \G;
*************************** 1. row ***************************
Name: user
Engine: InnoDB
Version: 10
Row_format: Dynamic
Rows: 499
Avg_row_length: 131
Data_length: 65536
Max_data_length: 0
Index_length: 0
Data_free: 0
Auto_increment: 500
Create_time: 2022-08-24 17:32:27
Update_time: NULL
Check_time: NULL
Collation: utf8mb4_0900_ai_ci
Checksum: NULL
Create_options:
Comment:
1 row in set (0.03 sec)
ERROR:
No query specified
修改表名
alter table old表名 rename to new表名
此外,ALTER关键字的操作也不仅仅局限于操作表,在以后的索引,外键上也有很多作用。
学无止境,留下一笔吧~