在数据库操作中, 尤其是碰到一些复杂一些的系统, 不可避免的, 会用到函数/自定义函数, 或者存储过程.

实际项目中, 自定义函数和存储过程是越少越好, 因为这个东西多了, 也是一个非常难以维护的地方.

一、自定义函数

1. 例子

mysql提供的函数, 不在这一篇讲了, 这里主要贴一下自定义函数.  前台js插件里面有一个zTree, 不知道大家知不知道, 效果是这样的:

这种结构的数据, 在数据库中, 我一般会设计到一个表中

create table ztree
(
id int(11) not null PRIMARY key auto_increment,
name varchar(20) not null comment '节点名称',
pid int(11) not null comment '父节点id'
) comment '树形表';

然后插入数据:

如果你拿到一个节点A, 想要获取A下面的节点(不只是子节点哦), 那么通过一个自定义函数来做, 能方便许多.

delimiter $
DROP FUNCTIONIF EXISTS GetChildNodes ;
CREATE FUNCTION `GetChildNodes` (`rootId` INT) RETURNS VARCHAR (1000)
BEGIN
DECLARE res VARCHAR (1000) DEFAULT '-1';
DECLARE temp VARCHAR (1000) DEFAULT CAST(rootId AS CHAR);
-- SET res = '' ;
-- SET DECLARE = CAST(rootId AS CHAR) ;
WHILE temp IS NOT NULL DO SET res = CONCAT(res, ',', temp) ; SELECT GROUP_CONCAT(id) INTO temp FROM ztree WHERE FIND_IN_SET(pid, temp) > 0 ; END WHILE ;
RETURN res ;
END$
delimiter ;

这里, 我将res的值默认为-1, 这样的话, 就可以在查询的时候, 将这个结果拼入sql中, 还是比较方便的.

mysql中, 自定义函数的调用, 使用 select, 接下来, 就看一下之前的成果:

SELECT GetChildNodes (2);

2. 语法

自定义函数与存储过程有一个很明显的地方, 就是, 自定义函数是有返回值的, 并且需要通过return的方式返回. 而存储过程没有return返回值. 但是, 程序在执行存储过程的时候, 其实是可以得到一个结果集的.

语法:

  create function 函数名 (参数名 参数类型) returns 参数类型

  begin

  return result;

  end

1) 自定义函数传参与存储过程不同, 不需要指定 in/out.

2) 自定义函数可以用到别的sql语句中, 可以单独使用, 也可以混入别的sql中使用

二、存储过程

既然前面已经讲了自定义函数的语法, 那这里就先上存储过程的语法, 以便比较

1. 语法

  CREATE PROCEDURE 存储过程名称 (IN 参数名 参数类型, OUT 参数名 参数类型)

  begin

  end

这里的参数都是非必须的, 可以有 IN/OUT 都是可以没有的

从语法格式上看, 与自定义函数的框架大致是一样的, 只是其中的细节不同.

1) 存储过程没有return返回值, 但是却可以通过OUT的方式, 来修改传入的参数, 可以当做是一种返回值,

2) 存储过程在end之前, 可以加上一句 select语句, 以便程序读取到结果集. 所以存储过程能返回的值其实更多

3) 并不能混入别的sql中使用, 只能通过 call 的方式, 单独使用

2. 例子

之前的项目中, 我碰到一个生成流水号的功能. 当时, 我是通过借助数据库的方式, 来生成流水号的.

我这里的流水号, 由前缀, 时间, 流水码 三部分组成

先建一张流水号表

