0

我正在使用 PL/SQL,我想检查表中是否存在记录,我尝试使用 when exists 或 if not exists,但最终出现异常。

Select name from nameTable where name = nameFromArgument;

当程序启动时,我输入了一个错误的名称,它不在数据库中我得到一个错误,甚至无法处理它

我试过这样的事情:

PL/SQL 检查查询是否返回空

编辑:

对不起,我之前放的选择只是一个例子,整个代码[对不起变量名...]在这里:

CREATE OR REPLACE
PROCEDURE SELLSTOCKS
( IloscAkcji IN NUMBER
, NazwaAkcji IN VARCHAR2
, IdRachunkuMaklerskiego IN NUMBER
)  AS
Bledna_ilosc EXCEPTION;
Bledna_nazwa EXCEPTION;
amount NUMBER;
stocksCash number;
idRachunku number;
stocksOnDealerSide number;
temp number;
tempus akcje_uzytkownika.nazwa % type;
BEGIN

--Select nazwa into tempus from akcje_uzytkownika where nazwa = nazwaakcji;
--dbms_output.put_line(tempus);

  dbms_output.put_line('pre');
 select nazwa into tempus from akcje_uzytkownika where nazwa = nazwaakcji;
 dbms_output.put_line('pre2');
 if tempus!=null then
    if ( iloscakcji> amount) THEN
  raise Bledna_ilosc;
  else
  select ilosc, cena_sztuki into amount, stockscash from akcje_uzytkownika where nazwa= NazwaAkcji and id_rachunku_maklerskiego= IdRachunkuMaklerskiego;
 dbms_output.put_line(amount);
 dbms_output.put_line(stockscash);
  amount :=amount - iloscakcji;
  dbms_output.put_line(amount);
  update akcje_uzytkownika set ilosc= amount  where nazwa=NazwaAkcji and  id_rachunku_maklerskiego= idrachunkumaklerskiego;
  stockscash:= iloscAkcji * stocksCash;
  dbms_output.put_line(stockscash);
  select id_rachunku into idrachunku from rachunek_maklerski where id_rachunku_maklerskiego= idrachunkumaklerskiego; 
  temp:=SALDOUPDATE(stockscash, idrachunku);
  select ilosc into stocksOnDealerSide from akcje where nazwa = nazwaAkcji;
  stocksOnDealerSide := stocksondealerside+ iloscakcji;
  dbms_output.put_line(stocksOnDealerSide);
  update akcje set ilosc = stocksondealerside where nazwa = nazwaAkcji;
  end if;
  else
  dbms_output.put_line('post');
  end if;
EXCEPTION

  when Bledna_ilosc then
  dbms_output.put_line('Bledna ilosc');
  when Bledna_nazwa then
  dbms_output.put_line('Bledna nazwa');

END SELLSTOCKS;

我正在使用 oracle sql developer 1.5.5

BEGIN

--Select nazwa into tempus from akcje_uzytkownika where nazwa = nazwaakcji;
--dbms_output.put_line(tempus);

  dbms_output.put_line('pre');
  select nazwa into tempus from akcje_uzytkownika where nazwa = nazwaakcji;
  dbms_output.put_line('pre2');
  exception when no_data_found then
  raise no_data_found;

-- is this the right place?

  select ilosc, cena_sztuki into amount, stockscash from akcje_uzytkownika where nazwa= NazwaAkcji and id_rachunku_maklerskiego= IdRachunkuMaklerskiego;
  dbms_output.put_line(amount);
  dbms_output.put_line(stockscash);
  amount :=amount - iloscakcji;
  dbms_output.put_line(amount);
  if ( iloscakcji> amount) THEN
  raise Bledna_ilosc;
  else
  update akcje_uzytkownika set ilosc= amount  where nazwa=NazwaAkcji and  id_rachunku_maklerskiego= idrachunkumaklerskiego;
  stockscash:= iloscAkcji * stocksCash;
  dbms_output.put_line(stockscash);
  select id_rachunku into idrachunku from rachunek_maklerski where id_rachunku_maklerskiego= idrachunkumaklerskiego; 
  temp:=SALDOUPDATE(stockscash, idrachunku);

...........

4

1 回答 1

3

你有四个主要的错误(我可以看到)。

首先,tempus!=null永远不会工作。由于 null 是缺少信息,因此您在检查 null 时不能使用(不)等式运算符。不存在!=不存在。正确的语法是使用is null, 或在你的情况下tempus is not null.

其次,if ( iloscakcji> amount) THEN由于您没有初始化 value ,因此永远不会评估为 true amount。一个数字不能大于,null因为它不存在。您必须初始化变量amount。也许您的意思是下一个选择语句在此之上?

第三,您的所有select ... into ...语句都包含在单个变量中 - 而不是用户定义的表类型。这意味着如果您的查询返回 0 行,no_data_found则会引发异常。如果返回多于一行,too_many_rows则会引发异常。

如果您只期望最多一排,那么您无需过多担心too_many_rows. 如果可能的话,这应该由你的表上的唯一约束来强制执行。例如,我希望这nazwa应该是唯一的akcje_uzytkownika

no_data_found略有不同,我希望,你的意思是tempus != null. 假设amount已初始化为一个值,则可以将其重写为:

begin

    -- do some stuff    
    begin

       select nazwa into tempus from akcje_uzytkownika where nazwa = nazwaakcji;

    -- If there-s no data catch the exception.
    exception when no_data_found then
       if ( iloscakcji > amount) then
          raise Bledna_ilosc;
       end if;

    end;        
    -- do some more stuff

end;

最后,这并不是一个真正的错误,但是您的代码相当不可读。我会考虑使用更多的空格,以便您可以看到发生了什么。

PS 通过 Oracle 版本,我的意思是 11gr2 或 10g 等,但在这种情况下没有任何区别。

于 2012-05-20T10:31:33.183 回答