搜尋
首頁資料庫mysql教程十分鐘帶你搞懂mysql儲存過程

這篇文章給大家分享學習mysql預存程序時的筆記,其咋哄詳細介紹了mysql儲存過程的相關知識,希望對大家有幫助。

十分鐘帶你搞懂mysql儲存過程

一、定義

       預存程序(Stored Procedure)是在大型資料庫系統中,一組為了完成特定功能的SQL 語句集,儲存在資料庫中,經過第一次編譯後呼叫不需要再次編譯,使用者透過指定預存程序的名字並給出參數(如果該預存程序帶有參數)來執行它。預存程序是資料庫中的重要物件。

二、預存程序的特性

    1、能完成較複雜的判斷與運算
    2、可程式行強,且靈活
    3、SQL編程的程式碼可重複使用
    4、執行的速度相對快一些
    5、減少網路之間的資料傳輸,節省開銷    

三、建立一個簡單的儲存流程

    1、建立一個簡單的儲存流程

   預存程序的簡單語法

create procedure 名称()
begin
.........
end

    2、建立一個簡單的預存程序

create procedure testa()
begin
    select * from users;
    select * from orders;
end;

    3、呼叫預存程序

call testa();

執行結果如圖(1)及圖(2 ):

                                 圖(1)# # 1# # # # # # # # # #             圖(2)

四、儲存過程的變數

    1、先透過一個簡單的例子來學習變數的宣告與賦值

create procedure test2()
begin
  -- 使用 declare语句声明一个变量
  declare username varchar(32) default '';
  -- 使用set语句给变量赋值
  set username='xiaoxiao';
  -- 将users表中id=1的名称赋值给username
  select name into username from users where id=1;
  -- 返回变量
  select username;
end;

    2、概括

        (1)、變數的宣告使用declare,一句declare只宣告一個變量,變數必須先宣告後使用;

        (2)、變數具有資料型別及長度,且與mysql的SQL資料型別一致,因此甚至還能製定預設值、字元集與排序規則等;

        (3)、變數可以透過set來賦值,也可以透過select into的方式賦值;

        (4)、變數需要傳回,可以使用select語句,如:select 變數名。
       

五、變數的作用域

    1.變數作用域說明:
        (1)、儲存過程中變數是有作用域的,作用範圍在begin和使用end塊之間,end結束變數的作用範圍即結束。

        (2)、需要多個區塊之間傳值,可以使用全域變量,即放在所有程式碼區塊之前

        (3)、傳參變數是全域的,可以在多個區塊之間起作用

    2、透過一個實例來驗證變數的作用域


         需求: 建立一個儲存過程,用來統計表users、orders表中行數數量和orders表中的最大金額和最小金額

create procedure test3()
begin
  begin
    declare userscount int default 0; -- 用户表中的数量
    declare ordercount int default 0; -- 订单表中的数量
    select count(*) into userscount from users;
    select count(*) into ordercount from orders;
    select userscount,ordercount; -- 返回用户表中的数量、订单表中的数量
  end;
  begin 
    declare maxmoney int default 0; -- 最大金额
    declare minmoney int default 0; -- 最小金额
    select max(money) into maxmoney from orders;
    select min(money) into minmoney from orders;
    select maxmoney,minmoney; -- 返回最金额、最小金额
   end;
end;

調用上述儲存過程,結果如圖(3)與圖表(4):

           

            

                                       (4)

    3、我將流程test(3)改為如下時:

## 

        declare userscount int default 0; -- 使用者表中的數量

        declare ordercount int default 0; -- 訂單表中的數量
 userscount from users;

        select count(*) into ordercount from orders;

        select userscount,ordercount; -- 返回使用者表中的數量、訂單表中的數量

# #      end;

      begin 

        declare maxmoney int default 0; -- 最大金額

# max (money) into maxmoney from orders;

        select min(money) into minmoney from orders;

        select userscount,ordercount,maxmoney,minmoney; -- 返回最金额、最小金额

       end;

    end;