CREATE TABLE `serialno` (
`Id` int(11) NOT NULL AUTO_INCREMENT COMMENT 'Id',
`Pre` varchar(10) NOT NULL COMMENT '编号',
`Description` varchar(10) DEFAULT NULL COMMENT '说明',
`Res` varchar(20) DEFAULT NULL COMMENT '流水号(不加编号的)',
PRIMARY KEY (`Id`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8 COMMENT='流水号表'

有了这张表, 就可以开始存储过程了.

delimiter $
drop PROCEDURE if EXISTS p_GetSerialNo;
CREATE PROCEDURE `p_GetSerialNo`(IN preValue VARCHAR(10), IN preDate VARCHAR(10),IN des varchar(20), in length int)
BEGIN
DECLARE t_error INT DEFAULT 0;
DECLARE resValue VARCHAR(20) DEFAULT NULL;
-- DECLARE
-- CONTINUE HANDLER FOR SQLEXCEPTION,
-- SQLWARNING,
-- NOT FOUND
-- SET t_error = 1; START TRANSACTION; SELECT Res INTO resValue FROM serialno WHERE Pre=preValue;
IF resValue IS NULL THEN
SET resValue= CONCAT(preDate, LPAD(1, length, ''));
INSERT INTO serialno (Pre, Description, Res) VALUES (preValue, des, resValue);
ELSE
IF preDate = (SUBSTRING(resValue,1,8) + '') THEN
SET resValue = CAST(resValue AS SIGNED) + 1;
if preDate <> SUBSTRING(resValue,1,8) THEN
set t_error = -1;
end if;
ELSE
SET resValue= CONCAT(preDate, LPAD(1, length, ''));
END IF;
UPDATE serialno SET Res = resValue WHERE Pre = preValue;
END IF; #IF t_error = 1 then
IF @@error_count <> 0 | t_error <> 0 THEN
ROLLBACK;
select t_error;
ELSE
COMMIT;
SELECT CONCAT(preValue, resValue);
END IF;
END $
delimiter ;

这里的参数preValue为前缀, preDate为8位的日期,格式如:"20161227", 参数des为说明, 此处并不参与逻辑, 只是更新一个字段. 最后一个length字段, 表示流水号的位数. 流水号的位数不能设置的太过小, 得视业务来确定. 当流水号溢出时, 会返回-1.

OK, 来看一下效果:

call p_GetSerialNo( 'b', '',  'b', 4);

Mysql - 存储过程/自定义函数的更多相关文章

  1. mysql创建自定义函数与存储过程

    mysql创建自定义函数与存储过程 一 创建自定义函数 在使用mysql的过程中,mysql自带的函数可能不能完成我们的业务需求,这时就需要自定义函数,例如笔者在开发过程中遇到下面这个问题 mysql ...

  2. MYSQL存储过程和函数学习笔记

    学至Tarena金牌讲师,金色晨曦科技公司技术总监沙利穆课程笔记的综合. 1. 什么是存储过程和函数 将SQL语句放入一个集合里,然后直接调用存储过程和函数来执行已经定义好的SQL语句,通过存储过程和 ...

  3. Paip.断点调试MYSQL存储过程跟函数的解决方案大法

    Paip.断点调试MYSQL存储过程跟函数的解决方案大法 作者Attilax ,  EMAIL:1466519819@qq.com  来源:attilax的专栏 地址:http://blog.csdn ...

  4. Mysql存储过程和函数

    Mysql存储过程和函数 基本概念: 创建存储过程和函数是指将经常使用的一组SQL语句的组合在一起,并将这些SQL语句当作一个整体存储在MySQL服务器中.例如,银行经常需要计算用户的利息.不同类别的 ...

  5. navicat与phpmyadmin做mysql的自定义函数和事件

    自定义函数和事件是mysql一个很方便的功能,navicat在5.1以上版本就支持了自定义函数和事件,phpmyadmim不清楚. 用这个是由于一些简单的事情,没有必要去做一个服务器计划使用 接下来我 ...

  6. MySql存储过程与函数详解

    存储过程和函数是在数据库中定义一些SQL语句的集合,然后直接调用这些存储过程和函数来执行已经定义好的SQL语句.存储过程和函数可以避免开发人员重复的编写相同的SQL语句.而且,存储过程和函数是在MyS ...

  7. MYSql 存储过程自定义跳出

    MYSql存储过程自定义跳出 我们有时会在存储过程中进行一些判断,当判断条件达成时候我们有时会直接跳出存储过程. 但是存储过程不支持return直接返回的操作, 所以我们只能采用另一种方法,'leav ...

  8. 12.Mysql存储过程和函数

    12.存储过程和函数12.1 什么是存储过程和函数存储过程和函数是事先经过编译并存储在数据库中的一段SQL语句的集合,调用存储过程和函数简化应用开发人员的工作,减少数据在数据库和应用服务器之间的传输, ...

  9. mysql 存储过程,函数,触发器

    存储过程和函数 mysql> HELP CREATE PROCEDURE; Name: 'CREATE PROCEDURE' Description: Syntax: CREATE [DEFIN ...

随机推荐

  1. js学习笔记:操作iframe

    iframe可以说是比较老得话题了,而且网上也基本上在说少用iframe,其原因大致为:堵塞页面加载.安全问题.兼容性问题.搜索引擎抓取不到等等,不过相对于这些缺点,iframe的优点更牛,跨域请求. ...

  2. 梅须逊雪三分白,雪却输梅一段香——CSS动画与JavaScript动画

    CSS动画并不是绝对比JavaScript动画性能更优越,开源动画库Velocity.js等就展现了强劲的性能. 一.两者的主要区别 先开门见山的说说两者之间的区别. 1)CSS动画: 基于CSS的动 ...

  3. Js new到底发生了什么

    在Js中,我们使用了new关键字来进行实例化 那么在这个new的过程中到底发生了什么? 关于构造函数的return 正常来讲构造函数中是不用写return语句的,因为它会默认返回新创建的对象. 但是, ...

  4. ASP.NET MVC5+EF6+EasyUI 后台管理系统(72)-微信公众平台开发-消息处理

    系列目录 前言 Senparc.Weixin.MP SDK提供了MessageHandler消息处理类 在作者的Wiki中也详细说明了如何定义这个类,下面我们来演示,消息的回复,及效果 了解Messa ...

  5. [C#] 简单的 Helper 封装 -- SecurityHelper 安全助手:封装加密算法(MD5、SHA、HMAC、DES、RSA)

    using System; using System.IO; using System.Security.Cryptography; using System.Text; namespace Wen. ...

  6. 解决IE8下不兼容rgba()的解决办法

    rgba()是css3的新属性,所以IE8及以下浏览器不兼容,这怎么办呢?终于我找到了解决办法. 解决办法 我们先来解释以下rgba rgba: rgba的含义,r代表red,g代表green,b代表 ...

  7. 【干货分享】流程DEMO-制度发文和干部任免

    流程名: 制度发文和干部任免  业务描述: 当员工在该出勤的工作日出勤但漏打卡时,于一周内填写补打卡申请.  流程相关文件: 流程包.xml  流程说明: 直接导入流程包文件,即可使用本流程  表单: ...

  8. VMware下对虚拟机Ubuntu14系统所在分区sda1进行磁盘扩容

    VMware下对虚拟机Ubuntu14系统所在分区sda1进行磁盘扩容 一般来说,在对虚拟机里的Ubuntu下的磁盘进行扩容时,都是添加新的分区,而并不是对其系统所在分区进行扩容,如在此链接中http ...

  9. springMVC初始化绑定器

    单日期 在处理器类中配置绑定方法  使用@InitBinder注解 在这里首先注册一个用户编辑器 参数一为目标类型   propertyEditor为属性编辑器,此处我们选用 CustomDateEd ...

  10. 马哥linux运维初级+中级+高级 视频教程 教学视频 全套下载(近50G)

    马哥linux运维初级+中级+高级 视频教程 教学视频 全套下载(近50G)目录详情:18_02_ssl协议.openssl及创建私有CA18_03_OpenSSH服务及其相关应用09_01_磁盘及文 ...