Як оголосити змінну і використовувати її в тому ж сценарії Oracle SQL?


133

Я хочу написати код для багаторазового використання і потрібно оголосити деякі змінні на початку та повторно використовувати їх у сценарії, наприклад:

DEFINE stupidvar = 'stupidvarcontent';

SELECT stupiddata
FROM stupidtable
WHERE stupidcolumn = &stupidvar;

Як я можу оголосити змінну і повторно використовувати її в операторах, що слідують за такими, як при використанні її SQLDeveloper.


Спроби

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

Але я отримую всілякі помилки під час моїх спроб (Незв'язна змінна, Синтаксична помилка, Очікувана SELECT INTO...).


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

Відповіді:


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>

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>

6
Все добре, за винятком використання терміна "прив'язка змінної". Декларація VAR створює змінну прив'язки, тоді як ACCEPT або DEFINE створює змінну заміщення.
Дейв Коста,

1
Чи можливо об'єднати змінні + рядки?
Екрополь

@Ecropolis - так, у SQL Plus використовується період за замовчуванням. Використовуйте SET CONCAT, щоб визначити символ, який відокремлює ім'я змінної заміни від буквено-цифрових символів, які негайно слідують за назвою змінної. У PL / SQL або SQL використовуйте подвійну трубу || об'єднати.
Ласло Лугосі

Якщо SQL є стандартною мовою, то чому так важко знайти канонічну довідку, яка працює просто скрізь? WTF ???
jww

1
@jww - SQL - це стандарт, але він не завжди вказує точний синтаксис, тому різні продукти RDBMS можуть реалізувати речі по-різному; Арифметика дати - хороший приклад. Також старі продукти бази даних, такі як Oracle, часто впроваджували функції до того, як Стандарт охоплював їх: наприклад, ієрархічний синтаксис CONNECT BY. Але в цьому випадку ми обговорюємо SQL * Plus, який є клієнтським інструментом і так не покритий стандартом ANSI.
APC

28

Спробуйте скористатися подвійними лапками, якщо це змінна ознака:

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

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 [UMN] є нова опція для зберігання значення запиту за назвою поля.

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
Тоха

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 Outputвікно.

Посилання на тут і тут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.


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. Він є перехресним з'єднанням, щоб використовувати його не раз у запиті.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.