SQL Server中IF...ELSE基本写法为:IF条件后接单条语句可省略BEGIN...END,执行多条语句则必须用BEGIN...END包裹;ELSE须紧接END后,不可有空行或注释;嵌套不宜超3层,否则影响可读性。
IF...ELSE 在 SQL Server 存储过程中是过程控制的核心,但它不支持像编程语言那样的嵌套块作用域——每个 IF 或 ELSE 后面的单条语句无需 BEGIN...END,但一旦要执行多条语句,BEGIN...END 就必不可少,漏掉会导致语法错误或逻辑错位。
IF @status = 1 SELECT 'active'; UPDATE users SET last_login = GETDATE() —— 第二条语句永远无条件执行IF @status = 1 BEGIN SELECT 'active'; UPDATE users SET last_login = GETDATE() END
ELSE 必须紧跟在 END 后,中间不能有空行或注释(某些版本会报错)IF 套 IF 超过 3 层,可读性急剧下降MySQL 的 IF 是语句级结构,必须以 END IF 显式结束,且不允许在普通 SQL 语句中直接使用(比如不能在 SELECT 里写 IF @x > 0 THEN ...),只能出现在存储过程、函数或触发器体内。
IF() 函数(三元表达式)和过程式 IF 混用:SELECT IF(status=1, 'Y', 'N') FROM t 是函数,不能执行 DML;而存储过程里的 IF status = 1 THEN INSERT ... END IF 才能控制流程ELSEIF 的缩写 ELIF,必须写全 ELSEIF
IS NULL 和 = NULL:后者永远为 FALSE,要用 IS NULL 判断PREPARE/EXECUTE,且变量作用域仅限当前 BEGIN...END 块PostgreSQL 的标准 SQL 函数(如 CREATE FUNCTION ... RETURNS TABLE AS $$ ... $$ LANGUAGE SQL)不支持过程式控制流。真要分支逻辑,必须切换到 plpgsql 语言,并用 IF / ELSIF / ELSE / END IF 结构。
BEGIN 之前:DECLARE x INT := 0;,否则报错 ERROR: 42601: syntax error at or near "IF"
CASE 表达式只能返回值,不能执行语句;想在 SELECT 中做分支计算可用 CASE WHEN ... THEN ... ELSE ... END,但不能在里面写 INSERT
IF 支持布尔表达式,也支持 NOT FOUND 这类异常条件,比如 IF NOT FOUND THEN RAISE NOTICE 'no row'; END IF;
PERFORM 替代 SELECT 用于只执行不返回结果的查询,避免意外返回结果集导致调用端报错把 SQL Server 的存储过程迁到 MySQL 或 PostgreSQL 时,IF 相关代码几乎必然要重写,不是语法微调,而是模型差异。
SET ANSI_NULLS OFF 下 @x = NULL 可能为 TRUE;MySQL 和 PostgreSQL 默认严格遵循 SQL 标准,= NULL 永远 FALSEBOOLEAN,MySQL 用 TINYINT(1) 模拟,SQL Server 用 BIT,在 IF 条件里写 IF @flag = 1 在 PG 里可能得改成 IF flag IS TRUE
THROW 或 RAISERROR;MySQL 用 SIGNAL;PostgreSQL 用 RAISE EXCEPTION —— 分支中抛错后是否继续执行后续语句,各引擎策略也不一样真正麻烦的不是写对某一种数据库的 IF 分支,而是当业务规则变更时,要在三套语法里同步维护同一套条件逻辑。建议把核心判断抽成视图或标量函数,让存储过程只做“调度”,降低耦合。