史上最简单的 MySQL 教程(四十三)「函数」

温馨提示:本系列博文已经同步到 GitHub,地址为「mysql-tutorial」,欢迎感兴趣的童鞋StarFork,纠错。

函数

函数,就是将一段代码封装到一个结构中,在需要执行该段代码的时候,直接调用该结构(函数)执行即可。此操作,实现了代码的复用。在 MySQL 中,函数有两种,分别为:系统函数和自定义函数。

系统函数

顾名思义,系统函数就是系统定义好的函数,在需要的时候,我们直接调用即可。

任何函数都有返回值(对于空函数,我们就认为其返回值为),而且在 MySQL 中任何有返回值的操作都是通过select来操作的,因此 MySQL 的函数调用就是通过select来实现的。

在 MySQL 中,字符是字符串操作中最常见的基本单位。此外,如果对中文的截取是按字节进行的话,很容易造成乱码的问题。

下面,我们介绍一些常见的、对字符进行操作的系统函数:首先,执行如下语句,定义一些变量,

set @cn = '你好世界';
set @en = 'hello world';
set @one = 'charies';
set @two = 'gavin';
set @three = 'Gavin';

select @cn, @en, @one, @two, @three;
1

然后,调用系统函数进行测试:

  • substring,截取字符串,单位为字符;
2

如上图所示,在 MySQL 中字符串的位置是从1开始,0含有特殊的意义。

  • char_length,获取字符长度;
  • length,获取字节长度;
3
  • instr,判断字符串中某个子串是否存在,存在则返回具体的位置,不存在则返回0
4
  • lpad,左填充,将字符串按照某个指定的填充方式,填充到指定(字符)长度;
5

如上图所示,在对@en进行填充的时候,填充结果为hellohellhello world,其中第二个hello并没有填充全,这是因为系统函数lpad的第二个参数限定的了变量@en的具体长度,如示例中我们设置其为20,而原@en的长度为11,因此只能向@en中在填充9个字符。

  • insert,找到目标位置,将指定长度的字符串替换为目标字符串;
6

如上图所示,insert函数并没有修改变量自身的值,只是对变量的值进行加工而已。

  • strcmp,比较字符串的大小;
7

如上图所示,在用strcmp函数对字符串进行比较的时候不区分大小写(默认校对集),并用0表示两个字符串相等;用-1表示第一个参数的字符串小于第二个参数的字符串;用1表示第一个参数的字符串大于第二个参数的字符串。

自定义函数

对于任意一个函数,其都包含如下要素:

  • 函数名;
  • 参数类别(可以为空);
  • 返回值;
  • 函数体(作用域)。

根据上面这些函数要素,我们就来尝试创建自定义函数。

创建函数

-- 基本语法
create function 函数名([参数列表]) returns 数据类型
begin
	-- 函数体
	-- 返回值,类型为 returns 指定的数据类型
end

如果我们定义的函数的函数体内仅含有返回值,则可以省略beginend。此外,自定义函数和系统函数的调用方式相同。执行如下语句,进行测试:

-- 自定义函数
create function showLove() returns int
return 521;
-- 调用自定义函数
select showLove();
8

查看函数

查看函数,基本语法为:

  • show function status + [like 'pattern'];
9

如上图所示,我们可以看到showLove函数是属于test数据库的,这引出了函数的一个性质,即函数是属于具体数据库的,在一个数据库定义的函数不能在其定义的数据库外使用,但是可以查看

查看函数的创建语句,基本语法为:

  • show create function + 函数名;
10

修改函数 & 删除函数

函数只能先删除后新增,不能修改。删除函数的基本语法为:

  • drop function + 函数名;

执行如下语句,进行测试:

-- 删除函数
drop function showLove;
-- 查看函数
show function status like 'showLove'\G;
11

函数参数

对于函数的参数,一共有两种,分别为形参和实参,其中,形参可以理解为定义函数时使用的参数,且形参必须指定数据类型;实参可以理解为在调用函数时传入的值或变量。因此,函数定义的具体形式应该为:

  • function 函数名(形参名字 形参类型) returns 返回数据类型

下面,我们定义一个函数,完成一个简单的需求,即求1到指定数值的和。代码如下:

