如何在同一Oracle SQL脚本中声明变量并使用它?


133

我想编写可重用的代码,并且需要在开始时声明一些变量,然后在脚本中重用它们,例如:

DEFINE stupidvar = 'stupidvarcontent';

SELECT stupiddata
FROM stupidtable
WHERE stupidcolumn = &stupidvar;

如何声明变量并在随后的语句(如使用SQLDeveloper)中重用它。


尝试次数

  • 使用DECLARE部分和插入下面的SELECT语句BEGINEND;。使用来访问变量&stupidvar
  • 使用关键字DEFINE并访问变量。
  • 使用关键字VARIABLE并访问变量。

但是我在尝试期间遇到各种错误(未绑定变量,语法错误,预期的SELECT INTO…)。


2
请注意,@ APC接受的答案中的方法可以在没有PL / SQL的情况下使用,例如,根据您的问题在SQL Developer工作表中使用。只需在一行中声明变量(无分号),然后在exec行中设置其值(以分号结尾),然后选择语句即可。最后,以脚本(F5)而不是语句(F9)的形式运行它。
阿莫斯·M·卡彭特

Answers:


139

在SQL * Plus脚本中有几种声明变量的方法。

第一种是使用VAR声明绑定变量。为VAR分配值的机制是通过EXEC调用:

SQL> var name varchar2(20)
SQL> exec :name := 'SALES'

PL/SQL procedure successfully completed.

SQL> select * from dept
  2  where dname = :name
  3  /

    DEPTNO DNAME          LOC
---------- -------------- -------------
        30 SALES          CHICAGO

SQL>

当我们要调用具有OUT参数或函数的存储过程时,VAR特别有用。

另外,我们可以使用替代变量。这些对于交互模式非常有用:

SQL> accept p_dno prompt "Please enter Department number: " default 10
Please enter Department number: 20
SQL> select ename, sal
  2  from emp
  3  where deptno = &p_dno
  4  /
old   3: where deptno = &p_dno
new   3: where deptno = 20

ENAME             SAL
---------- ----------
CLARKE            800
ROBERTSON        2975
RIGBY            3000
KULASH           1100
GASPAROTTO       3000

SQL>

当我们编写一个调用其他脚本的脚本时,预先定义变量可能很有用。该代码段运行时并未提示我输入值:

SQL> def p_dno = 40
SQL> select ename, sal
  2  from emp
  3  where deptno = &p_dno
  4  /
old   3: where deptno = &p_dno
new   3: where deptno = 40

no rows selected

SQL>

最后是匿名PL / SQL块。如您所见,我们仍然可以交互地为声明的变量赋值:

SQL> set serveroutput on size unlimited
SQL> declare
  2      n pls_integer;
  3      l_sal number := 3500;
  4      l_dno number := &dno;
  5  begin
  6      select count(*)
  7      into n
  8      from emp
  9      where sal > l_sal
 10      and deptno = l_dno;
 11      dbms_output.put_line('top earners = '||to_char(n));
 12  end;
 13  /
Enter value for dno: 10
old   4:     l_dno number := &dno;
new   4:     l_dno number := 10;
top earners = 1

PL/SQL procedure successfully completed.

SQL>

6
除您使用术语“绑定变量”外,其他都很好。VAR声明创建绑定变量,而ACCEPT或DEFINE创建替换变量。
戴夫·科斯塔

1
是否可以串联变量+字符串?
Ecropolis 2014年

@Ecropolis-是的,默认情况下在SQL Plus使用期间。使用SET CONCAT定义将替换变量的名称与紧随变量名称之后的字母数字字符分隔开的字符。在PL / SQL或SQL中使用双管道|| 连接。
拉斯洛·卢戈西

如果SQL是一种标准语言,那么为什么很难找到一个适用于所有地方的规范引用呢?WTF ???
jww

1
@jww-SQL是一个标准,但是它并不总是指定确切的语法,因此不同的RDBMS产品可以实现不同的功能;日期算术就是一个很好的例子。同样,较早的数据库产品(如Oracle)经常在标准涵盖这些功能之前就引入了这些功能:例如,分层的CONNECT BY语法。但是在这种情况下,我们讨论的是SQL * Plus,它是一种客户端工具,因此ANSI标准始终未涉及。
APC

