5

在 Oracle SQL 开发人员中是否有可能做这样的事情

CREATE FUNCTION fnCheckValid(accountidd IN NUMBER) 
RETURN NUMBER 
   IS retval NUMBER(4,0);
   BEGIN 
      SELECT COUNT(accountid_fk) 
      INTO retval 
      FROM tbl_AccountAuthentications 
      WHERE accountid_fk = accountidd; 
      RETURN(retval); 
    END;
/


ALTER TABLE tbl_AccountAuthentications
ADD CONSTRAINT chkCheckvalid CHECK(fnCheckValid(accountid_fk) <= 1);

我不断得到的错误是

Error starting at line 999 in command:
ALTER TABLE tbl_AccountAuthentications
ADD CONSTRAINT chkCheckvalid CHECK(fnCheckValid(accountid_fk) <= 1)
Error report:
SQL Error: ORA-00904: "FNCHECKVALID": invalid identifier
00904. 00000 -  "%s: invalid identifier"
*Cause:    
*Action:

该函数正在创建中,我可以找到它,但是当我尝试调用它时,我不断收到该错误

这就是我想要实现的目标

AccountID    RegularID     OpenID
 1            5             null
 1            null          10
 1            null          11
 1            6                             <-- Forbidden

这样用户就不能创建 2 个常规帐户,但可以拥有任意数量的 OpenID 帐户

表设置如下

CREATE TABLE tbl_AccountAuthentications(
        newAuthID NUMBER(4,0)
          CONSTRAINT naid_pk PRIMARY KEY,        
        accountid_fk NUMBER(4,0)
          CONSTRAINT accid_fk 
            REFERENCES tbl_UserAccounts(account_id),
        regularid_fk NUMBER(4,0)
          CONSTRAINT rgid_fk
            REFERENCES tbl_StrongRoom(password_id),
        openid_fk NUMBER(4,0)
          CONSTRAINT opid_fk
            REFERENCES tbl_OpenID(openid)
);
4

2 回答 2

10

不,您不能这样做,请参阅检查约束的限制

  • 调用用户定义的函数

但是您可以使用虚拟列来解决

ALTER TABLE tbl_AccountAuthentications ADD (fnCheck NUMBER GENERATED ALWAYS AS (fnCheckValid(accountid_fk)) VIRTUAL);


ALTER TABLE tbl_AccountAuthentications
ADD CONSTRAINT chkCheckvalid CHECK(fnCheck <= 1);

注意,函数必须是确定的,否则它不起作用。Oracle 不会验证您的函数是否实际上是确定性的,它只是检查关键字。这是允许的(尽管它根本没有任何意义):

CREATE OR REPLACE FUNCTION DET_FUNCTION RETURN NUMBER DETERMINISTIC IS 
BEGIN 
    RETURN DBMS_RANDOM.RANDOM();
END;
/
于 2017-01-04T15:23:29.380 回答
6

无论有多少OpenID值,您都希望确保AccountIDRegularID列是唯一的。

正如您已经确定的那样,这样做的唯一方法是约束它。您在评论中注意到您已经尝试过触发器。这不会限制数据库中的值,它只确保它尝试验证触发器何时启用。我希望,当您尝试它时,您会收到错误“ORA-04091:正在变异,触发器/函数可能看不到它。” 当您从表中进行选择时,您正在更改(插入或更新)。

如果你必须限制这一点,那么这就是你应该做的;这样做的问题是您的表格没有标准化。所以,正常化它。使用两张桌子而不是您现有的一张。

首先,您需要您的RegularID与AccountID 相比是唯一的,这意味着它应该存储在此级别。它似乎tbl_UserAccounts在此标识符上是唯一的,因此请更改此表并将您的RegularID存储在那里。

接下来,您需要一个包含尽可能多的OpenID的表,该表的数量与用户可能需要的一样多。这意味着您需要一个在AccountIDOpenID 1上唯一的表。

create table openid_account_auth (
   , account_id number(4,0)
   , open_id number(4,0)
   , constraint pk_openid_account_auth
       primary key (account_id, open_id)
   , constraint fk_openid_account_auth_accid
       foreign key (account_id)
       references tbl_UserAccounts(account_id)
   , constraint fk_openid_account_auth_open
       foreign key (open_id)
       references tbl_OpenID (openid)
     );

这张表上有一点(和你自己的),这意味着多个账户可以有相同的OpenID。如果您不打算这样做,那么您应该在 中添加AccountID作为外键tbl_OpenID,这是确保每个OpenID与一个且只有一个AccountID关联的唯一方法。

然后,您可以创建一个视图,以便以相同的方式获取信息,如果您真的觉得需要使用它。我不确定你为什么会。

create or replace view AccountAuthentications as
 select account_id, regular_id, null
   from user_accounts
  union all
 select account_id, null, open_id
   from openid_account_auth;

简而言之,除非受到严格的限制,否则您应该始终将数据存储在其自然级别并使用数据库来确保保持完整性。如果您随后需要稍微不同地使用数据,您可以使用视图或物化视图等来执行此操作。

1. 很抱歉,我不能让自己在每张桌子的名称前加上tbl_.

于 2013-05-27T21:20:48.367 回答