再次调用call test3(); 会报错如图(5):

                               图(5)

    4、将userscount,ordercount改为全局变量,再次验证

    create procedure test3()

    begin

        declare userscount int default 0; -- 用户表中的数量

        declare ordercount int default 0; -- 订单表中的数量

        begin

            select count(*) into userscount from users;

            select count(*) into ordercount from orders;

            select userscount,ordercount; -- 返回用户表中的数量、订单表中的数量

      end;

      begin 

        declare maxmoney int default 0; -- 最大金额

        declare minmoney int default 0; -- 最小金额

        select max(money) into maxmoney from orders;

        select min(money) into minmoney from orders;

        select userscount,ordercount,maxmoney,minmoney; -- 返回最金额、最小金额

       end;

    end;

再次调用call test3(); 会报错如图(6)和图(7):

       

                    图(6)       

                                                             

                    图(7)

因此,存储过程中变量的作用域,作用范围在begin和end块之间,end结束变量的作用范围即结束
    

六、存储过程参数

  1、基本语法

create procedure 名称([IN|OUT|INOUT] 参数名 参数数据类型 )
begin
.........
end

 存储过程的参数类型有:IN,OUT,INOUT,下面分别介绍这个三种类型:
    
  2、存储过程的传出参数IN

     说明:        

        (1)、传入参数:类型为in,表示该参数的值必须在调用存储过程事指定,如果不显示指定为in,那么默认就是in类型。

        (2)、IN类型参数一般只用于传入,在调用过程中一般不作为修改和返回

        (3)、如果调用存储过程中需要修改和返回值,可以使用OUT类型参数

通过一个实例来演示:

需求:编写存储过程,传入id,根据id返回name

 create procedure test4(userId int)
    begin
            declare username varchar(32) default '';
            declare ordercount int default 0;
            select name into username from users where id=userId;
            select username;
    end;

运行如图(8)


    

                                                       图(8)
  
    
   3、存储过程的传出参数out

        需求:调用存储过程时,传入userId返回该用户的name

        create procedure test5(in userId int,out username varchar(32))

        begin

            select name into username from users where id=userId;

        end;

        调用及运行结果如图(9):

                                   图(9)
        
  概括:

        1、传出参数:在调用存储过程中,可以改变其值,并可返回;

        2、out是传出参数,不能用于传入参数值;

        3、调用存储过程时,out参数也需要指定,但必须是变量,不能是常量;

        4、如果既需要传入,同时又需要传出,则可以使用INOUT类型参数

    (3).存储过程的可变参数INOUT
    
        需求:调用存储过程时,传入userId和userName,即使传入,也是传出参数。

create procedure test6(inout userId int,inout username varchar(32))
begin
    set userId=2;
    set username='';
    select id,name into userId,username from users where id=userId;
end;

 调用及运行结果如图(10)


        

                                       图(10)
    概括:
        1、可变变量INOUT:调用时可传入值,在调用过程中,可修改其值,同时也可返回值;
        2、INOUT参数集合了IN和OUT类型的参数功能;
        3、INOUT调用时传入的是变量,而不是常量;

七、存储过程条件语句

   1、基本结构

   (1)、条件语句基本结构:

if() then...else...end if;

   (2)、多条件判断语句:

if() then...
elseif() then...
else ...
end if;

   2、实例
    实例1:编写存储过程,如果用户userId是偶数则返回username,否则返回userId

create procedure test7(in userId int)
begin
   declare username varchar(32) default '';
   if(userId%2=0)
   then 
      select name into username from users where id=userId;
      select username;
   else
      select userId;
      end if;
end;

    调用及运行结果如图(11)和图(12):

       

                        图(11)           

                                     

                               图(12)

    2、存储过程的多条件语句应用示例

        需求:根据用户传入的uid参数判断

        (1)、如果用户状态status为1,则给用户score加10分;

        (2)、 如果用户状态status为2,则给用户score加20分;

        (3)、 其他情况加30分

create procedure test8(in userid int)
begin
   declare my_status int default 0;
   select status into my_status from users where id=userid;
   if(my_status=1)
   then 
       update users set score=score+10 where id=userid;
    elseif(my_status=2)
    then 
       update users set score=score+20 where id=userid;
    else 
       update users set score=score+30 where id=userid;
    end if;
end;

调用程之前的users表的数据如图(13),调用 call test8(1); 及运行结果图(14):

           
                                   图(13)       

     

                                    图(14)
      

八、存储过程循环语句


    1、while语句

       (1)、while语句的基本结构

while(表达式) do 
   ......  
end while;

         (2)、示例
    需求:使用循环语句,向表test1(id)中插入10条连续的记录

