最近、MySQL ストアド プロシージャと関数をいくつか作成しましたが、インターネット上に貴重なドキュメントがほとんどなく、そのほとんどがマニュアルからコピーされたものであり、変数をテーブル名として使用するなど、いくつかの実用的な問題が解決できないことがわかりました。
デバッグを繰り返した結果、最終的に解決策を見つけました。比較的断片化された簡単なレコードをいくつか示します。内容の一部は http://my.opera.com/Dereky/blog/show.dml/322997 から転載しています
1. 変数をテーブル名として使用します:
set または destroy ステートメントで変数を定義するだけです。 SQL として直接使用すると、テーブル名は機能しません。mysql は変数名をテーブル名として扱います。他の SQL データベースでも同じことが当てはまります。mssql の解決策は、SQL ステートメント全体を変数として使用し、テーブル名として変数を散在させ、sp_executesql を使用してステートメントを呼び出すことです。
これは mysql5.0 より前では不可能でした。5.0 以降では、sp_executesql と同様の関数を実現できる新しいステートメントが導入されました (プロシージャに対してのみ有効で、関数は動的クエリをサポートしません)。 EXECUTE stmt_name [USING @var_name [, @var_name] ...];
{DROP} PREPARE stmt_name
ここでいくつかの小さな例を示します。 'SELECT SQRT(POW(?,2) + POW(?,2)) AS 斜辺';
mysql> SET @b = 4; b;
+----------------+
|
+----------------+
| ---------- --+
mysql> DEALLOCATE PREPARE stmt1; SET @s = 'SELECT SQRT(POW(?,2) + POW(?,2)) AS 斜辺';
mysql> @s; から stmt2 を準備します;
mysql> SET @b = 8;
+ ---+
+ ------------+
mysql> を準備します。
MySQL バージョンが 5.0.7 以降の場合は、LIMIT 句でも使用できます。例は次のとおりです:
mysql> SET @a=1;mysql> PREPARE STMT FROM "SELECT * FROM tbl LIMIT ?" ;
mysql> @a を使用して STMT を実行 ;
mysql> SET @skip=1;
mysql> を使用して STMT を実行スキップする, @numrows;
PREPARE の使用 注意点:
A: PREPARE stmt_name FROM preparable_stmt; ステートメントを事前定義して stmt_name に割り当てる場合、大文字と小文字は区別されません。
B: preparable_stmt ステートメント内の ? が文字列を表す場合でも、? を引用符で囲む必要はありません。
C: 新しい PREPARE ステートメントが既存の stmt_name を使用する場合、元のステートメントはすぐに解放されます。この新しい PREPARE ステートメントがエラーにより正しく実行できない場合でも。
D: PREPARE stmt_name のスコープは、現在のクライアント接続セッションに表示されます。
E: 事前定義されたステートメントのリソースを解放するには、DEALLOCATE PREPARE 構文を使用できます。
F: EXECUTE stmt_name 構文では、stmt_name が存在しない場合、エラーがトリガーされます。
G: クライアント接続セッションを終了するときにリソースを解放するために DEALLOCATE PREPARE 構文が明示的に呼び出されない場合、サーバーは自動的にリソースを解放します。
H: 事前定義されたステートメントのうち、CREATE TABLE、DELETE、DO、INSERT、REPLACE、SELECT、SET、UPDATE、およびほとんどの SHOW 構文がサポートされています。
I: PREPARE ステートメントはストアド プロシージャやカスタム関数には使用できません。ただし、MySQL 5.0.13 以降では、ストアド プロシージャで使用できるようになりましたが、関数での使用はまだサポートされていません。
これが例です:
CREATE PROCEDURE `p1`(IN id INT UNSIGNED,IN name VARCHAR(11))
BEGIN lable_exit:
BEGIN
SET @SqlCmd = 'SELECT * FROM tA ';
IF id IS NOT NULL THEN
SET @SqlCmd = CONCAT(@SqlCmd, 'WHERE id=?');
SET @a = id;
LEAVE lable_exit;
name が NULL でない場合は、
SET @SqlCmd = CONCAT(@SqlCmd, 'WHERE name LIKE ?');
SET @a = CONCAT(name, '%');
EXECUTE stmt ;
lable_exit;
END lable_exit;
CALL `p1`(NULL,'QQ');理解する PREPARE の使用法を理解すると、変数をテーブル名として使用するのが簡単になります。ただし、実際の操作中に、変数の定義、宣言変数と set @var=value 変数の使用法、パラメーターとして渡される変数など、他のいくつかの問題が発見されました。
テストの結果、set @var=value のように定義された変数を文字列に直接記述すると、宣言された変数とパラメーターで渡された変数を CONCAT を使用して接続する必要があることがわかりました。具体的な原理は研究されていません。
EXECUTE stmt USING @a; このようなステートメントの USING 以降の変数は、宣言によって渡された変数のみを設定でき、パラメーターは使用できません。
さらに、PHP では、mysql ストアド プロシージャを呼び出すときに、PROCEDURE p が指定されたコンテキストで結果セットを返せないなどの問題が常に発生します。
http://www.bkjia.com/PHPjc/317913.html
www.bkjia.com
true
http://www.bkjia.com/PHPjc/317913.html
技術記事