28

如果是char变量,请尝试使用双引号:

DEFINE stupidvar = "'stupidvarcontent'";

要么

DEFINE stupidvar = 'stupidvarcontent';

SELECT stupiddata  
FROM stupidtable  
WHERE stupidcolumn = '&stupidvar'

更新:

SQL*Plus: Release 10.2.0.1.0 - Production on Wed Aug 25 17:13:26 2010

Copyright (c) 1982, 2005, Oracle.  All rights reserved.

SQL> conn od/od@etalon
Connected.
SQL> define var = "'FL-208'";
SQL> select code from product where code = &var;
old   1: select code from product where code = &var
new   1: select code from product where code = 'FL-208'

CODE
---------------
FL-208

SQL> define var = 'FL-208';
SQL> select code from product where code = &var;
old   1: select code from product where code = &var
new   1: select code from product where code = FL-208
select code from product where code = FL-208
                                      *
ERROR at line 1:
ORA-06553: PLS-221: 'FL' is not a procedure or is undefined

感谢您的回答,但是如果将var包含在双引号中,则会得到ORA-01008: not all variables bound
bl4ckb0l7

1
当然!DEFINE num = 1; SELECT &num FROM dual;导致: ORA-01008: not all variables bound
bl4ckb0l7

@ bl4ckb0l7-我敢打赌,您正在SQL * Plus中尝试此操作。
Laszlo Lugosi

20

在PL / SQL v.10中

关键字clarify用于声明变量

DECLARE stupidvar varchar(20);

分配一个值,您可以在声明时设置它

DECLARE stupidvar varchar(20) := '12345678';

或在使用您的INTO语句的变量中选择内容,但是您需要将语句包装在BEGIN和中END,还需要确保仅返回单个值,并且不要忘记分号。

因此完整的声明如下:

DECLARE stupidvar varchar(20);
BEGIN
    SELECT stupid into stupidvar FROM stupiddata CC 
    WHERE stupidid = 2;
END;

您的变量仅在内部可用BEGINEND因此如果要使用多个变量,则必须进行多次BEGIN END包装

DECLARE stupidvar varchar(20);
BEGIN
    SELECT stupid into stupidvar FROM stupiddata CC 
    WHERE stupidid = 2;

    DECLARE evenmorestupidvar varchar(20);
    BEGIN
        SELECT evenmorestupid into evenmorestupidvar FROM evenmorestupiddata CCC 
        WHERE evenmorestupidid = 42;

        INSERT INTO newstupiddata (newstupidcolumn, newevenmorestupidstupidcolumn)
        SELECT stupidvar, evenmorestupidvar 
        FROM dual

    END;
END;

希望这可以节省您一些时间


7

如果要声明日期,然后在SQL Developer中使用它。

DEFINE PROPp_START_DT = TO_DATE('01-SEP-1999')

SELECT * 
FROM proposal 
WHERE prop_start_dt = &PROPp_START_DT

5

只想添加Matas的答案。也许很明显,但是我已经搜索了很长时间才能弄清楚该变量只能在BEGIN-END结构内部访问,因此,如果以后需要在某些代码中使用它,则需要将此代码放在BEGIN中-END块

请注意,这些块可以嵌套

DECLARE x NUMBER;
  BEGIN
    SELECT PK INTO x FROM table1 WHERE col1 = 'test';

    DECLARE y NUMBER;
    BEGIN
    SELECT PK INTO y FROM table2 WHERE col2 = x;

    INSERT INTO table2 (col1, col2)
      SELECT y,'text'
      FROM dual
      WHERE exists(SELECT * FROM table2);
    COMMIT;
  END;
END;

5

问题是在脚本中使用变量对我来说意味着它将在SQL * Plus中使用。

问题是您错过了引号,Oracle无法将值解析为数字。

SQL> DEFINE num = 2018
SQL> SELECT &num AS your_num FROM dual;
old   1: SELECT &num AS your_num FROM dual
new   1: SELECT 2018 AS your_num FROM dual

  YOUR_NUM
----------
      2018

