• mysql 存储过程与存储函数


    1  存储过程

    存储过程就是一条或者多条SQL语句的集合,可以视为批文件。它可以定义批量插入的语句,也可以定义一个接收不同条件的SQL。

    1.1 存储过程语法

    1.1.1  创建

    CREATE  PROCEDURE `NewProc`(IN `contid` int,OUT `contNo` varchar(200))
    BEGIN
    SELECT CONT_NO into contNo from bs_cont where id= `contid` ; --分号要加
    END

    语法: CREATE PROCEDURE sp_name(定义输入输出参数) [ 存储特性 ]  BEGIN SQL语句;  END

    IN 表示输入参数,OUT表示输出参数,INOUT表示既可以输入也可以输出的参数。sp_name为存储过程的名字。

    如果此存储过程没有任何输入输出,其实就没什么意义了,但是sp_name()的括号不能省略。

    1.1.2 注意:

      1. 先定义名称,在定义类型  ( IN `id` int)。

      2. 给出参变量赋值要用 into

    1.1.3 查看刚才创建的存储过程。

       SHOW PROCEDURE STATUS LIKE 'g%';

    1.1.4 下面是调用存储过程。对于存储过程提供的临时变量而言,MySQL规定要加上@开头。

    1. #study 是当前数据库名称
    2. CALL study.findCont(1,@contNo);
    SELECT @contNo;
     
    1.1.5  相对于直接使用SQL 语句,在应用程序中直接调用存储过程(存储函数)有以下好处:

     

      (1)减少网络通信量。
    存储过程直接就在数据库服务器上跑,所有的数据访问都在数据库服务器内部进行,不需要传输数据到其它服务器,所以会减少一定的网络传输。
      (2)执行速度更快。
    有两个原因:首先,在存储过程创建的时候,数据库已经对其进行了一次解析和优化(编译)。其次,存储过程一旦执行,在内存中就会保留一份这个存储过程,这样下次再执行同样的存储过程时,可以从内存中直接调用。

      (3)可维护性:的存储过程有些时候比程序更容易维护,这是因为可以实时更新DB端的存储过程。  有些bug,直接改存储过程里的业务逻辑,就搞定了

      (4) 分布式工作。
    应用程序和数据库的编码工作可以分别独立进行,而不会相互压制。

      (5) 增强安全性:提高代码安全,防止 SQL注入。这一点sql语句也可以做到。

    1.1.6 缺点:
      1.SQL本身是一种结构化查询语言,我们不应该用存储过程处理复杂的业务逻辑——让SQL回归它“结构化查询语言”的功用。复杂的业务逻辑,还是交给代码去处理吧。 

      2.  开发调试复杂,存储过程的开发调试不能像一般的程序代码那样debug调试排除错。 
      3. 不支持群集,数据库服务器无法水平扩展,或者数据库的切割(水平或垂直切割)。数据库切割之后,存储过程并不清楚数据存储在哪个数据库中。

    2  存储函数

    存储函数与存储过程本质上是一样的,都是封装一系列SQL语句,简化调用。我们自己编写的存储函数可以像MySQL函数那样自由的被调用。

    2.1 存储函数语法

    2.1.1  创建 (类型 1 和类型2)

    CREATE FUNCTION getStuNameById(stuId INT)  -- 默认是IN,但是不能写上去。stuId视为输入的临时变量
    RETURNS VARCHAR(255)   -- 指明返回值类型
    RETURN  (SELECT name FROM t_student WHERE id = stuId); //  -- 指明SQL语句,并使用结束标记。注意分号位置
    CREATE DEFINER=`vx`@`%` FUNCTION `calInvoiceAmountForRe`(sourceType VARCHAR(20),sourceId BIGINT) RETURNS decimal(12,2)
    begin
    -- 定义两个变量
    declare invocieAmountRe decimal(13,2);  
    declare invocieAmountAr decimal(13,2); 
    -- 来自于实收的时候:检查sourece_id对应的实收已经开票的金额合计
    select  sum(a.amount_invoice) from fin_invoice_detail a inner join fin_invoice b
    on a.INVOICE_NO = b.INVOICE_NO
    where b.STATUS <> '5' and b.is_del ='0' and a.type is null 
    and a.SOURCE_ID = sourceId and a.SOURCE_TYPE = sourceType
    group by a.SOURCE_TYPE,a.SOURCE_ID into invocieAmountRe;
    
    
    --来自于实收的时候:检查sourece_id对应的实收已经开票的金额
    select  sum(a.amount_invoice) from fin_invoice_detail a inner join fin_invoice b
    on a.INVOICE_NO = b.INVOICE_NO
    where b.STATUS <> '5' and b.is_del ='0' and a.type is null 
    and a.SOURCE_ID in (select charge_detail_id from fin_received_charge_relate where id = sourceId )  and a.SOURCE_TYPE = 'AR'
    group by a.SOURCE_TYPE,a.SOURCE_ID into invocieAmountAr;
       
    return IFNULL(invocieAmountRe,0) +IFNULL(invocieAmountAr,0);
    end

    2.2.2 使用存储函数。

      SELECT getStuNameById(1);


    2.2.3 注意:

      1. 在RETURN 语句后面,有趣的是,分号在SQL语句的外面。如果不加分号,查询结果居然查询出两条记录,很奇怪。

      2. 给出参变量赋值要用 into。

    3  相对于直接使用SQL 语句,在应用程序中直接调用存储过程(存储函数)有以下好处:

      (1)减少网络通信量。
    存储过程直接就在数据库服务器上跑,所有的数据访问都在数据库服务器内部进行,不需要传输数据到其它服务器,所以会减少一定的网络传输。
      (2)执行速度更快。
    有两个原因:首先,在存储过程创建的时候,数据库已经对其进行了一次解析和优化(编译)。其次,存储过程一旦执行,在内存中就会保留一份这个存储过程,这样下次再执行同样的存储过程时,可以从内存中直接调用。

      (3)可维护性:的存储过程有些时候比程序更容易维护,这是因为可以实时更新DB端的存储过程。  有些bug,直接改存储过程里的业务逻辑,就搞定了

      (4) 分布式工作。
    应用程序和数据库的编码工作可以分别独立进行,而不会相互压制。

      (5) 增强安全性:提高代码安全,防止 SQL注入。这一点sql语句也可以做到。

    3.1 缺点:
      1.SQL本身是一种结构化查询语言,我们不应该用存储过程处理复杂的业务逻辑——让SQL回归它“结构化查询语言”的功用。复杂的业务逻辑,还是交给代码去处理吧。 

      2.  开发调试复杂,存储过程的开发调试不能像一般的程序代码那样debug调试排除错。 
      3. 不支持群集,数据库服务器无法水平扩展,或者数据库的切割(水平或垂直切割)。数据库切割之后,存储过程并不清楚数据存储在哪个数据库中。

    4. 存储函数和存储过程的区别

      (1)存储函数的限制比较多,例如不能用临时表,只能用表变量,而存储过程的限制较少,存储过程的实现功能要复杂些,而函数的实现功能针对性比较强。

      (2)返回值不同。存储函数必须有返回值(必须包含一条return语句  ),且仅返回一个结果值(一行一列);存储过程可以没有返回值(不允许包含return语句),但是能返回结果集(out,inout)(一行多列)。

      (3)调用时的不同。存储函数嵌入在SQL中使用,可以在select 存储函数名(变量值);存储过程通过call语句调用 call 存储过程名。

      (4)参数的不同。存储函数的参数类型类似于IN参数,没有类似于OUT和INOUT的参数(没有出参)。存储过程的参数类型有三种,IN、out和INOUT:

        a. in:数据只是从外部传入内部使用(值传递),可以是数值也可以是变量

        b. out:只允许过程内部使用(不用外部数据),给外部使用的(引用传递:外部的数据会被先清空才会进入到内部),只能是变量

        c. inout:外部可以在内部使用,内部修改的也可以给外部使用,典型的引用 传递,只能传递变量。

    https://blog.csdn.net/yanluandai1985/article/details/83656374   

    存储函数和存储过程

    https://blog.csdn.net/yanluandai1985/article/details/83689524   存储过程 语法

  • 相关阅读:
    .Net/C# 应用程序直接读取本地 Cookies 文件(WinXP SP2 调用 API: InternetGetCookie 无果)
    wininet.dll函数库:不会过期的cookie
    WinForm中TextBox控件循环自动滚动示例
    JScript中Date.getTime转.Net中的DateTime
    js gettime c# ticks
    mysql查看整库个表详情
    rds分区实践
    mysql5.7.21源码安装
    EXPLAIN详解
    C#基础温习(4):C#中string数组和list的相互转换
  • 原文地址:https://www.cnblogs.com/xiaowangbangzhu/p/13208727.html
Copyright © 2020-2023  润新知