create procedure test9()
begin
  declare i int default 0;
  while(i<10) do 
    begin 
        select i;
        set i=i+1;
        insert into test1(id) values(i);
     end;
  end while;
end;

    调用及运行结果结果如图(15)和图(16):


    

                                                                    图(15)

   

         图(16)


    2、repeat语句
    (1)、repeat语句基本的结构:

repeat...until...end repeat;

     (2)、示例

需求:给test1表中的id字段插入数据,从1到10

create procedure test10()
begin
    declare i int default 0;
    repeat 
    begin 
        select i;
        set i=i+1;
        insert into test1(id) values(i);
    end;
    until i>=10 -- 如果i>=10,则跳出循环
    end repeat;
end;

   调用及运行结果结果如图(17)和图(18)

   

    图(17)             

   

              图(18)
        
    概括:
        until判断返回逻辑真或者假,表达式可以是任意返回真或者假的表达式,只有当until语句为真是,循环结束。
        

九、存储过程游标的使用

    1、什么是游标
        游标是保存查询结果的临时区域
    2、示例
    需求:编写存储过程,使用游标,把users表中 id为偶数的记录逐一更新用户名
    

create procedure test11()
    begin
        declare stopflag int default 0;
        declare username VARCHAR(32);
        -- 创建一个游标变量,declare 变量名 cursor ...
        declare username_cur cursor for select name from users where id%2=0;
        -- 游标是保存查询结果的临时区域
        -- 游标变量username_cur保存了查询的临时结果,实际上就是结果集
        -- 当游标变量中保存的结果都查询一遍(遍历),到达结尾,将变量stopflag设置为1,用于循环中判断是否结束
        declare continue handler for not found set stopflag=1;

        open username_cur; -- 打卡游标
        fetch username_cur into username; -- 游标向前走一步,取出一条记录放到变量username中
        while(stopflag=0) do -- 如果游标还没有结尾,就继续
            begin 
                -- 在用户名前门拼接 &#39;_cur&#39; 字符串
                update users set name=CONCAT(username,&#39;_cur&#39;) where name=username;
                fetch username_cur into username;
            end;
        end while; -- 结束循环
        close username_cur; -- 关闭游标
    end;

调用结果如图(19):

                                     图(19)

十、自定义函数

    函数与存储过程最大的区别是函数必须有返回值,否则会报错
    
    1、一个简单的函数

create function getusername(userid int) returns varchar(32)
    reads sql data  -- 从数据库中读取数据,但不修改数据
    begin
        declare username varchar(32) default &#39;&#39;;
        select name into username from users where id=userid;
        return username;
    end;

    调用及运行结果如图(20):

                图(20)
    

    概括:

    1.创建函数使用create function 函数名(参数) returns 返回类型;

    2.函数体放在begin和end之间;

    3.returns指定函数的返回值;

    4.函数调用使用select getusername()。

2、示例

    需求:根据userid,获取accoutid,id,name组合成UUID作为用户的唯一标识

  create function getuuid(userid int) returns varchar(64)
    reads sql data  -- 从数据库中读取数据,但不修改数据
    begin
        declare uuid varchar(64) default &#39;&#39;;
        select concat(accontid,&#39;_&#39;,id,&#39;_&#39;,name) into uuid from users where id=userid;
        return uuid;
    end;

调用及运行结果如图(21)

             图(21)

   
十一、触发器

    1、什么是触发器

    触发器与函数、存储过程一样,触发器是一种对象,它能根据对表的操作时间,触发一些动作,这些动作可以是insert,update,delete等修改操作。

    2、示例1

(1)、需求:出于审计目的,当有人往表users插入一条记录时,把插入的userid,username,插入动作和操作时间记录下来。

create trigger tr_users_insert after insert on users
    for each row 
    begin 
        insert into oplog(userid,username,action,optime)
        values(NEW.id,NEW.name,&#39;insert&#39;,now());
    end;

    创建成功后,给uses表中插入一条记录:

insert into users(id,name,age,status,score,accontid)
    values(6,&#39;小周&#39;,23,1,&#39;60&#39;,&#39;10001&#39;);

  执行成功后,打开oplog表,可以看到oplog表中插入了一条记录如图(22)
    

                                                     图(22)
  

  (2)、总结

        1、创建触发器使用create trigger 触发器名
        2、什么时候触发?after insert on users,除了after还有before,是在对表操作之前(before)或者之后(after)触发动作的。
        3、对什么操作事件触发? after insert on users,操作事件包括insert,update,delete等修改操作;
        4、对什么表触发? after insert on users
        5、影响的范围?for each row

3、示例2

    需求:出于审计目的,当删除users表时,记录删除前该记录的主要字段值

create trigger tr_users_delete before delete on users
    for each row 
    begin 
        insert into oplog(userid,username,action,optime)
        values(OLD.id,OLD.name,&#39;delete&#39;,now());
    end;

    删除users表中的一条记录

delete from users where id=6;

    执行成功后,打开oplog表,可以看到oplog表中插入了一条记录如图(23)

                                              图(23)

         
十二、流程控制

 1、case分支

   (1)、基本语法结构

case ...
when ... then....
when.... then....
else ... 
end case;

(2)、示例

users表中,根据userid获取status值,如果status为1,则修改score为10;如果status为2,则修改为20,如果status3,则修改为30;否则修改为40。

 create procedure testcate(userid int)
    begin 
        declare my_status int default 0;
        select status into my_status from users where id=userid;

        case my_status
            when 1 then update users set score=10 where id=userid;
            when 2 then update users set score=20 where id=userid;
            when 3 then update users set score=30 where id=userid;
            else update users set score=40 where id=userid;
        end case;
    end;

    调用过程 call testcate(1); ,执行结果如图(24);

                             图(24)

十四、存储过程+event(事件)

     1、使用存储过程+事件事件一个简单的实现福彩3D开奖
        
        需求:设计一个福彩的开奖过程,没3分钟开奖一次
            第一步:先编写一个存储过程open_lottery,产生3个随机数,生成一条开奖记录
            第二步:编写一个时间调度器,每3分钟调用一次这个过程
          

create procedure open_lottery()
        begin 
            insert into lottery(num1,num2,num3,ctime)
            select FLOOR(rand()*9)+1,FLOOR(rand()*9)+1,FLOOR(rand()*9)+1,now();
        end;
create event if not exists lottery_event -- 创建一个事件
        on schedule every  3 minute  -- on schedule 什么时候来执行,没三分钟执行一次
        on completion preserve 
        do call open_lottery;

        运行结果如图(25)


        

                                                         图(25)

注意,如果event之一没有运行,请按照以下办法解决:

(1)、 show variables like '%event_scheduler%';
        set global event_scheduler=on;

(2)、 alert event lottery_event enable;

    2、解析event的创建格式
    (1)、基本语法

create event[IF NOT EXISTS]event_name -- 创建使用create event
    ON SCHEDULE schedule -- on schedule 什么时候来执行
    [ON COMPLETION [NOT] PRESERVE] -- 调度计划执行完成后是否还保留
    [ENABLE | DISABLE] -- 是否开启事件,默认开启
    [COMMENT &#39;comment&#39;] -- 事件的注释
    DO sql_statement; -- 这个调度计划要做什么?

(2)、执行时间说明

    1.单次计划任务示例
        在2019年2月1日4点执行一次

        on schedule at    '2019-02-01 04:00:00'
         
    2. 重复计划执行
        on schedule every 1 second 每秒执行一次
        on schedule every 1 minute 每分钟执行一次
        on schedule every 1 day 没天执行一次
        
    3.指定时间范围的重复计划任务
        每天在20:00:00执行一次
        on schedule every 1 day starts '2019-02-01 20:00:00'

十五、本文所用到的表

1、lottery表

2、oplog表

3、orders表

4、test1表

5、user表

推荐学习:mysql视频教程

以上是十分鐘帶你搞懂mysql儲存過程的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:CSDN。如有侵權,請聯絡admin@php.cn刪除
MySQL:世界上最受歡迎的數據庫的簡介MySQL:世界上最受歡迎的數據庫的簡介Apr 12, 2025 am 12:18 AM

MySQL是一種開源的關係型數據庫管理系統,主要用於快速、可靠地存儲和檢索數據。其工作原理包括客戶端請求、查詢解析、執行查詢和返回結果。使用示例包括創建表、插入和查詢數據,以及高級功能如JOIN操作。常見錯誤涉及SQL語法、數據類型和權限問題,優化建議包括使用索引、優化查詢和分錶分區。

MySQL的重要性:數據存儲和管理MySQL的重要性:數據存儲和管理Apr 12, 2025 am 12:18 AM

MySQL是一個開源的關係型數據庫管理系統,適用於數據存儲、管理、查詢和安全。 1.它支持多種操作系統,廣泛應用於Web應用等領域。 2.通過客戶端-服務器架構和不同存儲引擎,MySQL高效處理數據。 3.基本用法包括創建數據庫和表,插入、查詢和更新數據。 4.高級用法涉及復雜查詢和存儲過程。 5.常見錯誤可通過EXPLAIN語句調試。 6.性能優化包括合理使用索引和優化查詢語句。

為什麼要使用mysql?利益和優勢為什麼要使用mysql?利益和優勢Apr 12, 2025 am 12:17 AM

選擇MySQL的原因是其性能、可靠性、易用性和社區支持。 1.MySQL提供高效的數據存儲和檢索功能,支持多種數據類型和高級查詢操作。 2.採用客戶端-服務器架構和多種存儲引擎,支持事務和查詢優化。 3.易於使用,支持多種操作系統和編程語言。 4.擁有強大的社區支持,提供豐富的資源和解決方案。

描述InnoDB鎖定機制(共享鎖,獨家鎖,意向鎖,記錄鎖,間隙鎖,下一鍵鎖)。描述InnoDB鎖定機制(共享鎖,獨家鎖,意向鎖,記錄鎖,間隙鎖,下一鍵鎖)。Apr 12, 2025 am 12:16 AM

InnoDB的鎖機制包括共享鎖、排他鎖、意向鎖、記錄鎖、間隙鎖和下一個鍵鎖。 1.共享鎖允許事務讀取數據而不阻止其他事務讀取。 2.排他鎖阻止其他事務讀取和修改數據。 3.意向鎖優化鎖效率。 4.記錄鎖鎖定索引記錄。 5.間隙鎖鎖定索引記錄間隙。 6.下一個鍵鎖是記錄鎖和間隙鎖的組合,確保數據一致性。

MySQL查詢性能差的常見原因是什麼?MySQL查詢性能差的常見原因是什麼?Apr 12, 2025 am 12:11 AM

MySQL查询性能不佳的原因主要包括没有使用索引、查询优化器选择错误的执行计划、表设计不合理、数据量过大和锁竞争。1.没有索引导致查询缓慢,添加索引后可显著提升性能。2.使用EXPLAIN命令可以分析查询计划,找出优化器错误。3.重构表结构和优化JOIN条件可改善表设计问题。4.数据量大时,采用分区和分表策略。5.高并发环境下,优化事务和锁策略可减少锁竞争。

您什麼時候應該使用複合索引與多個單列索引?您什麼時候應該使用複合索引與多個單列索引?Apr 11, 2025 am 12:06 AM

在數據庫優化中,應根據查詢需求選擇索引策略:1.當查詢涉及多個列且條件順序固定時,使用複合索引;2.當查詢涉及多個列但條件順序不固定時,使用多個單列索引。複合索引適用於優化多列查詢,單列索引則適合單列查詢。

如何識別和優化MySQL中的慢速查詢? (慢查詢日誌,performance_schema)如何識別和優化MySQL中的慢速查詢? (慢查詢日誌,performance_schema)Apr 10, 2025 am 09:36 AM

要優化MySQL慢查詢,需使用slowquerylog和performance_schema:1.啟用slowquerylog並設置閾值,記錄慢查詢;2.利用performance_schema分析查詢執行細節,找出性能瓶頸並優化。

MySQL和SQL:開發人員的基本技能MySQL和SQL:開發人員的基本技能Apr 10, 2025 am 09:30 AM

MySQL和SQL是開發者必備技能。 1.MySQL是開源的關係型數據庫管理系統,SQL是用於管理和操作數據庫的標準語言。 2.MySQL通過高效的數據存儲和檢索功能支持多種存儲引擎,SQL通過簡單語句完成複雜數據操作。 3.使用示例包括基本查詢和高級查詢,如按條件過濾和排序。 4.常見錯誤包括語法錯誤和性能問題,可通過檢查SQL語句和使用EXPLAIN命令優化。 5.性能優化技巧包括使用索引、避免全表掃描、優化JOIN操作和提升代碼可讀性。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

WebStorm Mac版

WebStorm Mac版

好用的JavaScript開發工具

MantisBT

MantisBT

Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器