Heim > Datenbank > MySQL-Tutorial > Hauptteil

Ein leistungsstarkes Tool zur Verbesserung der Datenbankverarbeitungsgeschwindigkeit – ausführliche Erläuterung der gespeicherten MySQL-Prozeduren

黄舟
Freigeben: 2017-02-22 11:21:04
Original
1628 Leute haben es durchsucht

Einführung in gespeicherte Prozeduren

Unsere häufig verwendeten SQL-Anweisungen in der Betriebsdatenbanksprache müssen zuerst kompiliert und dann ausgeführt werden, wenn sie ausgeführt werden. Gespeicherte Prozeduren (gespeicherte Prozeduren) sind eine Gruppe von Prozeduren, die zum Abschließen entwickelt wurden Spezifische Funktionen werden kompiliert und in der Datenbank gespeichert. Der Benutzer ruft die gespeicherte Prozedur auf und führt sie aus, indem er den Namen angibt und Parameter angibt (sofern die gespeicherte Prozedur über Parameter verfügt).

Eine gespeicherte Prozedur ist eine programmierbare Funktion, die erstellt und in der Datenbank gespeichert wird. Es kann aus SQL-Anweisungen und einigen speziellen Kontrollstrukturen bestehen. Gespeicherte Prozeduren sind nützlich, wenn Sie dieselbe Funktion auf verschiedenen Anwendungen oder Plattformen ausführen oder bestimmte Funktionen kapseln möchten. Gespeicherte Prozeduren in einer Datenbank können als Simulation des objektorientierten Ansatzes in der Programmierung angesehen werden. Es ermöglicht die Kontrolle darüber, wie auf Daten zugegriffen wird.

Gespeicherte Prozeduren haben normalerweise die folgenden Vorteile:

  1. Gespeicherte Prozeduren verbessern die Funktionalität und Flexibilität der SQL-Sprache. Gespeicherte Prozeduren können mithilfe von Flusskontrollanweisungen geschrieben werden, sind äußerst flexibel und können komplexe Beurteilungen und komplexere Operationen durchführen.

  2. Gespeicherte Prozeduren ermöglichen die Programmierung von Standardkomponenten. Nachdem eine gespeicherte Prozedur erstellt wurde, kann sie im Programm mehrmals aufgerufen werden, ohne dass die SQL-Anweisung der gespeicherten Prozedur neu geschrieben werden muss. Und Datenbankexperten können gespeicherte Prozeduren jederzeit ändern, ohne den Quellcode der Anwendung zu beeinträchtigen.

  3. Gespeicherte Prozeduren können eine schnellere Ausführungsgeschwindigkeit erreichen. Wenn ein Vorgang eine große Menge an Transaction-SQL-Code enthält oder mehrmals ausgeführt wird, wird die gespeicherte Prozedur viel schneller ausgeführt als die Stapelverarbeitung. Weil gespeicherte Prozeduren vorkompiliert sind. Wenn eine gespeicherte Prozedur zum ersten Mal ausgeführt wird, wird die Abfrage vom Optimierer analysiert und optimiert und schließlich wird ein Ausführungsplan in der Systemtabelle gespeichert. Die Batch-Transaction-SQL-Anweisung muss bei jeder Ausführung kompiliert und optimiert werden, und die Geschwindigkeit ist relativ langsamer.

  4. Gespeicherte Prozeduren können den Netzwerkverkehr reduzieren. Wenn bei Vorgängen für dasselbe Datenbankobjekt (z. B. Abfrage, Änderung) die an diesem Vorgang beteiligte Transaction-SQL-Anweisung in einer gespeicherten Prozedur organisiert ist, wird beim Aufruf der gespeicherten Prozedur auf dem Clientcomputer nur der Aufruf übertragen die Netzwerkanweisungen, wodurch der Netzwerkverkehr erheblich erhöht und die Netzwerklast verringert wird.

  5. Gespeicherte Prozeduren können vollständig als Sicherheitsmechanismus genutzt werden. Durch die Einschränkung der Berechtigungen zum Ausführen einer bestimmten gespeicherten Prozedur kann der Systemadministrator die Zugriffsberechtigungen für die entsprechenden Daten einschränken, den Zugriff unbefugter Benutzer auf die Daten verhindern und die Sicherheit der Daten gewährleisten.

