Как объявить переменную и использовать ее в том же сценарии Oracle SQL?

133

Я хочу написать повторно используемый код, и мне нужно объявить некоторые переменные в начале и повторно использовать их в скрипте, например:

DEFINE stupidvar = 'stupidvarcontent';

SELECT stupiddata
FROM stupidtable
WHERE stupidcolumn = &stupidvar;

Как я могу объявить переменную и повторно использовать ее в следующих инструкциях, таких как SQLDeveloper.


попытки

  • Используйте раздел DECLARE и вставьте следующий оператор SELECT в BEGINи END;. Доступ к переменной с помощью &stupidvar.
  • Используйте ключевое слово DEFINEи получите доступ к переменной.
  • Использование ключевого слова VARIABLEи доступ к переменной.

Но во время попыток я получаю всевозможные ошибки (несвязанная переменная, синтаксическая ошибка, ожидаемая SELECT INTO...).

bl4ckb0l7
источник
2
Обратите внимание, что подход в принятом ответе @APC может использоваться без PL / SQL, например, в рабочем листе разработчика SQL в соответствии с вашим вопросом. Просто объявите переменную в одной строке (без точки с запятой), затем в строке exec, чтобы установить ее значение (заканчивая точкой с запятой), затем свой оператор select. Наконец, запустите его как сценарий (F5), а не как инструкцию (F9).
Амос М. Карпентер

Ответы:

139

Есть несколько способов объявления переменных в скриптах SQL * Plus.

Первый - использовать 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>

VAR особенно полезен, когда мы хотим вызвать хранимую процедуру, которая имеет параметры OUT или функцию.

В качестве альтернативы мы можем использовать подстановочные переменные. Это хорошо для интерактивного режима:

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>
APC
источник
6
Все хорошо, за исключением того, что вы использовали термин «связываемая переменная». Объявление VAR создает переменную связывания, а ACCEPT или DEFINE создает переменную подстановки.
Дэйв Коста,
1
Можно ли объединить переменные + строки?
Ecropolis
@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'

UPD:

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
Кирилл Леонтьев
источник
Спасибо за ваш ответ, но если я добавлю переменную в двойные кавычки, я получу 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.
Ласло Лугоши
20

В PL / SQL v.10

ключевое слово объявлять используется для объявления переменной

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;

Ваша переменная может использоваться только в пределах BEGINи ENDпоэтому , если вы хотите использовать более чем один , вам придется сделать несколько 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
SVK
источник
5

Просто хочу добавить ответ Мэйтаса . Может быть, это очевидно, но я долго искал, чтобы выяснить, что переменная доступна только внутри конструкции 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

Этот образец отлично работает благодаря автоматическому преобразованию типов (или как он там называется).

Если вы проверите, набрав DEFINE в SQL * Plus, он покажет, что переменная 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, используя значение столбца запроса .

COL [УМН] имеет 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; ВЫБЕРИТЕ число ИЗ двойного; И что я получаю: 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.

Ссылка здесь и здесь 2 .

Ю Ян Цзянь
источник
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.

user12273401
источник
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. Он перекрестно объединен для использования в запросе более одного раза.

Диего Кейроз
источник