Elapsed: 00:00:00.01

由于自动类型转换(或任何所谓的),该示例工作正常。

如果通过在SQL * Plus中键入DEFINE进行检查,它将显示num变量为CHAR。

SQL>define
DEFINE NUM             = "2018" (CHAR)

在这种情况下这不是问题,因为Oracle可以将字符串解析为数字(如果它是有效数字)。

当字符串无法解析为数字时,则Oracle无法对其进行处理。

SQL> DEFINE num = 'Doh'
SQL> SELECT &num AS your_num FROM dual;
old   1: SELECT &num AS your_num FROM dual
new   1: SELECT Doh AS your_num FROM dual
SELECT Doh AS your_num FROM dual
       *
ERROR at line 1:
ORA-00904: "DOH": invalid identifier

用引号引起来,所以不要强迫Oracle将其解析为数字,可以:

17:31:00 SQL> SELECT '&num' AS your_num FROM dual;
old   1: SELECT '&num' AS your_num FROM dual
new   1: SELECT 'Doh' AS your_num FROM dual

YOU
---
Doh

因此,要回答原始问题,应该像以下示例所示:

SQL> DEFINE stupidvar = 'X'
SQL>
SQL> SELECT 'print stupidvar:' || '&stupidvar'
  2  FROM dual
  3  WHERE dummy = '&stupidvar';
old   1: SELECT 'print stupidvar:' || '&stupidvar'
new   1: SELECT 'print stupidvar:' || 'X'
old   3: WHERE dummy = '&stupidvar'
new   3: WHERE dummy = 'X'

'PRINTSTUPIDVAR:'
-----------------
print stupidvar:X

Elapsed: 00:00:00.00

还有另一种方法可以通过使用以下方式在SQL * Plus中存储变量 Query Column Value

COL [UMN]具有NEW_VALUE从查询的字段名选项储存价值。

SQL> COLUMN stupid_column_name new_value stupid_var noprint
SQL> SELECT dummy || '.log' AS stupid_column_name
  2  FROM dual;

Elapsed: 00:00:00.00
SQL> SPOOL &stupid_var.
SQL> SELECT '&stupid_var' FROM DUAL;
old   1: SELECT '&stupid_var' FROM DUAL
new   1: SELECT 'X.log' FROM DUAL

X.LOG
-----
X.log

Elapsed: 00:00:00.00
SQL>SPOOL OFF;

如您所见,X.log值已设置到stupid_var变量中,因此我们可以在当前目录中找到一个X.log文件,其中包含一些日志。


2

这是您的答案:

DEFINE num := 1;       -- The semi-colon is needed for default values.
SELECT &num FROM dual;

1
和我一样。我使用ODT并运行:DEFINE num:= 1; SELECT NUM FROM dual; 我得到的是:ORA-00904:“ NUM”:无效的标识符00904。00000-“%s:无效的标识符” *原因:*操作:行错误:2列:8
toha

0

在Toad中,我可以使用以下方法:

declare 
    num number;
begin 
    ---- use 'select into' works 
    --select 123 into num from dual;

    ---- also can use :=
    num := 123;
    dbms_output.Put_line(num);
end;

然后,该值将打印到DBMS OutputWindow。

参考herehere2


0

有时您需要使用宏变量而不要求用户输入值。通常,这必须使用可选的脚本参数来完成。以下代码功能齐全

column 1 noprint new_value 1
select '' "1" from dual where 2!=2;
select nvl('&&1', 'VAH') "1" from dual;
column 1 clear
define 1

在rdbms / sql目录中找到了类似的代码。


0

如果您只需要指定一次参数并将其复制到多个位置,则一种可能的方法是执行以下操作:

SELECT
  str_size  /* my variable usage */
  , LPAD(TRUNC(DBMS_RANDOM.VALUE * POWER(10, str_size)), str_size, '0') rand
FROM
  dual  /* or any other table, or mixed of joined tables */
  CROSS JOIN (SELECT 8 str_size FROM dual);  /* my variable declaration */

此代码生成一个由8个随机数字组成的字符串。

注意,我创建了一种别名str_size,该别名包含常量8。它在查询中被交叉连接多次使用。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.