Gespeicherte Prozeduren sind eine wichtige Funktion der Datenbankspeicherung, aber MySQL unterstützte gespeicherte Prozeduren vor 5.0 nicht, was die Anwendung von MySQL stark einschränkte. Glücklicherweise hat MySQL 5.0 endlich damit begonnen, gespeicherte Prozeduren zu unterstützen, was die Verarbeitungsgeschwindigkeit der Datenbank erheblich verbessern und auch die Flexibilität der Datenbankprogrammierung verbessern kann.

Gespeicherte Prozedur erstellen

1. Format

Von MySQL gespeicherte Prozedur erstellt:

mysql> DELIMITER // 
mysql> CREATE PROCEDURE proc1(OUT s int) 
     -> BEGIN 
     -> SELECT COUNT(*) INTO s FROM user; 
     -> END 
     -> // 
mysql> DELIMITER ;
Nach dem Login kopieren

Hinweis:

  1. Was hier beachtet werden muss, ist DELIMITER // und DELIMITER; in zwei Sätzen bedeutet DELIMITER Trennzeichen, da MySQL standardmäßig „;“ als Trennzeichen verwendet Dann behandelt der Compiler die gespeicherte Prozedur als SQL-Anweisung und der Kompilierungsprozess der gespeicherten Prozedur meldet einen Fehler. Daher müssen Sie das Schlüsselwort DELIMITER verwenden, um das aktuelle Segmenttrennzeichen im Voraus zu deklarieren, damit MySQL es behandelt. ;“ als Code in der gespeicherten Prozedur. Diese Codes werden ausgeführt und das Trennzeichen muss nach der Verwendung wiederhergestellt werden.

  2. Die gespeicherte Prozedur kann je nach Bedarf Eingabe-, Ausgabe-, Eingabe- und Ausgabeparameter haben. Hier gibt es einen Ausgabeparameter. Der Typ ist int. Wenn mehrere Parameter vorhanden sind, verwenden Sie „. ,“ um sie zu trennen.

  3. Der Anfang und das Ende des Prozesskörpers werden mit BEGIN und END markiert.

Auf diese Weise ist eine unserer gespeicherten MySQL-Prozeduren nicht einfach. Es macht nichts, wenn Sie es nicht verstehen. Als nächstes erklären wir es im Detail.

2. Deklarationstrennzeichen

Tatsächlich wurde die obige Anmerkung sehr klar geschrieben, aber Sie müssen ein wenig Aufmerksamkeit schenken Ja: Wenn Sie das MySQL Administrator-Verwaltungstool verwenden, können Sie es direkt erstellen und müssen es nicht mehr deklarieren.

3. Parameter

Die Parameter von gespeicherten MySQL-Prozeduren werden in drei Parametertypen verwendet: IN, OUT, INOUT lautet wie folgt:

CREATE PROCEDURE([[IN |OUT |INOUT ] 参数名 数据类形...])
Nach dem Login kopieren

IN-Eingabeparameter: Gibt an, dass der Wert dieses Parameters beim Aufruf der gespeicherten Prozedur angegeben werden muss. Der während der gespeicherten Prozedur geänderte Wert dieses Parameters kann nicht zurückgegeben werden und ist der Standardwert Wert

OUT Ausgabeparameter: Dieser Wert kann innerhalb der gespeicherten Prozedur geändert und zurückgegeben werden

INOUT Eingabe- und Ausgabeparameter: werden beim Aufruf angegeben und können geändert und zurückgegeben werden

1) IN-Parameterbeispiel

Erstellung:

mysql > DELIMITER // 
mysql > CREATE PROCEDURE demo_in_parameter(IN p_in int) 
    -> BEGIN 
    -> SELECT p_in; 
    -> SET p_in=2; 
    -> SELECT p_in; 
    -> END; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

Ausführungsergebnis:

mysql > SET @p_in=1;
mysql > CALL demo_in_parameter(@p_in);
+------+
| p_in |
+------+
| 1 |
+------+

+------+
| p_in |
+------+
| 2 |
+------+

mysql> SELECT @p_in;
+-------+
| @p_in |
+-------+
| 1 |
+-------+
Nach dem Login kopieren

Wie aus dem Obigen ersichtlich ist, obwohl p_in während des geändert wird Bei einer gespeicherten Prozedur hat dies keinen Einfluss auf den Wert von @p_id.

2) Beispiel für einen OUT-Parameter

Erstellung:

mysql > DELIMITER // 
mysql > CREATE PROCEDURE demo_out_parameter(OUT p_out int) 
    -> BEGIN 
    -> SELECT p_out; 
    -> SET p_out=2; 
    -> SELECT p_out; 
    -> END; 
    -> // 
mysql > DELIMITER;
Nach dem Login kopieren

Ausführungsergebnis:

mysql > SET @p_out=1;
mysql > CALL sp_demo_out_parameter(@p_out);
+-------+
| p_out |
+-------+
| NULL |
+-------+

+-------+
| p_out |
+-------+
| 2 |
+-------+

mysql> SELECT @p_out;
+-------+
| p_out |
+-------+
| 2 |
+-------+
Nach dem Login kopieren

3) Beispiel für einen INOUT-Parameter

Erstellen:

 mysql > DELIMITER //
 mysql > CREATE PROCEDURE demo_inout_parameter(INOUT p_inout int)
     -> BEGIN
     -> SELECT p_inout;
     -> SET p_inout=2;
     -> SELECT p_inout;
     -> END;
     -> //
 mysql > DELIMITER;
Nach dem Login kopieren

Ausführungsergebnis:

mysql > SET @p_inout=1;
mysql > CALL demo_inout_parameter(@p_inout) ;
+---------+
| p_inout |
+---------+
| 1 |
+---------+

+---------+
| p_inout |
+---------+
| 2 |
+---------+

mysql > SELECT @p_inout;
+----------+
| @p_inout |
+----------+
| 2 |
+----------+
Nach dem Login kopieren

Variablen

1)变量定义

DECLARE variable_name [,variable_name...] datatype [DEFAULT value];
Nach dem Login kopieren

其中,datatype为MySQL的数据类型,如:int、 float、 date、varchar(length),例如:

DECLARE l_int int unsigned default 4000000; 
DECLARE l_numeric number(8,2) DEFAULT 9.95; 
DECLARE l_date date DEFAULT '1999-12-31'; 
DECLARE l_datetime datetime DEFAULT '1999-12-31 23:59:59'; 
DECLARE l_varchar varchar(255) DEFAULT 'This will not be padded';
Nach dem Login kopieren

2)变量赋值

SET 变量名 = 表达式值 [,variable_name = expression ...]
Nach dem Login kopieren

3)用户变量

在MySQL客户端使用用户变量:

mysql > SELECT 'Hello World' into @x;
mysql > SELECT @x;
+-------------+
| @x |
+-------------+
| Hello World |
+-------------+
mysql > SET @y='Goodbye Cruel World';
mysql > SELECT @y;
+---------------------+
| @y |
+---------------------+
| Goodbye Cruel World |
+---------------------+

mysql > SET @z=1+2+3;
mysql > SELECT @z;
+------+
| @z |
+------+
| 6 |
+------+
Nach dem Login kopieren

在存储过程中使用用户变量:

mysql > CREATE PROCEDURE GreetWorld( ) SELECT CONCAT(@greeting,' World');
mysql > SET @greeting='Hello';
mysql > CALL GreetWorld( );
+----------------------------+
| CONCAT(@greeting,' World') |
+----------------------------+
| Hello World |
+----------------------------+
Nach dem Login kopieren

