我有以下情况:
+-----------+
| my_column |
+-----------+
| A |
| B |
| C |
| D |
| E |
+-----------+
在此示例中,我简化了my_function波纹管;
DROP FUNCTION IF EXISTS my_function;
CREATE FUNCTION my_function(
phrase VARCHAR(255),
column_value VARCHAR(255)
)
RETURNS FLOAT(20,10)
READS SQL DATA
SQL SECURITY INVOKER
BEGIN
IF(column_value = 'A') THEN RETURN 1.0000000000;
ELSEIF(column_value = 'B') THEN RETURN 0.7500000000;
ELSEIF(column_value = 'C') THEN RETURN 0.7500000000;
ELSEIF(column_value = 'D') THEN RETURN 0.5000000000;
ELSEIF(column_value = 'E') THEN RETURN 0.0000000000;
END IF;
END;
这是我的主要存储过程:
DROP PROCEDURE IF EXISTS my_procedure;
CREATE PROCEDURE my_procedure(
IN phrase VARCHAR(255)
)
READS SQL DATA
SQL SECURITY INVOKER
BEGIN
SET @phrase = phrase;
SET @query = "
SELECT
my_column,
@score_var := my_function(?,my_column) as score,
@score_var
FROM my_table
ORDER BY score DESC;
";
PREPARE stmt FROM @query;
EXECUTE stmt USING @phrase;
DEALLOCATE PREPARE stmt;
END;
现在,如果我叫my_procedure
call my_procedure('anything');
结果是:
+-----------+--------------+------------+
| my_column | score | @score_var |
+-----------+--------------+------------+
| A | 1.0000000000 | 1 |
| B | 0.7500000000 | 0.75 |
| C | 0.7500000000 | 0.75 |
| D | 0.5000000000 | 0.5 |
| E | 0.0000000000 | 0 |
+-----------+--------------+------------+
但是,如果我在WHERE @score_var > 0.5
的查询中添加my_procedure
,结果将是:
+-----------+--------------+------------+
| my_column | score | @score_var |
+-----------+--------------+------------+
| A | 1.0000000000 | 1 |
| C | 0.7500000000 | 0.75 |
| E | 0.0000000000 | 0 |
+-----------+--------------+------------+
预期结果´> 0.5´:
+-----------+--------------+------------+
| my_column | score | @score_var |
+-----------+--------------+------------+
| A | 1.0000000000 | 1 |
| B | 0.7500000000 | 0.75 |
| C | 0.7500000000 | 0.75 |
+-----------+--------------+------------+
我已经看到一些使用子查询的答案,但是我的问题是:(在这种情况下)我可以不使用子查询吗?
也欢迎使用其他方法。
答案 0 :(得分:1)
在同一条语句中读写用户变量时,该行为记录为“未记录”。换句话说,除非您阅读并理解了MySQL版本的源代码,否则结果是不可预测的。
但是-我认为您在这里不必要地使事情复杂化。我看不到使用预处理语句的原因,也看不到使用u用户变量的原因。您的过程主体可能就是:
SELECT
my_column,
my_function(phrase, my_column) as score,
FROM my_table
HAVING score > 0.5
ORDER BY score DESC
也可以用更少的重复代码编写函数:
RETURN
CASE column_value
WHEN 'A' THEN 1.0000000000
WHEN 'B' THEN 0.7500000000
WHEN 'C' THEN 0.7500000000
WHEN 'D' THEN 0.5000000000
WHEN 'E' THEN 0.0000000000
END