delimiter $$
create function addAll(num int) returns int
begin
	-- 定义条件变量
	set @i = 1;
	set @res = 0;    -- 保存求和结果
	-- 循环求和
	while @i <= num do
		-- 任何变量想要修改值都必须使用 set 关键字,且 MySQL 中没有 += 或者 ++ 运算符
		set @res = @res + @i;
		-- 修改循环变量
		set @i = @i + 1;
	end while;
	-- 返回求和结果
	return @res;
end
$$
delimiter ;
12

如上图所示,函数已经定义成功。接下来,执行如下语句,进行测试:

-- 调用函数求和
select addAll(100);

-- 查询自定义变量 @res 和 @i
select @res, @i;
13

如上图所示,求和函数addAll已经正确执行。此外,我们发现在函数内部使用@符号定义的变量@res@i在函数外部也是可以查看使用的,这说明:使用@符号定义的变量为全局变量

变量作用域

在 MySQL 中,变量的作用域有两种,分别为全局和局部,其中,全局变量可以在任何地方使用;局部变量只能在函数内部使用。

  • 全局变量:使用set关键字定义,用@符号标识;
  • 局部变量:使用declare关键字声明,且所用的局部变量必须在函数体开始之前进行声明。

接下来,我们利用局部变量定义一个函数,完成一个简单的需求,即求1到指定数值的和,要求10的倍数不加。代码如下:

delimiter $$
create function addAll2(num int) returns int
begin
	-- 声明变量,包含循环变量和结果变量
	declare i int default 1;    -- 定义局部变量可以含有属性
	declare res int default 0;
	-- 循环求和
	mywhile:while i <= num do
		-- 条件判断
		if i % 10 = 0 then
			-- 修改循环变量
			set i = i + 1;
			-- 重新循环
			iterate mywhile;
		end if;
		-- 修改结果变量
		set res = res + i;
		-- 修改循环变量
		set i = i + 1;
	end while;
	-- 返回求和结果
	return res;
end
$$
delimiter ;
14

如上图所示,函数已经定义成功。接下来,执行如下语句,进行测试:

-- 调用函数求和
select addAll(100), addAll2(100);
15

如上图所示,函数已经正确执行。


温馨提示:符号[]括起来的内容,表示可选项;符号+,则表示连接的意思。


———— ☆☆☆ —— 返回 -> 史上最简单的 MySQL 教程 <- 目录 —— ☆☆☆ ————

原创声明,本文系作者授权云+社区-专栏发表,未经许可,不得转载。

如有侵权,请联系 zhuanlan_guanli@qq.com 删除。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏python3

python3--函数初识

函数能提高应用的模块性,和代码的重复利用率。已经知道python提高了许多内建函数,比如print(),len()等。但你也可以自己创建函数,这被叫做用户自定义...

801
来自专栏蜉蝣禅修之道

算法考试填数字问题

1612
来自专栏LEo的网络日志

python技巧分享(六)

2716
来自专栏老马说编程

计算机程序的思维逻辑 (8) - char的真正含义

看似简单的char 通过前两节,我们应该对字符和文本的编码和乱码有了一个清晰的认识,但前两节都是与编程语言无关的,我们还是不知道怎么在程序中处理字符和文本。 ...

1626
来自专栏V站

30个php操作redis常用方法代码例子

Redis的操作很多的,以前看到一个比较全的博客,但是现在找不到了。查个东西搜半天,下面整理一下PHP处理Redis的例子,个人觉得常用一些例子。下面的例子都是...

834
来自专栏重庆的技术分享区

关于eslint使用规则,和各种报错对应规则

1815
来自专栏宏伦工作室

全栈 - 4 Python 先学会基本语法

1837
来自专栏流浪猫的golang

go panic与recover分析及错误处理

error 是一种类型,表示错误状态的类型,如果没有错误则是nil。直白点将:error 类型就是描述错误的一种类型。

603
来自专栏C/C++基础

C++之IO格式控制

C语言中,我们可以通过函数printf和scanf进行格式化控制,而在C++中仍然包含了前者,但还提供了以下两种格式控制的方法: (1)使用流成员函数进行格...

521
来自专栏飞雪无情的博客

Go语言中new和make的区别

Go语言中new和make是内建的两个函数,主要用来创建分配类型内存。在我们定义生成变量的时候,可能会觉得有点迷惑,其实他们的规则很简单,下面我们就通过一些示例...

932

扫描关注云+社区