在存储过程间传递全局范围的用户变量:

mysql> CREATE PROCEDURE p1() SET @last_procedure='p1';
mysql> CREATE PROCEDURE p2() SELECT CONCAT('Last procedure was ',@last_proc);
mysql> CALL p1( );
mysql> CALL p2( );
+-----------------------------------------------+
| CONCAT('Last procedure was ',@last_proc |
+-----------------------------------------------+
| Last procedure was p1 |
+-----------------------------------------------+
Nach dem Login kopieren

注意:用户变量名一般以@开头,滥用用户变量会导致程序难以理解及管理

5、注释

MySQL存储过程可使用两种风格的注释

  • 双模杠:--(该风格一般用于单行注释)

  • C语言风格: 一般用于多行注释

例如:

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc1 --name存储过程名 
    -> (IN parameter1 INTEGER) 
    -> BEGIN 
    -> DECLARE variable1 CHAR(10); 
    -> IF parameter1 = 17 THEN 
    -> SET variable1 = 'birds'; 
    -> ELSE 
    -> SET variable1 = 'beasts'; 
    -> END IF; 
    -> INSERT INTO table1 VALUES (variable1); 
    -> END 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

调用存储过程

用call和你过程名以及一个括号,括号里面根据需要,加入参数,参数包括输入参数、输出参数、输入输出参数。具体的调用方法可以参看上面的例子。

查询存储过程

我们像知道一个数据库下面有那些表,我们一般采用show tables;进行查看。那么我们要查看某个数据库下面的存储过程,是否也可以采用呢?答案是,我们可以查看某个数据库下面的存储过程,但是是令一钟方式。我们可以用

select name from mysql.proc where db=’数据库名’;
Nach dem Login kopieren

或者

select routine_name from information_schema.routines where routine_schema='数据库名';
Nach dem Login kopieren

或者

show procedure status where db='数据库名';
Nach dem Login kopieren

进行查询。

如果我们想知道,某个存储过程的详细,那我们又该怎么做呢?是不是也可以像操作表一样用describe 表名进行查看呢?

答案是:我们可以查看存储过程的详细,但是需要用另一种方法:

SHOW CREATE PROCEDURE 数据库.存储过程名;
Nach dem Login kopieren

就可以查看当前存储过程的详细。

修改存储过程

ALTER PROCEDURE
Nach dem Login kopieren

更改用CREATE PROCEDURE 建立的预先指定的存储过程,其不会影响相关存储过程或存储功能。

删除存储过程

删除一个存储过程比较简单,和删除表一样:

DROP PROCEDURE
Nach dem Login kopieren

从MySQL的表格中删除一个或多个存储过程。

存储过程的控制语句

1、变量作用域

内部的变量在其作用域范围内享有更高的优先权,当执行到end。变量时,内部变量消失,此时已经在其作用域外,变量不再可见了,应为在存储
过程外再也不能找到这个申明的变量,但是你可以通过out参数或者将其值指派给会话变量来保存其值。

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc3() 
    -> begin 
    -> declare x1 varchar(5) default 'outer'; 
    -> begin 
    -> declare x1 varchar(5) default 'inner'; 
    -> select x1; 
    -> end; 
    -> select x1; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

2、条件语句

if-then -else语句

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc2(IN parameter int) 
    -> begin 
    -> declare var int; 
    -> set var=parameter+1; 
    -> if var=0 then 
    -> insert into t values(17); 
    -> end if; 
    -> if parameter=0 then 
    -> update t set s1=s1+1; 
    -> else 
    -> update t set s1=s1+2; 
    -> end if; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

case语句:

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc3 (in parameter int) 
    -> begin 
    -> declare var int; 
    -> set var=parameter+1; 
    -> case var 
    -> when 0 then 
    -> insert into t values(17); 
    -> when 1 then 
    -> insert into t values(18); 
    -> else 
    -> insert into t values(19); 
    -> end case; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

3、循环语句

while ···· end while:

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc4() 
    -> begin 
    -> declare var int; 
    -> set var=0; 
    -> while var<6 do 
    -> insert into t values(var); 
    -> set var=var+1; 
    -> end while; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

repeat···· end repeat:

它在执行操作后检查结果,而while则是执行前进行检查。

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc5 () 
    -> begin 
    -> declare v int; 
    -> set v=0; 
    -> repeat 
    -> insert into t values(v); 
    -> set v=v+1; 
    -> until v>=5 
    -> end repeat; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

loop ·····end loop:

loop循环不需要初始条件,这点和while 循环相似,同时和repeat循环一样不需要结束条件, leave语句的意义是离开循环。

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc6 () 
    -> begin 
    -> declare v int; 
    -> set v=0; 
    -> LOOP_LABLE:loop 
    -> insert into t values(v); 
    -> set v=v+1; 
    -> if v >=5 then 
    -> leave LOOP_LABLE; 
    -> end if; 
    -> end loop; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

LABLES 标号:

标号可以用在begin repeat while 或者loop 语句前,语句标号只能在合法的语句前面使用。可以跳出循环,使运行指令达到复合语句的最后一步。

4、ITERATE迭代

ITERATE:通过引用复合语句的标号,来从新开始复合语句

mysql > DELIMITER // 
mysql > CREATE PROCEDURE proc10 () 
    -> begin 
    -> declare v int; 
    -> set v=0; 
    -> LOOP_LABLE:loop 
    -> if v=3 then 
    -> set v=v+1; 
    -> ITERATE LOOP_LABLE; 
    -> end if; 
    -> insert into t values(v); 
    -> set v=v+1; 
    -> if v>=5 then 
    -> leave LOOP_LABLE; 
    -> end if; 
    -> end loop; 
    -> end; 
    -> // 
mysql > DELIMITER ;
Nach dem Login kopieren

MySQL存储过程的基本函数

1、字符串类

CHARSET(str) //返回字串字符集
CONCAT (string2 [,... ]) //连接字串
INSTR (string ,substring ) //返回substring首次在string中出现的位置,不存在返回0
LCASE (string2 ) //转换成小写
LEFT (string2 ,length ) //从string2中的左边起取length个字符
LENGTH (string ) //string长度
LOAD_FILE (file_name ) //从文件读取内容
LOCATE (substring , string [,start_position ] ) 同INSTR,但可指定开始位置
LPAD (string2 ,length ,pad ) //重复用pad加在string开头,直到字串长度为length
LTRIM (string2 ) //去除前端空格
REPEAT (string2 ,count ) //重复count次
REPLACE (str ,search_str ,replace_str ) //在str中用replace_str替换search_str
RPAD (string2 ,length ,pad) //在str后用pad补充,直到长度为length
RTRIM (string2 ) //去除后端空格
STRCMP (string1 ,string2 ) //逐字符比较两字串大小
SUBSTRING (str , position [,length ]) //从str的position开始,取length个字符
Nach dem Login kopieren

注:mysql中处理字符串时,默认第一个字符下标为1,即参数position必须大于等于1

mysql> select substring(&#39;abcd&#39;,0,2);
+-----------------------+
| substring(&#39;abcd&#39;,0,2) |
+-----------------------+
| |
+-----------------------+
1 row in set (0.00 sec)

mysql> select substring(&#39;abcd&#39;,1,2);
+-----------------------+
| substring(&#39;abcd&#39;,1,2) |
+-----------------------+
| ab |
+-----------------------+
1 row in set (0.02 sec)
TRIM([[BOTH|LEADING|TRAILING] [padding] FROM]string2) //去除指定位置的指定字符
UCASE (string2 ) //转换成大写
RIGHT(string2,length) //取string2最后length个字符
SPACE(count) //生成count个空格
Nach dem Login kopieren

2、数学类

ABS (number2 ) //绝对值
BIN (decimal_number ) //十进制转二进制
CEILING (number2 ) //向上取整
CONV(number2,from_base,to_base) //进制转换
FLOOR (number2 ) //向下取整
FORMAT (number,decimal_places ) //保留小数位数
HEX (DecimalNumber ) //转十六进制
注:HEX()中可传入字符串,则返回其ASC-11码,如HEX(&#39;DEF&#39;)返回4142143
也可以传入十进制整数,返回其十六进制编码,如HEX(25)返回19
LEAST (number , number2 [,..]) //求最小值
MOD (numerator ,denominator ) //求余
POWER (number ,power ) //求指数
RAND([seed]) //随机数
ROUND (number [,decimals ]) //四舍五入,decimals为小数位数]
Nach dem Login kopieren

注:返回类型并非均为整数,如:

默认变为整形值

mysql> select round(1.23);
+-------------+
| round(1.23) |
+-------------+
| 1 |
+-------------+
1 row in set (0.00 sec)

mysql> select round(1.56);
+-------------+
| round(1.56) |
+-------------+
| 2 |
+-------------+
1 row in set (0.00 sec)
Nach dem Login kopieren

可以设定小数位数,返回浮点型数据

mysql> select round(1.567,2); 
+----------------+ 
| round(1.567,2) | 
+----------------+ 
| 1.57 | 
+----------------+ 
1 row in set (0.00 sec) 
SIGN (number2 ) //
Nach dem Login kopieren

3、日期时间类

ADDTIME (date2 ,time_interval ) //将time_interval加到date2
CONVERT_TZ (datetime2 ,fromTZ ,toTZ ) //转换时区
CURRENT_DATE ( ) //当前日期
CURRENT_TIME ( ) //当前时间
CURRENT_TIMESTAMP ( ) //当前时间戳
DATE (datetime ) //返回datetime的日期部分
DATE_ADD (date2 , INTERVAL d_value d_type ) //在date2中加上日期或时间
DATE_FORMAT (datetime ,FormatCodes ) //使用formatcodes格式显示datetime
DATE_SUB (date2 , INTERVAL d_value d_type ) //在date2上减去一个时间
DATEDIFF (date1 ,date2 ) //两个日期差
DAY (date ) //返回日期的天
DAYNAME (date ) //英文星期
DAYOFWEEK (date ) //星期(1-7) ,1为星期天
DAYOFYEAR (date ) //一年中的第几天
EXTRACT (interval_name FROM date ) //从date中提取日期的指定部分
MAKEDATE (year ,day ) //给出年及年中的第几天,生成日期串
MAKETIME (hour ,minute ,second ) //生成时间串
MONTHNAME (date ) //英文月份名
NOW ( ) //当前时间
SEC_TO_TIME (seconds ) //秒数转成时间
STR_TO_DATE (string ,format ) //字串转成时间,以format格式显示
TIMEDIFF (datetime1 ,datetime2 ) //两个时间差
TIME_TO_SEC (time ) //时间转秒数]
WEEK (date_time [,start_of_week ]) //第几周
YEAR (datetime ) //年份
DAYOFMONTH(datetime) //月的第几天
HOUR(datetime) //小时
LAST_DAY(date) //date的月的最后日期
MICROSECOND(datetime) //微秒
MONTH(datetime) //月
MINUTE(datetime) //分返回符号,正负或0
SQRT(number2) //开平方
Nach dem Login kopieren

以上就是提高数据库处理速度的利器——MySQL存储过程详解的内容,更多相关内容请关注PHP中文网(www.php.cn)!



Verwandte Etiketten:
Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage
Über uns Haftungsausschluss Sitemap
Chinesische PHP-Website:Online-PHP-Schulung für das Gemeinwohl,Helfen Sie PHP-Lernenden, sich schnell weiterzuentwickeln!