序列生成的主键似乎不适用于 select 语句

Tec*_*ins 2 sql database oracle plsql foreign-keys

序列生成的主键似乎不适用于 select 语句

我正在尝试创建一个传输数据库。我已经在 Access 和 MySQL 中完成了。现在我在 Oracle 中使用它,而且我对使用它完全陌生。我遇到了序列生成字段的麻烦。或者至少我是这么认为的。当您阅读整篇文章时,您将上手。

这是我的 driverRecords 表的代码:

CREATE TABLE TRANSPORT_SCHEMA.driverRecords
(
    driverReg varchar2(15) PRIMARY KEY,
    driverIDNumber varchar2(10) UNIQUE,
    surname varchar2(15) NOT NULL,
    otherNames varchar2(30) NOT NULL,
    address varchar2(20),
    email varchar2(320) NOT NULL,
    DOB date NOT NULL,
    employmentDate date NOT NULL
);
Run Code Online (Sandbox Code Playgroud)

这是驱动许可证表的代码:

CREATE TABLE TRANSPORT_SCHEMA.drivingLicences
(
    licenceID varchar2(6) PRIMARY KEY,
    vehicleClass varchar2(15),
    issueDate date,
    expiryDate date,
    driverReg varchar2(15) UNIQUE,
    CONSTRAINT fk_driverReg FOREIGN KEY(driverReg) REFERENCES driverRecords(driverReg),
    CONSTRAINT Chk_vehicleClass CHECK(vehicleClass IN('A','B','C','D','E','F','G'))
);
Run Code Online (Sandbox Code Playgroud)

所以我试图将记录插入到驾驶执照中。driveLicences 中的 driverReg 是 driverRecords 中 driverReg 的子键。这个想法是,如果 driverRecords 表中不存在驱动程序的记录,则驱动程序表中不应存在任何驱动程序

这是我用来进行插入的代码;

insert into drivingLicences VALUES('123456', 'D', '12-12-2019','12-12-2020', 'DRV/1003/2020');
Run Code Online (Sandbox Code Playgroud)

这是返回的错误:

ORA-02291: integrity constraint (TRANSPORT_SCHEMA.FK_DRIVERREG) violated - parent key not found

现在我完全明白错误是由违反完整性约束引起的......但问题是......我实际上有一个驱动程序DRV/1003/2020,我的记录中有注册号

我的驱动程序注册号是由这个序列和触发器生成的:

序列:

CREATE SEQUENCE driverNumberSequence
    START WITH 1000
    INCREMENT BY 1
    NOCACHE
    NOCYCLE;
Run Code Online (Sandbox Code Playgroud)

扳机:

CREATE OR REPLACE TRIGGER regNumberGenerator BEFORE 
INSERT ON driverRecords
FOR EACH ROW

    DECLARE
        prefix char(3) := 'DRV';
        driverNumber char(4);
        postfix char(4);
        generatedDriverNumber char(15);

    BEGIN
        SELECT TO_CHAR(SYSDATE, 'YYYY') INTO postfix FROM sys.dual;
        SELECT TO_CHAR(driverNumberSequence.nextval) INTO driverNumber FROM dual;
        generatedDriverNumber:= prefix || '/' || driverNumber || '/' || postfix;
        SELECT TO_CHAR(generatedDriverNumber) INTO :NEW.driverReg FROM dual;
        DBMS_OUTPUT.PUT_LINE('Driver Reg Number generated:' || generatedDriverNumber);
    END;
Run Code Online (Sandbox Code Playgroud)

我试图做两个选择语句来找出问题所在:

1)SELECT * FROM DRIVERRECORDS where DRIVERREG = 'DRV/1001/2020';

这个返回 0 行

2)SELECT * FROM DRIVERRECORDS where DRIVERREG LIKE 'DRV/1001/2020%';

这个其实返回了一个结果

我不知道我做错了什么。但我认为这与我的 driverReg 信息的保存方式有关。任何帮助将不胜感激

如果有帮助,以下是我保存的所有驱动程序注册号:

SELECT driverReg FROM driverRecords;

*DRV/1001/2020
DRV/1002/2020
DRV/1003/2020
DRV/1004/2020
DRV/1005/2020*
Run Code Online (Sandbox Code Playgroud)

简而言之,我需要知道为什么:

a) select 语句适用于LIKE '%'和不适用= ''

b) 即使我没有违反完整性约束,我也无法进行插入

Mur*_*nik 5

generatedDriverNumber在您的触发器中定义为char(15)-长度为 15的固定长度字符字段。由于您用更短的值填充它,它会将其填充到 15 的全长。这解释了为什么在开头使用 alike进行查询value 有效,但不适用于=.

长话短说,将定义更改为varchar2,您应该没问题:

generatedDriverNumber varchar2(15);
Run Code Online (Sandbox Code Playgroud)