zoukankan      html  css  js  c++  java
  • 自定义变量和语句结束分隔符

    有时候为了完成一个常用的功能需要执行许多条语句,每次都在客户端里一条一条的去输入这么多语句是很烦的,我们希望有一种批处理的形式,让我们以很简单的方式一次性的执行完这些语句,MySQL 中的存储程序本质上封装了一些可执行的语句,然后给用户提供一种简单的调用方式来执行这些语句。根据调用方式的不同,可以把存储程序分为存储例程、触发器和事件这几种类型。其中,存储例程又可以被细分为存储函数和存储过程。画个图表示一下:

    在正式介绍存储程序之前,需要先了解一下 MySQL 中的自定义变量和复合语句的概念。

    自定义变量

    变量是和常量相对的,一般的程序语言都提供对变量的支持,MySQL 中对自定义的变量的命名有个要求,那就是变量名称前必须加一个 @ 符号。自定义变量的值的类型可以是任意 MySQL 支持的类型,比方说来自定义一个变量:

    mysql> set @a = 1;
    Query OK, 0 rows affected (0.00 sec)
    

    自定义了一个名叫 a 的变量,并且把整数 1 赋值给这个变量。如果想查看这个变量的值的话,使用 SELECT 语句就好了,不过仍然需要在变量名称加一个 @ 符号:

    mysql> select @a;
    +------+
    | @a   |
    +------+
    |    1 |
    +------+
    1 row in set (0.00 sec)
    

    同一个变量也可以存储存储不同类型的值,比方说再把一个字符串值赋值给变量 a:

    mysql> set @a = '哈哈哈';
    Query OK, 0 rows affected (0.00 sec)
    

    除了把一个常量赋值给一个变量以外,还可以把一个变量赋值给另一个变量:

    mysql> set @b = @a;
    Query OK, 0 rows affected (0.00 sec)
    
    mysql> select @b;
    +-----------+
    | @b        |
    +-----------+
    | 哈哈哈    |
    +-----------+
    1 row in set (0.00 sec)
    

    这样变量 a 和 b 就有了相同的值 '哈哈哈'

    还可以将某个查询的结果赋值给一个变量,前提是这个查询的结果只有一个值:

    mysql> set @a = (select m1 from t1 limit 1);
    Query OK, 0 rows affected (0.00 sec)
    

    还可以用另一种形式的语句来将查询的结果赋值给一个变量:

    mysql> select n1 from t1 limit 1 into @b;
    Query OK, 1 row affected (0.00 sec)
    

    因为语句 select m1 from t1 limit 1select n1 from t1 limit 1 的查询结果都只有一个值,所以它们可以直接赋值给变量 a 或者 b。可以查看一下这两个变量的值:

    mysql> select @a, @b;
    +------+------+
    | @a   | @b   |
    +------+------+
    |    1 | a    |
    +------+------+
    1 row in set (0.00 sec)
    

    如果查询结果是一条记录,该记录中有多个列的值的话,想把这几个值分别赋值到不同的变量中,就只能使用 INTO 语句了:

    mysql> select m1, n1 from t1 limit 1 into @a, @b;
    Query OK, 1 row affected (0.00 sec)
    

    这条查询语句只得到一条记录,把这条记录的 m1 列的值赋值到了变量 a 中,m2 列的值赋值到了变量 b 中。

    语句结束分隔符

    在 MySQL 客户端的交互界面处,当完成键盘输入并按下回车键时,MySQL 客户端会检测输入的内容中是否包含 ;g 或者 G 这三个符号之一,如果有的话,会把输入的内容发送到服务器。这样一来,如果想给服务器发送复合语句(也就是由一条或多条语句组成的语句)的话,就需要把这些语句写到一行中,比如这样:

    mysql> SELECT * FROM t1 LIMIT 1; SELECT * FROM t2 LIMIT 1; SELECT * FROM t3 LIMIT 1;
    +------+------+
    | m1   | n1   |
    +------+------+
    |    1 | a    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m2   | n2   |
    +------+------+
    |    2 | b    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m3   | n3   |
    +------+------+
    |    3 | c    |
    +------+------+
    1 row in set (0.00 sec)
    

    造成这一不便的原因在于,MySQL 客户端检测输入结束用的符号和分隔各个语句的符号是一样的!其实也可以用 delimiter 命令来自定义 MySQL 的检测输入结束的符号,比如这样:

    mysql> delimiter $
    mysql> SELECT * FROM t1 LIMIT 1;
        -> SELECT * FROM t2 LIMIT 1;
        -> SELECT * FROM t3 LIMIT 1;
        -> $
    +------+------+
    | m1   | n1   |
    +------+------+
    |    1 | a    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m2   | n2   |
    +------+------+
    |    2 | b    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m3   | n3   |
    +------+------+
    |    3 | c    |
    +------+------+
    1 row in set (0.00 sec)
    

    delimiter $ 命令意味着修改 MySQL 客户端检测输入结束的符号为 $,所以虽然连续输入了 3 个以分号结尾的查询语句并且按了回车键,输入的内容并没有被提交,直到敲下 $ 符号并回车,MySQL 客户端才会将输入的内容提交到服务器,此时输入的内容里已经有 3 个独立的查询语句了,所以返回了 3 个结果集。

    可以使用任何符号来作为 MySQL 客户端检测输入结束的符号,也包括多个字符,比如这样:

    mysql> delimiter EOF
    mysql> SELECT * FROM t1 LIMIT 1;
        -> SELECT * FROM t2 LIMIT 1;
        -> SELECT * FROM t3 LIMIT 1;
        -> EOF
    +------+------+
    | m1   | n1   |
    +------+------+
    |    1 | a    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m2   | n2   |
    +------+------+
    |    2 | b    |
    +------+------+
    1 row in set (0.00 sec)
    
    +------+------+
    | m3   | n3   |
    +------+------+
    |    3 | c    |
    +------+------+
    1 row in set (0.00 sec)
    

    这里采用了 EOF 作为 MySQL 客户端检测输入结束的符号,当然,这个只是为了方便一次性输入多个语句,在输入完成之后最好还是改回常用的分号

    mysql> delimiter ;
    
  • 相关阅读:
    bzoj 1367
    codeforces 757F
    bzoj 3600
    比赛环境设置
    线段树合并
    BZOJ2105: 增强型LCP
    BZOJ3156: 防御准备
    BZOJ3252: 攻略
    BZOJ2464: 中山市选[2009]小明的游戏
    Beta Round #9 (酱油杯noi考后欢乐赛)乌鸦喝水
  • 原文地址:https://www.cnblogs.com/qiuxirufeng/p/12632657.html
Copyright © 2011-2022 走看看