Recently I did note that quite many people are using in Oracle database packages queries as the dynamic SQL. They have red in somewhere that the bind variables will make the query run faster and save some perfomance. To clear some basics about database I had to write the following examples that I did post here too. Don’t understand me incorrectly the bind variables are fast with dynamic SQL but only for an external programming languages like C or Java. To keep your code in database the dynamic SQL is needed only some extreme circumnstanses. To keep your queries in a static way in your prodecures and packages will save the compiling time before executing and any missing column or table will be raised as an error during you are compiling the code.
To make sure all SQL queries have the same “starting point” and none of them will be cached empty the share_pool. Do it before running any of the following queries. You may need to use your SYS user to execute it.
ALTER SYSTEM FLUSH SHARED_POOL;
Image may be NSFW.
Clik here to view.
The first PL/SQL anonymous block will execute 50000 times SQL query “SELECT rownum INTO v_cnt FROM dual WHERE rownum = 1;” and to make sure we will get the closest timing we will execute the block 5 times. To measure the execution time we are using SYSDATE and not some fancy procedure that can have bugs. In other words we will keep the block as simple as possible.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..50000 LOOP DECLARE v_cnt PLS_INTEGER; BEGIN SELECT rownum INTO v_cnt FROM dual WHERE rownum = 1; END; END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END;
Image may be NSFW.
Clik here to view.
The output shows execution time in seconds as 4, 4, 4, 5 and 4 and that makes in average total: 4.2.
The second query is a dynamic SQL query without the bind variables and we will execute the same query as above.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..50000 LOOP EXECUTE IMMEDIATE 'DECLARE v_cnt PLS_INTEGER; BEGIN SELECT rownum INTO v_cnt FROM dual WHERE rownum = '|| 1 ||'; END;'; END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END;
Image may be NSFW.
Clik here to view.
The SQL output is very different from the first anonymous block and the times are 9, 9, 10, 9 and 9. That makes the average total: 9.2.
The third anonymous block is a dynamic sql with the bind variables and the query is the same as on the last two examples.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..50000 LOOP DECLARE v_cnt PLS_INTEGER; BEGIN EXECUTE IMMEDIATE 'SELECT rownum FROM dual WHERE rownum = :v' INTO v_cnt USING 1; END; END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END;
Image may be NSFW.
Clik here to view.
The output shows 4, 5, 4, 4 and 5 and the average is 4.4. The last average is not so far from the first anonymous block with a static SQL but still in long run it will make some difference and even less sense makes to write all your queries dynamically into the database procedure or package.
The following group of examples are with Oracle Insert and to try them on your database you would need to create a new table so it wouldn’t break anything existsing.
CREATE TABLE my_oracle_test (id NUMBER(17) NOT NULL, text VARCHAR2(2000));
Image may be NSFW.
Clik here to view.
The first insert is done to make sure the table has needed tablespace size taken and the inser statements wouldn’t loose any time by extending the tablespace. Once the insert is done we will delete all lines using TRUNCATE TABLE command and don’t worry about the COMMIT truncate table does commit.
BEGIN FOR i IN 1..150000 LOOP INSERT INTO my_oracle_test (id, text) VALUES (i, 'This is My Row '||i); END LOOP; END; / TRUNCATE TABLE my_oracle_test;
Image may be NSFW.
Clik here to view.
Now we are ready to execute the first insert examples and this is static SQL inserting into just created my_oracle_test table and it does it 150000 times. We are keep measuring the time as on the example above and after the anonymous block is done it will truncate the table data. Once again I am running the test 5 times to get the average time.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..150000 LOOP INSERT INTO my_oracle_test (id, text) VALUES (i, 'This is My Row '||i); END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END; / TRUNCATE TABLE my_oracle_test;
Image may be NSFW.
Clik here to view.
The output shows 15.99, 16.99, 14.99, 15.99 and 16.99 and that makes the average 16.19.
The next anonymous block is with a dynamic SQL and without the bind variables. The insert statement is the same and we will truncate the rows after the block is done.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..150000 LOOP EXECUTE IMMEDIATE 'INSERT INTO my_oracle_test (id, text) VALUES ('||i||', ''This is My Row ''||'||i||')'; END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END; / TRUNCATE TABLE my_oracle_test;
Image may be NSFW.
Clik here to view.
The second insert is way slower than the static SQL and the times are 107, 107, 105, 105 and 105. The average in seconds is 105.8.
The third insert is a dynamic SQL and with the bind variables. The rest of code has left as on the other examples above.
DECLARE v_starttime DATE; v_endtime DATE; BEGIN v_starttime := SYSDATE; FOR i IN 1..150000 LOOP EXECUTE IMMEDIATE 'INSERT INTO my_oracle_test (id, text) VALUES (:v, ''This is My Row ''||:v)' using i, i; END LOOP; v_endtime := SYSDATE; dbms_output.put_line('time in seconds:'||((v_endtime - v_starttime) * 86400)); END; / TRUNCATE TABLE my_oracle_test;
Image may be NSFW.
Clik here to view.
The SQL output shows times as 15.99, 16.99, 16.99, 15.99 and 15.99. The average is 16.36 and again the timing is not so far from the static SQL (16.19) but we can’t say that a dynamic SQL with the bind variables is quicker than a static SQL.
In short, keep your code in database static and don’t write it as dynamic unless this is something extreme and can’t be written as static. When you are using C or Java application USE bind variables calling out the database procedure or in SQL queries.
See Also:
Oracle Select Oracle Count Oracle Group By Oracle Having Online Tech Support Home