3

我必须编写一个 pl/sql 代码(实际上是一个函数),当以郊区为参数时,它返回一个邮政编码,代码如下:

create or replace 
FUNCTION get_postCode(p_suburb IN varchar2)
RETURN varchar2 
IS
--
v_postcode varchar2;
--
CURSOR c1 IS 
SELECT locality, postcode FROM table_postcode;
--
BEGIN
--
FOR r1 IN c1 
loop
    IF upper(r1.locality) = upper(p_suburb)
    THEN 
      v_postcode := r1.postcode;
      return v_postcode;
    END IF;
exit WHEN c1%notfound;
END loop;
-- return v_postcode;
--
exception WHEN others then
v_postcode := null;  
END; 

table_postcode 是从 Post Office 获得的,它包含郊区(地区作为表格中的列)和邮政编码以及与此案例无关的其他字段。

当我使用该函数时,它返回正确的值,当我将这个函数用作 SELECT 子句的列时,它仅在我不在 FROM 子句之后添加任何其他子句时才返回。这对我来说很奇怪。

情况是:

select street, suburb, get_postcode(suburb) from my_table;

上面的行给了我结果,但是

select street, subur, get_postcode(suburb) from my_table order by suburb;

失败并给我以下错误消息:

ORA-06503: PL/SQL: Function returned without value
ORA-06512: at "U11254683.GET_POSTCODE", line 25
06503. 00000 -  "PL/SQL: Function returned without value"
*Cause:    A call to PL/SQL function completed, but no RETURN statement was
       executed.
*Action:   Rewrite PL/SQL function, making sure that it always returns
       a value of a proper type.

如果我在一个块中调用该函数,例如:

Declare
v_post varchar2(10);
Begin
v_post := get_postcode('Sydney');
DBMS_OUTPUT.PUT_LINE('The post code is '||v_post);
End;

结果是正确的,给了我 2000。

4

2 回答 2

7

当什么都没找到时,你应该返回一些东西。在exception声明中发表评论之前

return v_postcode;

编写例程的方式可能什么也没找到,然后在没有返回任何内容的情况下到达函数的末尾,因此出现错误“函数返回没有值”。字面意思就是这样。

但是,您可以通过直接选择郊区来更轻松地完成此操作。但是,如果您在郊区收到多个邮政编码(如果可能的话),您会怎么做?

create or replace FUNCTION get_postCode(i_suburb IN varchar2)
RETURN varchar2 
IS
  l_postcode varchar2;
BEGIN
  select postcode 
    into l_postcode
   where upper(i_suburb) = upper(locality);
-- 
  return l_postcode;
exception 
  WHEN others then
    return null;  
END; 

在此版本中,如果一个郊区存在于多个邮政编码中,您将获得 null。现在由您的设计决定如何处理这种情况。然后你可以在异常中处理它

exception
   when TOO_MANY_ROWS then
        return '**Error** Several Values'; -- do something meaningful here 
   when NO_DATA_FOUND then
        return null; -- if you want to return null on nothing found
于 2013-05-12T07:54:45.297 回答
0

尝试这个:

CREATE OR REPLACE FUNCTION get_postCode
(p_suburb IN varchar2)
RETURN varchar2 
IS
    v_postcode varchar2;
    CURSOR c1 IS 
        SELECT locality, postcode FROM table_postcode;
BEGIN
    FOR r1 IN c1 LOOP
        EXIT WHEN c1%notfound;
        IF upper(r1.locality) = upper(p_suburb) THEN 
          v_postcode := r1.postcode;
          EXIT;
        END IF;
    END LOOP;
    return v_postcode;
    exception WHEN others then
    return null;  
END; 
/

通过FOR在找出第一个邮政编码时退出循环,您可以确保没有TOO_MANY_ROWS引发异常。此外,通过将语句放在return外面,IF您可以确保NULL在没有找到任何内容的情况下返回至少 a。

于 2013-05-12T08:36:31.103 回答