自定义函数:
用户自定义函数(user-defined function,UDF)是一种对MySQL扩展的途经,其用法与内置函数相同。
自定义函数有两个必要条件:
1. 参数
2. 返回值
函数可以返回任意类型的值,同样可以接收这些类型的参数。
函数的参数和返回值之间没有必然的内在的联系。
示例1:
创建一个无参数的函数,返回当前时间的,年月日 时分秒。
例如:
mysql> SELECT NOW();
+---------------------+
| NOW() |
+---------------------+
| 2016-09-08 21:17:17 |
+---------------------+
1 row in set (0.00 sec)
mysql> SELECT DATE_FORMAT(NOW(),'%Y年%m月%d日 %H点:%i分:%s秒');
+--------------------------------------------------+
| DATE_FORMAT(NOW(),'%Y年%m月%d日 %H点:%i分:%s秒') |
+--------------------------------------------------+
| 2016年09月08日 21点:19分:54秒 |
+--------------------------------------------------+
1 row in set (0.01 sec)
对上述过程经行封装:
mysql> CREATE FUNCTION f1() RETURNS VARCHAR(30)
-> RETURN DATE_FORMAT(NOW(),'%Y年%m月%d日 %H点:%i分:%s秒');
Query OK, 0 rows affected (0.08 sec)
mysql> SELECT f1();
+-------------------------------+
| f1() |
+-------------------------------+
| 2016年09月08日 21点:21分:25秒 |
+-------------------------------+
1 row in set (0.02 sec)
注:returns后面的为返回值类型,return后面的为返回值。
示例2:
创建有参数的函数,例如求两个数的平均值。
mysql> CREATE FUNCTION f2(num1 SMALLINT UNSIGNED, num2 SMALLINT UNSIGNED)
-> RETURNS FLOAT(10,2) UNSIGNED
-> RETURN (num1+num2)/2;
Query OK, 0 rows affected (0.00 sec)
调用:
mysql> SELECT f2(4,6);
+---------+
| f2(4,6) |
+---------+
| 5.00 |
+---------+
1 row in set (0.02 sec)
如果要删除上面所创建的两个自定义函数:
mysql> DROP FUNCTION f2;
Query OK, 0 rows affected (0.11 sec)
mysql> DROP FUNCTION f1;
Query OK, 0 rows affected (0.00 sec)
创建具有复合结构函数体的自定义函数
例如:创建一个函数向表中插入数据,并返回所插入数据的id。
mysql> DESC tdb_test;
+------------+------------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+------------+------------------+------+-----+---------+----------------+
| id | int(10) unsigned | NO | PRI | NULL | auto_increment |
| first_name | varchar(20) | NO | | NULL | |
| last_name | varchar(20) | NO | | NULL | |
+------------+------------------+------+-----+---------+----------------+
3 rows in set (0.00 sec)
mysql> CREATE FUNCTION adduser(first_name VARCHAR(20),last_name VARCHAR(20))
-> RETURNS INT UNSIGNED
-> RETURN
-> INSERT tdb_test(first_name,last_name) VALUES(first_name,last_name);
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'tdb_test(first_name,last_name) VALUES(first_name,last_name)' at line 4
这时我们发现,在输入要执行的sql语句后,后面的分号为当前默认的分隔符,导致函数无法再继续输入。因此需要修改默认的分隔符。
mysql> DELIMITER //
意思是用//结束
mysql> CREATE FUNCTION adduser(first_name VARCHAR(20),last_name VARCHAR(20))
-> RETURNS INT UNSIGNED
-> RETURN
-> INSERT tdb_test(first_name,last_name) VALUES(first_name,last_name);
-> LAST_INSERT_ID();
-> //
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'tdb_test(first_name,last_name) VALUES(first_name,last_name);
LAST_INSERT_ID()' at line 4
同样返回错误,因为,返回有两个语句要执行,需放入begin和end构成一个聚合体,下面来看正确的做法。
mysql> CREATE FUNCTION adduser(first_name VARCHAR(20),last_name VARCHAR(20))
-> RETURNS INT UNSIGNED
-> BEGIN
-> INSERT tdb_test(first_name,last_name) VALUES(first_name,last_name);
-> RETURN LAST_INSERT_ID();
-> END
-> //
Query OK, 0 rows affected (0.00 sec)
再将分隔符改回来
mysql> DELIMITER ;
测试结果如下:
mysql> SELECT adduser("ttt","ddd");
+----------------------+
| adduser("ttt","ddd") |
+----------------------+
| 5 |
+----------------------+
1 row in set (0.11 sec)
mysql> SELECT * FROM tdb_test;
+----+------------+-----------+
| id | first_name | last_name |
+----+------------+-----------+
| 1 | A | B |
| 2 | Jack | Bob |
| 3 | tom% | 123 |
| 4 | 11 | 22 |
| 5 | ttt | ddd |
+----+------------+-----------+
5 rows in set (0.00 sec)
关于函数体
- 函数体由合法的sql语句构成;
- 函数体可以是简单的SELECT或INSERT语句;
- 函数体如果为复合结构则使用BEGIN…END语句;
- 复合结构可以包含声明,循环,控制结构;
删除函数:
DROP FUNCTION [IF EXISTS] function_name