Practical Use of the RDB$DB_KEY
By Bjørge Sæther
About the sample code
Note
From Firebird 2.0 declare the RDB$DB_KEY variable in a stored procedure as CHARACTER SET OCTETS.
The examples below are the results of months of efforts trying to perform a large financial analysis on a large database (> 4Gb) within reasonable timings. The real breakthrough came with the introduction of the RDB$DB_KEY. In addition to enormous speed increase, it minimized the need for indexing, thus making both development and implementation both safer and easier.
It should be emphasized, though, that you should make comparisons from time to time. After "having seen the light", you may easily overlook simpler (and faster) possible statements.
The samples could of course be rewritten in a more tutorial fashion. This takes quite some time, and I also wanted to show a few real-world examples (as opposed to the "CUSTOMER - like" examples).
And remember: These procedures are being run on tables having up to 6 million rows. The "success stories" of some of them is like reducing execution time from 1 hr 30 min down to 7 min when introducing the db_key.
Another tip when working with large databases: Field/Record size.
Although VARCHAR fields don't occupy more space in the database than the actual string lengths require, there is one often overlooked consequence of having some "head room" in VARCHAR field sizes:
When selecting rows from a table, equally sized record buffers are allocated, so the size is determined from max possible VARCHAR lengths. This is true when selecting from a client, but also when the server is storing a copy of the record (multi-generation handling) while updating & deleting !
So, watch out ! The following update statement will result in the server allocating 100Mb of RAM per some 80-90,000 records.
Table
CREATE TABLE TEST ( ID INTEGER, STATUS INTEGER, SOMESTRING VARCHAR 1024); UPDATE TEST SET STATUS = 1;
...so, trim your fields !
General
The RDB$DB_KEY may be used is an unique key representing no cost in maintenance or in use. In stored procedures and as a parameter in an update- or delete-sql it is the most efficient record identification you may find in InterBase.
The db_key is not troubled with index balancing,
The db_key is available also when having no unique index. Very useful when it is desirable to drop a unique key while performing updates or deletes.
The db_key does not depend on available server RAM to be efficient, so it's more and more useful as the ration [size of dataset] / [database RAM size] grows. It's possible to perform fast updates/deletes with very little RAM allocated on the server using the db_key.
The db_key is slightly faster than locating rows through an unique index.
No PLANS necessary whatsoever.
Some tips
Use DB_KEY with a FOR SELECT loop when a large part of the records are to be updated, and you would normally use a where-clause.
In such cases, indices are not very efficient.
Example: Different possible sources for getting a certain value, no joins. Approx. 3/4 of the records will be updated
CREATE PROCEDURE COMPUTE_DELIVER_BONUS AS DECLARE VARIABLE v_KEY CHAR(8) CHARACTER SET OCTETS; DECLARE VARIABLE v_V0 VARCHAR(10); DECLARE VARIABLE v_V2 VARCHAR(10); DECLARE VARIABLE v_V6 VARCHAR(10); DECLARE VARIABLE v_VK1 VARCHAR(10); DECLARE VARIABLE v_OT VARCHAR(2); DECLARE VARIABLE v_BONUS FLOAT; BEGIN FOR SELECT RDB$DB_KEY, V0, V6, VK1, V2, ORDER_TYPE FROM INVOICE WHERE DIREKTE <> 9 INTO :v_KEY, :v_V0, :v_V6, :v_VK1, :v_V2, :v_OT DO BEGIN SELECT MAX(VK2_NUM1) FROM VK2_IMP WHERE VK2 = :v_V6 AND VK2_TEKST1 = :v_VK1 AND (VK2_NAVN ="A" OR VK2_NAVN = :v_OT) AND :v_V2 >= VK2_TEKST2 AND :v_V2 <= VK2_TEKST3 AND :v_V0 >= VK2_TEKST4 AND :v_V0 <= VK2_TEKST5 INTO :v_BONUS; IF (:v_BONUS = 0 OR :v_BONUS IS NULL) THEN BEGIN SELECT MAX(VK2_NUM1) FROM VK2_IMP WHERE VK2 = :v_V6 AND VK2_TEKST1 = :v_VK1 AND (VK2_NAVN ="A" OR VK2_NAVN = :v_OT) AND :v_V2 >= VK2_TEKST2 AND :v_V2 <= VK2_TEKST3 AND VK2_TEKST5 = "0000000" INTO :v_BONUS; END; IF (:v_BONUS IS NULL) THEN v_BONUS = 0; UPDATE INVOICE SET COST5 = BASIC_COST * :v_BONUS /1000000 WHERE RDB$DB_KEY = :v_KEY; END END
Avoid creating an index needed only for conditional updates/deletes, replace with FOR SELECT loop and DB_KEY
Indexing has a cost: It's slowing down delete and update. And, if you do a lot of changes in indexed columns, indices are getting unbalanced, with a penalty in performance.
When you need to do a conditional update / delete on a table it may be a solution to use the db_key, where no indices are needed. The larger the ratio updates / rowcount, the larger is the gain of using db_key.
Setting a value in table based on values in joined tables - use an optimized FOR SELECT loop with DB_KEY and values from joined tables selected into variables
One way to do fast updates when you want to set a value in a table based on field values of a related table, is to create one FOR SELECT statement with related tables joined, and perform updates based on the db_key value of the driving table.
Example: Updating value in smaller table with value retrieved from master table
CREATE PROCEDURE UPD_RESKONT_IMP_FROM_Fimp AS DECLARE VARIABLE v_DBKey CHAR(8) CHARACTER SET OCTETS; DECLARE VARIABLE v_Faktno VARCHAR(10); DECLARE VARIABLE v_FaktDato DATE; DECLARE VARIABLE v_K0 VARCHAR(10); DECLARE VARIABLE v_FAKT_PERIODE INTEGER; DECLARE VARIABLE v_IMPORT_PERIODE INTEGER; DECLARE VARIABLE v_RESKONT_PERIODE INTEGER; BEGIN SELECT CAST(DATA AS INTEGER) FROM PROD_PARAMS WHERE ID_KEY = "IMPORT_PER" INTO :v_IMPORT_PERIODE; FOR SELECT R.RDB$DB_KEY, F.FAKTNO, F.FAKT_DATO, F.K0, F.FAKT_PERIODE FROM RESKONT_IMP R LEFT JOIN INVOICE_IMP F ON (R.FAKTNO = F.FAKTNO) WHERE F.FAKT_DATO IS NOT NULL OR R.IMPORT_PERIODE = :v_IMPORT_PERIODE INTO :v_DBKey, :v_Faktno, :v_FaktDato, :v_K0, :v_FAKT_PERIODE DO BEGIN IF (:v_FaktDato IS NULL) THEN v_RESKONT_PERIODE = NULL; ELSE v_RESKONT_PERIODE = :v_IMPORT_PERIODE; UPDATE RESKONT_IMP SET FAKT_DATO = :v_FaktDato, FAKT_PERIODE = :v_FAKT_PERIODE, IMPORT_PERIODE = :v_RESKONT_PERIODE, K0 = :v_K0 WHERE RDB$DB_KEY = :v_DBKey; END /*FOR SELECT*/ END
Example: Joining master table with 2 smaller tables
CREATE PROCEDURE ADD_CENTRALLAGER AS DECLARE VARIABLE v_KEY CHAR(8) CHARACTER SET OCTETS; DECLARE VARIABLE v_IsCL VARCHAR(2); /*K4 er Centrallager*/ DECLARE VARIABLE v_HasCL VARCHAR(2); /*K4 har Centrallager*/ DECLARE VARIABLE v_V7 VARCHAR(2); BEGIN FOR SELECT F.RDB$DB_KEY, K_Has.K2_TEKST1, V_Is.V7 FROM INVOICE_IMP F LEFT JOIN K2_IMP K_Has ON (F.K4 = K_Has.K2) LEFT JOIN V7_IMP V_Is ON (F.K4 = V_Is.V7) INTO :v_Key, :v_HasCL, :v_IsCL DO BEGIN IF (:v_IsCL IS NULL) THEN :v_V7 = :v_HasCL; ELSE :v_V7 = :v_IsCL; UPDATE INVOICE_IMP SET V7 = :v_IsCL WHERE RDB$DB_KEY = :v_KEY; END END
Example: Joining master table with smaller table joined twice
CREATE PROCEDURE ADD_REGIONS /*Region*/ AS DECLARE VARIABLE v_KEY CHAR(8) CHARACTER SET OCTETS; DECLARE VARIABLE v_K3_S VARCHAR(2); DECLARE VARIABLE v_K3_L VARCHAR(2); BEGIN FOR SELECT F.RDB$DB_KEY, KSelg.K3, KLev.K3 FROM INVOICE_IMP F LEFT JOIN K2_IMP KSelg ON (F.K2 = KSelg.K2) LEFT JOIN K2_IMP KLev ON (F.K4 = KLev.K2) INTO :v_KEY, :v_K3_S, :v_K3_L DO UPDATE INVOICE_IMP SET K3 = :v_K3_L, K9 = :v_K3_S WHERE RDB$DB_KEY = :v_KEY; END
Example: Master table joined with two smaller tables
CREATE PROCEDURE ADJUST_MHO_FIMP AS DECLARE VARIABLE v_KEY CHAR(8) CHARACTER SET OCTETS; DECLARE VARIABLE pV7 VARCHAR(20); /*=Centrallager*/ DECLARE VARIABLE pV5_T1 VARCHAR(20); /*=MHO Clager*/ DECLARE VARIABLE pV5_T2 VARCHAR(20); /*=MHO Inte Cl*/ DECLARE VARIABLE pMHO VARCHAR(5); /*=MHO justerad*/ BEGIN /* Sett justert MHO for selgende filial som ikke er sentrallager*/ FOR SELECT F.RDB$DB_KEY, V.V5_TEKST1, V.V5_TEKST2, CL.V7 FROM INVOICE_IMP F LEFT JOIN V5_IMP V ON (F.V5 = V.V5) LEFT JOIN V7_IMP CL ON (F.K2_LIKO = CL.V7) INTO :v_KEY, :pV5_T1, :pV5_T2, :pV7 DO BEGIN IF (:pV7 IS NULL OR :pV7 = "") THEN pMHO = :pV5_T2; ELSE pMHO = :pV5_T1; UPDATE INVOICE_IMP SET MHO_SELG = :pMHO WHERE RDB$DB_KEY = :v_KEY; END END