Skip to Main Content

Breadcrumb

Question and Answer

Connor McDonald

Thanks for the question, Hassan.

Asked: July 04, 2001 - 1:48 pm UTC

Last updated: April 12, 2021 - 5:17 am UTC

Version: 8.0

Viewed 100K+ times! This question is

You Asked

I was wondering if there is some possible way that I can fetch the last record from a table. In this particular case, obviously one is unaware of the number of records in a table. All I want is to write a query whcih will simply fetch all the columns of the last record. I have tried using ROWNUM, but I guess it does'nt work that way. If I generate a report, then I want to base my report on the same query whcih will simply fecth all the columns of the last record.

Are there any built-ins in report builder which can fetch the last record of a table?

Thanks


Hassan

and Tom said...

What is your definition of the last record?

You know, the last record inserted may very well be the first record returned by "select * from t", it may be the 100'th, it may be the 1000'th, it may in fact be ANY record.

Records are not returned in order of anything.

If you want the last record inserted, you need to have a timestamp or sequence number assigned to each record as they are inserted, then:


select * from t where TIMESTAMP_COLUMN = ( select max(timestamp_column) from T )
and rownum = 1;


will get the "last" record. It is the ONLY way.

followup to the first 6 comments below

Proof by example (this proof shows all of the suggestions below are WRONG - the ONLY way to do this is to have some column you can order on to find the "last" row):

If ANY of these approaches worked -- the answer would be 4. It is NEVER 4 in the following example:

tkyte@TKYTE816> create table t ( x int, a char(2000), b char(2000), c char(2000) );

Table created.

tkyte@TKYTE816>
tkyte@TKYTE816>
tkyte@TKYTE816> insert into t (x) values ( 1 );

1 row created.

tkyte@TKYTE816> commit;

Commit complete.

tkyte@TKYTE816> insert into t (x,a,b,c) values ( 2, 'x', 'x', 'x' );

1 row created.

tkyte@TKYTE816> commit;

Commit complete.

tkyte@TKYTE816> insert into t (x,a,b,c) values ( 3, 'x', 'x', 'x' );

1 row created.

tkyte@TKYTE816> commit;

Commit complete.

tkyte@TKYTE816> delete from t where x = 2;

1 row deleted.

tkyte@TKYTE816> commit;

Commit complete.

tkyte@TKYTE816> insert into t (x,a) values ( 4, 'x' );

1 row created.

tkyte@TKYTE816> commit;

Commit complete.

tkyte@TKYTE816>
tkyte@TKYTE816>
tkyte@TKYTE816> select x from t where rowid in(select max(rowid) from t);

X
----------
3

tkyte@TKYTE816>
tkyte@TKYTE816> select x from t where rownum = 1 order by rowid desc;

X
----------
1

tkyte@TKYTE816>
tkyte@TKYTE816> select x from ( select * from t order by rowid desc ) where rownum = 1;

X
----------
3

tkyte@TKYTE816>
tkyte@TKYTE816> select x from t minus select x from t where rownum < (select count(*) from t);

X
----------
3

tkyte@TKYTE816>
tkyte@TKYTE816>


Trust me on this one -- there is exactly ONE way to sort of get the last row inserted. You MUST have some column that you can order by.

Rowid - won't work. Its based on flie/block/slot. We reuse rowids, we can even change them (partitioned tables, IOTS). You might have extent #1 in file 55 and extent #2 in file 2. Extent 4 might start on block 555 of file 3, Extent 5 might start on block 2 of file 3. Rowids are NOT sortable.

Rownum -- doesn't work either, it just doesn't work.





Rating

  (65 ratings)

Is this answer out of date? If it is, please let us know via a Comment

Comments

Last record

AKS, July 05, 2001 - 1:01 am UTC

I think this can be done by using max(rowid) in the where clause.

Fetching Last Record from a table

Rajasekar N, July 05, 2001 - 1:01 am UTC

Tom,
Can't we fetch the last record like this...?

select * from t
minus
select * from t
where rownum < (select count(*) from t)



Fetching Last Record from a table

Rajasekar N, July 05, 2001 - 1:03 am UTC

Tom,
Can't we fetch the last record like this...?

select * from t
minus
select * from t
where rownum < (select count(*) from t)



Use ROWNUM

Dhawal Shah, July 05, 2001 - 2:27 am UTC

This is very much possible with rownum.
The solution which is given here calls for the overhead of
one column.

Though the cost for the following query may be a bit high,
you can reduce the overhead of storing a column.

" select * from emp where rownum = 1 order by rowid desc; "



one more way ?

Vikram, July 05, 2001 - 7:39 am UTC

can't we use this ...
select * from t where rowid in(select max(rowid) from t);

Last row

Danial, July 05, 2001 - 9:03 am UTC

"I think this can be done by using max(rowid) in the where clause" -- would it still work if you do alot of deletes and updates?

Great

Reader, July 05, 2001 - 11:48 am UTC

What a greate clarification for a daily misconcept.

Thanks

Great

Reader, July 05, 2001 - 11:50 am UTC

What a greate clarification for a daily misconcept.

Thanks

Fetching last record from a table

DATTA KUAMR, July 05, 2001 - 12:57 pm UTC

I feel, whatever Mr.Tom has given is right. By using select max(rowid) from table, no doubt gives the result. If we try this soon after inserting a record into the table, we will get the correct result. But, try after inserting a row, select ... from table, order by ..., and again try select max(rowid) from table, I am sure you will not get the desired result.

Fetching of Last Record

Hassan Ayub, July 06, 2001 - 2:07 am UTC

Thanks a lot Tom for a very thorough and well explained answer.. I tried the approaches you suggested and you are certainly right!!!


clearest solution

Daikun, July 27, 2001 - 4:29 am UTC

the securest way to get consistent results! once more you made the theory behind sql and database-concepts clear! This ist more important than using 'dirty' tricks to improve performance...

Fetching last record from a table

Sujoy Hore, February 16, 2005 - 8:04 am UTC

Using the above concept to fetch last record, I wish to fetch the 2nd instance of a record containing one field with common value. Say,in this table there are 5 records against XYZ (name field),ABC has one record,DEF has three records.What I would like to fetch is second record details of XYZ. How do I do that? I can find out the first and last record against XYZ using the procedure earlier explained by you.

Tom Kyte
February 16, 2005 - 8:40 am UTC

are ABC and DEF in the problem description just to be confusing?

I ask because I see two possible questions here:

a) give me the second (YOU need to provide some definition of what second means here, you have not yet) record by NAME_FIELD (eg: we return the 2cd record by name_field after sorting by SOMETHING YOU NEED TO SPECIFY for XYZ, DEF -- ABC having one record does not have a 2cd)

b) given a name, say XYZ, find the 2cd record after sorting.


for a) you use:


select *
from ( select name_field,
row_number() over (partition by name_field ORDER BY SOMETHING ) rn
from t )
where rn = 1;


for b) you use:

select *
from (select a.*, rownum r
from (select * from t where name_field = :x order by SOMETHING ) a
where rownum <= 2 )
where r = 2;



rowid

KARTHICK, September 30, 2005 - 6:56 am UTC

Can you tell me how the ROWID are generated by oracle? That is suppose I have a table and sequentially I insert 100 rows and after that I delete few and again continue with my insert. Does the ROWID of the rows deleted will be reused ???

Tom Kyte
September 30, 2005 - 9:39 am UTC

the rowid is not "generated" in as much as "derived"

The rowid is the address of the row on disk.

Since space can and will be reused, rowids can and will be "reused"

A rowid implies NOTHING other than the location of the row physically. It does not imply age, order of insertion, or anything like that.

rowid

KARTHICK, October 03, 2005 - 10:26 am UTC

i insert into a table the following data

rowid value
-------------
AAA 10
BBB 20
CCC 30

as you said rowid is a value derived by oracle and just for example iam having the above specified values.

now i delete the value 20

rowid value
-------------
AAA 10
BBB
CCC 30


now i insert another row with value 40

rowid value
-------------
AAA 10
BBB
CCC 30
DDD 40

here i dint use the row id BBB again. i dint because when i was going through HIGH WATER MARK in your book it said
the free space will not be used untill the table is rebuild or truncated. and it also says, when a full table scan is done the row with rowid BBB will also be scaned.

so now if i rebuild how the records will be organized, will new rowid will be derived for all the rows i.e. new blocks will be assigned to all the rows in a sequential order or ???

( and another thing to be noted is you dont suggest rebuild of table )

Tom Kyte
October 03, 2005 - 11:24 am UTC

rowids can and will be reused randomly.  That you showed a case where it wasn't (a made up case) isn't really meaningful.

That space can and will be reused.



ops$tkyte@ORA9IR2> create table t ( x int, y varchar2(4000), z varchar2(4000) ) tablespace tools;
 
Table created.
 
ops$tkyte@ORA9IR2>
ops$tkyte@ORA9IR2> insert into t values ( 1, null, null );
 
1 row created.
 
ops$tkyte@ORA9IR2> insert into t values ( 2, rpad('*',4000,'*'), rpad('*',3000,'*') );
 
1 row created.
 
ops$tkyte@ORA9IR2> insert into t values ( 3, null, null );
 
1 row created.
 
ops$tkyte@ORA9IR2> commit;
 
Commit complete.
 
ops$tkyte@ORA9IR2>
ops$tkyte@ORA9IR2> select x, rowid, dbms_rowid.rowid_block_number(rowid) from t;
 
         X ROWID              DBMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
---------- ------------------ ------------------------------------
         1 AAAKODAAFAAAC/qAAA                                12266
         2 AAAKODAAFAAAC/qAAB                                12266
         3 AAAKODAAFAAAC/qAAC                                12266
 
ops$tkyte@ORA9IR2>
ops$tkyte@ORA9IR2> delete from t where x = 2;
 
1 row deleted.
 
ops$tkyte@ORA9IR2> commit;
 
Commit complete.
 
ops$tkyte@ORA9IR2>
ops$tkyte@ORA9IR2> insert into t values ( 4, rpad( '*',1000,'*'), null );
 
1 row created.
 
ops$tkyte@ORA9IR2> select x, rowid, dbms_rowid.rowid_block_number(rowid) from t;
 
         X ROWID              DBMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
---------- ------------------ ------------------------------------
         1 AAAKODAAFAAAC/qAAA                                12266
         4 AAAKODAAFAAAC/qAAB                                12266
         3 AAAKODAAFAAAC/qAAC                                12266



There is a "demo" of it happening:

manual segment space management
8k block




Rowids are not "sequential" things, they say WHERE a row exists, not when a row was inserted or anything like that. 

rowid

karthick, October 05, 2005 - 8:00 am UTC

Tom said...

Another is way is to consider a SEGMENT to be a set of blocks (these blocks are
collected together into extents). Say you create a segment (initially totally
empty). It might look like this:


Oeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee

where O = block used for Overhead (oracle used that block for space management)
e = empty block, block has NEVER contained data.


Now, you insert 1,000 rows into the segment. It now looks like this:

Oddddddddddddddddddddddddddddeeeeeeeeeeeeeeeeee

where d = block with data on it. The set of blocks with d's on them are the set
of blocks under the HWM -- the HWM is the set of blocks that have contained
data. Now I delete 1,000 rows from the table. it will now look like this:

Oddddddddddddddddddddddddddddeeeeeeeeeeeeeeeeee

My Question...

can you tell me what happens if i insert another 100 rows

???????????????????????????????????????????????

if eeeee's are going to change as ddddd's then new row takes new blocks am i right
then they will be given a new address that is rowid am i right. then when the rowid
is going to be reused ( if i dont preform truncate )

if iam wrong please correct me.

Tom said continues...

that is -- it is the same. Just like in a flood -- a HWM can only be advanced.
A HWM never decreases for an object -- you have to in effect recreate the object
to decrease the HWM. So a truncate table would do this:

Oeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee


Tom Kyte
October 05, 2005 - 11:00 am UTC

when you did the delete - the blocks went onto the freelist (manual segment space management) or the bitmaps used by ASSM were updated to say "empty" and that space will be reused before the HWM of the segment is extended.


the e's would not be used if the 100 rows can fit in the d's.

Primary key ordering

San, May 23, 2006 - 11:47 am UTC

I have a question here. When we create a primary key on a column in a table, (presuming that a unique index would be created on that column), how are the rows ordered ?
Considering table "test_order"

create table test_order (sno number primary key, name varchar2(50));

insert into test_order
(select rownum, object_name
from all_objects where rownum < 100);

commit;

delete from test_order where rownum < 60;

commit;

insert into test_order
(select rownum, object_name
from user_objects
where rownum < 60);

commit;

When we do a "select * from test_order", it wouldn't be sorted by the primary key column which is "sno".
Could you please explain the phenomenon.

Thanks
San

Tom Kyte
May 23, 2006 - 3:47 pm UTC

the rows are not ordered in a heap table, they just go where they fit.


rows in relational tables don't have an order really.

One Another way to retrieve last record

Mahmood Lebbai, June 26, 2006 - 1:55 am UTC

Consider this query...

select * from (select rownum r,empno,ename,job,sal,deptno from emp) where r=(select max(rownum) from emp)

Tom, any comments...


By the way, On a personal note, I have been trying to ask some questions related to performance on SQL queries...And I get this "Sorry..I have a huge backlog...", all the time. I do not seem to find some time where you are not overwhelmed with questions. I do not plan to ask the question on this review forum after seeing your stern warning not to use this forum for additional questions. Should I hold on to my waiting game?

Tom Kyte
June 26, 2006 - 7:41 am UTC

that would be a rather inefficient method to get a random record from the table.

select rownum r, ... from emp 

will full scan the table, hitting the rows in the order they happen to exist on disk (which has no relation to the order of insertion or anything like that!).


(select max(rownum) from emp)

is an inefficient method to do a count(*) against the table.


and the result would be some random row - not the last inserted row.


ops$tkyte@ORA10GR2> create table t ( x int, y varchar2(4000) ) tablespace mssm;
 
Table created.
 
ops$tkyte@ORA10GR2>
ops$tkyte@ORA10GR2> insert into t values ( 1, rpad('*',4000,'*') );
 
1 row created.
 
ops$tkyte@ORA10GR2> insert into t values ( 2, rpad('*',3000,'*') );
 
1 row created.
 
ops$tkyte@ORA10GR2> insert into t values ( 3, rpad('*',3000,'*') );
 
1 row created.
 
ops$tkyte@ORA10GR2> commit;
 
Commit complete.
 
ops$tkyte@ORA10GR2> delete from t where x = 2;
 
1 row deleted.
 
ops$tkyte@ORA10GR2> commit;
 
Commit complete.
 
ops$tkyte@ORA10GR2> insert into t values ( 4, null );
 
1 row created.
 
ops$tkyte@ORA10GR2> commit;
 
Commit complete.
 
ops$tkyte@ORA10GR2> select * from ( select rownum r, x from t ) where r = (select max(rownum) from t );
 
         R          X
---------- ----------
         3          3

<b>the only way to have a "last record" is for you to 

a) define your concept of "last"
b) record attribute values that support your definition of "last"
c) execute a query that retrieves this "last" record based on a and b.</b>
 

Excellent

Mahmood Lebbai, June 26, 2006 - 4:47 pm UTC

Tom,

Your response was very good. And I did not expect I would get the reply, the same day. Wow!! Great!!!

I tried the same examples on my schema and I have to agree that you were absolutely right. I usually apply my strategy of "By hook or by crook" to solve the problem like these. Sometimes or (all the time, I don't know :-)), this approach compromises on its efficiency and performance. That is not a good thing, I Understand.

I appreciate your prompt rejoinder. Thanks, again.


Is there an exception to this rule?

A reader, August 15, 2006 - 4:52 pm UTC

"Trust me on this one -- there is exactly ONE way to sort of get the last row
inserted. You MUST have some column that you can order by.

Rowid - won't work. Its based on flie/block/slot. We reuse rowids, we can even
change them (partitioned tables, IOTS). You might have extent #1 in file 55 and
extent #2 in file 2. Extent 4 might start on block 555 of file 3, Extent 5
might start on block 2 of file 3. Rowids are NOT sortable."

Let's assume that the task of the application is to:

-loop until no more rows
- get a group of rows where each group can have many rows with the same (repeating) name, e.g.
AAAA .....
AAAA .....
BBBB ....
CCCC ....
CCCC ....
CCCC ...
...

- process this group of rows
-end loop

If this is the case, we can then sort by rowid in the inline view, can't we? Even if the rows change location/rowid in between application runs we will still get correct processing result, as all the application need to do is to process rows in groups of names where each group contains many names and each name can span many rows (rows with identical name column) and to ensure it does not process same row twice. Is that true? Is this an exception to the NO ORDER BY ROWID rule?


Tom Kyte
August 15, 2006 - 5:01 pm UTC

why would you sort by rowid?

I don't know what you are asking really - the original poster wanted to get things out in order of insertion.

what is your goal?


If you need to process in groups of names, you BETTER be sorting by - well - names don't you thing? sorting by rowid would give you an ordering roughly equivalent to "dbms_random.random" when you talk about the NAME attribute.

The goal

A reader, August 15, 2006 - 5:57 pm UTC

as I said is just to process, i.e. "touch" each row, exactly once, every time application runs, that's all, nothing else. And to do that not row by row, but group by group where each group has many rows with different names and each of these names can be repeated. The order of those rows with the same name is not important. What is important is to process in certain naming order, e.g.

Aaron - DDave
DDave - MMilton
...
MMilton - ZZed

but to process each row exactly once (but group by group)

i.e. in the first group there are many DDaves. All of these DDaves should be processed in that group as name can only be allocated to one and only one group

Tom Kyte
August 15, 2006 - 6:25 pm UTC

but they you obviously have to sort... by NAME


if you sorted by rowid, the names would be returned, well, randomly.


so, if the goal is to process the data by groups of "X", you best sort by "X" to get all of the "X"'s with the same value together.

last entry

A reader, September 24, 2008 - 11:17 pm UTC

Tom:

What do you think of those designs that add a special derived (Y/N) column to indicate the last record entered and use a trigger to keep this column up-to-date and saying that MAX function created overhead on the DB. It seems very bad design to me and extra code and many updates and you end up with one derived extra column per child table plus the mutating trigger error. is this correct.

ROWNUM=1 vs ROWNUM<=1

Bilal, March 25, 2009 - 10:52 am UTC

Tom, why these queries are different in performance.
for me its very strange, would you please tell me the reason for this.

1) CPU - I/O usage is very high

SELECT inv_number,
inv_type,
inv_sub_type,
carrier_local_flag,
inv_run,
invoice_mode_code
FROM (SELECT insih.inv_number,
insih.inv_type,
insih.inv_sub_type,
insih.carrier_local_flag,
insih.inv_run,
insih.invoice_mode_code
FROM sales_inv_headers insih
WHERE :p_company_code IS NULL
AND:p_system_id IS NULL
AND:p_inv_number IS NULL
AND:p_inv_run IS NULL
AND:p_bulk_suffix IS NULL)
WHERE ROWNUM =1


2) CPU - I/O usage is very low

SELECT inv_number,
inv_type,
inv_sub_type,
carrier_local_flag,
inv_run,
invoice_mode_code
FROM (SELECT insih.inv_number,
insih.inv_type,
insih.inv_sub_type,
insih.carrier_local_flag,
insih.inv_run,
insih.invoice_mode_code
FROM sales_inv_headers insih
WHERE :p_company_code IS NULL
AND:p_system_id IS NULL
AND:p_inv_number IS NULL
AND:p_inv_run IS NULL
AND:p_bulk_suffix IS NULL)
WHERE ROWNUM <=1
Tom Kyte
March 29, 2009 - 8:22 pm UTC

and you didn't include.....

a tkprof report which would have told us a LOT

so, provide that.


I don't think your analysis is correct, something else was happening. All objects is a very very complex view, however:

select *
from
 all_objects where rownum = 1


call     count       cpu    elapsed       disk      query    current        rows
------- ------  -------- ---------- ---------- ---------- ----------  ----------
Parse        1      0.06       0.07          0          0          0           0
Execute      1      0.00       0.00          0          0          0           0
Fetch        2      0.00       0.00          0         57          0           1
------- ------  -------- ---------- ---------- ---------- ----------  ----------
total        4      0.06       0.08          0         57          0           1

********************************************************************************

select *
from
 all_objects where rownum <= 1


call     count       cpu    elapsed       disk      query    current        rows
------- ------  -------- ---------- ---------- ---------- ----------  ----------
Parse        1      0.07       0.07          0          0          0           0
Execute      1      0.00       0.00          0          0          0           0
Fetch        2      0.00       0.00          0         57          0           1
------- ------  -------- ---------- ---------- ---------- ----------  ----------
total        4      0.07       0.07          0         57          0           1




they are the same.

ROWNUM=1 vs ROWNUM<=1

Bilal, March 30, 2009 - 8:20 am UTC

yes here it is with TKPROF report.
Here :p_rownum = 1.


1) CPU - I/O usage is very high

SELECT inv_number,
inv_type,
inv_sub_type,
carrier_local_flag,
inv_run,
invoice_mode_code
FROM (SELECT insih.inv_number,
insih.inv_type,
insih.inv_sub_type,
insih.carrier_local_flag,
insih.inv_run,
insih.invoice_mode_code
FROM sales_inv_headers insih
WHERE :p_company_code IS NULL
AND:p_system_id IS NULL
AND:p_inv_number IS NULL
AND:p_inv_run IS NULL
AND:p_bulk_suffix IS NULL)
WHERE ROWNUM = :p_rownum

call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 3 0.03 0.03 0 0 0 0
Execute 3 0.01 0.53 0 0 0 0
Fetch 3 0.36 0.49 4151 12480 0 3
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 9 0.40 1.06 4151 12480 0 3

2) CPU - I/O usage is very low

SELECT inv_number,
inv_type,
inv_sub_type,
carrier_local_flag,
inv_run,
invoice_mode_code
FROM (SELECT insih.inv_number,
insih.inv_type,
insih.inv_sub_type,
insih.carrier_local_flag,
insih.inv_run,
insih.invoice_mode_code
FROM sales_inv_headers insih
WHERE :p_company_code IS NULL
AND:p_system_id IS NULL
AND:p_inv_number IS NULL
AND:p_inv_run IS NULL
AND:p_bulk_suffix IS NULL)
WHERE ROWNUM <= :p_rownum


call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 3 0.00 0.00 0 0 0 0
Execute 4 0.01 0.52 0 0 0 0
Fetch 3 0.00 0.00 56 12 0 3
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 10 0.01 0.54 56 12 0 3

Tom Kyte
March 30, 2009 - 4:48 pm UTC

oh, you didn't use the number 1, you used a bind variable.


do you have the row source operation there? looks like an explain plan but shows the amount of work PLUS the plan.

This is useful but i want something like this

riteshtiwary, April 24, 2009 - 6:41 am UTC

select feild from demo where date_colum=(date_colum between sysdate-2 and sysdate) and TIMESTAMP_COLUMN = ( select max(timestamp_column) from demo);
Tom Kyte
April 27, 2009 - 11:58 am UTC

so??? go for it I guess???


guess you better make sure that timestamp column with the max value is in the last two dates.

ROWNUM=1 vs ROWNUM<=1

bilal, June 03, 2009 - 3:47 am UTC

sorry tom due to medical probs i ws on leave.

ys im applying row operation.
so is there any difference if i use parameter or number in ROWNUM criteria?
please define the difference.

Last inserted rows

Meth, September 16, 2010 - 10:30 pm UTC

We can use ORA_ROWSCN and come up with the last rows which have been inserted to a table when we sort it desc.
Tom Kyte
September 17, 2010 - 7:04 am UTC

only if you rebuild the table from scratch with rowdependencies.

and it would not return the last rows inserted, it would return the last rows MODIFIED - which is very different.

I would not recommend it for this.

last N records for a logging table with only Insert operations

A reader, November 15, 2010 - 2:13 pm UTC

Tom:
I trust and understand your statememt. "You MUST have some column that you can order by"

my situation is a little bit different.
I have a huge table for logging purpose, the only DML operation on it is insert, rows are never updated or deleted.
the first column is a seq_nbr column generated by a sequence, and it is the primary key, it is a regular heap table.

right now , I am using :
SELECT *
FROM
(
SELECT * FROM z_log
ORDER BY seq_nbr desc
)
WHERE ROWNUM<20;

to get the last 20 rows.

check the plan, Oracle will do a "INDEX FULL SCAN DESCENDING INDEX(unique)" first , followed by "Table access by Global Index ROWID".

for this case, if the index is really huge, INDEX FULL SCAN will be expensive. and because this table is only allowing Insert operations, do you think there can be another way to make the query faster by avoiding index full scan?

I am thinking using IOT table might be a better option, but not quite sure the Pros and Cons.

can you clarify?

thanks


Tom Kyte
November 16, 2010 - 3:35 am UTC

there is also a count stopkey in there isn't there.

that means the plan stops being run when the stopkey is hit.

it will index full scan desc for only 20 entries and then stop. trace it - you'll see that very little IO is done on the index, it is not reading the entire thing.

ROWNUM and stopkey

James Su, November 16, 2010 - 12:37 pm UTC

Hi Tom,

For the original question, is this as good as your answer?

SELECT * FROM (select * from T ORDER BY timestamp_column DESC)
WHERE rownum = 1;

Tom Kyte
November 17, 2010 - 5:49 am UTC

yes, that would be an equivalent query assuming timestamp_column was NOT NULL.


Followup on "Last N records for a logging table with only Insert operations"

Zack, February 03, 2011 - 3:21 pm UTC

Regarding the query to get the last 20 (19) rows:

SELECT * FROM
(
SELECT * FROM z_log
ORDER BY seq_nbr desc
)
WHERE ROWNUM < 20;

Its cost is 3 (with a similar table of 23,000 rows). However, the cost increases to 21 if we want the last 1000 rows, and 40 for last 2000. A friend of mine suggests I use the following query which has a constant cost of 8.

SELECT * FROM z_log
WHERE
seq_nbr >= (SELECT (MAX(seq_nbr) - 2000) FROM z_log)

Could you please comment on these 2 queries?

Thanks much in advance!
Tom Kyte
February 03, 2011 - 4:15 pm UTC

I think you need new friends.

So what if the cost goes up - but we do less work?

we could

a) index range scan descending to find N index entries
b) do a table access by index rowid for each one

(and the cost of getting 1000 rows IS MORE than the cost of getting 20 rows and so on, the cost really does go up!)


or we could

a) do an index min/max scan
b) then do an index range scan descending for find about N entries (remember please, SEQUENCES ARE NEVER EVER IN A MILLION BILLION YEARS GAP FREE!!!!!!!!)
c) and then do a table access by index rowid for each one.



I don't know about you - but I'd rather do less work and get the right answer.


Tell me, what is the logic behind choosing a query that has a constant cost? So what if the cost is constant? Why is that "good", why is that better? They are entirely different queries and in fact return different answers.


Stick with your first query please.

last row

A reader, February 04, 2011 - 8:02 am UTC


Followup on "Last N records for a logging table with only Insert operations"

Zack, February 04, 2011 - 3:57 pm UTC

Thanks a ton, Tom.

I hear you - the less work, the better. I wish the query plans would support this but the query plans seem to say the 2nd query is better cost/bytes wise. So I'm a bit confused. I understand there are exceptions that a lower cost might not result in a shorter time. Can you please comment on this a bit more? Please refer to the complete query plans for the 2 queries below.

>> They are entirely different queries and in fact return different answers. 

Understood. It's not a problem in our case as we don't have to get exactly N rows back.

>> what is the logic behind choosing a query that has a constant cost? So what if the cost is constant? Why is that "good", why is that better?

Constant or not, what's more important is of curse a cheaper cost which the 2nd query seems to give according to the paln. In our case, the N in "last N" varies based on the user input.

===== EXPLAIN PLAN Outputs =====
SQL> EXPLAIN PLAN FOR select * from (select * from Z_LOG order by seq_nbr desc) f1 where ROWNUM <= 2000;
---------------------------------------------------------------------------------------------------
| Id  | Operation                     | Name              | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT              |                   |  2000 |   414K|    40   (0)| 00:00:01 |
|*  1 |  COUNT STOPKEY                |                   |       |       |            |          |
|   2 |   VIEW                        |                   |  2000 |   414K|    40   (0)| 00:00:01 |
|   3 |    TABLE ACCESS BY INDEX ROWID| Z_LOG             | 22417 |  2583K|    40   (0)| 00:00:01 |
|   4 |     INDEX FULL SCAN DESCENDING| XPK_Z_LOG         |  2000 |       |     5   (0)| 00:00:01 |
---------------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - filter(ROWNUM<=2000)

SQL> EXPLAIN PLAN FOR SELECT * FROM Z_LOG WHERE seq_nbr >= (SELECT (MAX(seq_nbr) - 2000) FROM Z_LOG);
--------------------------------------------------------------------------------------------------
| Id  | Operation                    | Name              | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT             |                   |  1121 |   129K|     8   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID | Z_LOG             |  1121 |   129K|     6   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN           | XPK_Z_LOG         |   202 |       |     2   (0)| 00:00:01 |
|   3 |    SORT AGGREGATE            |                   |     1 |     5 |            |          |
|   4 |     INDEX FULL SCAN (MIN/MAX)| XPK_Z_LOG         | 22417 |   109K|     2   (0)| 00:00:01 |
--------------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   2 - access("seq_nbr">= (SELECT MAX("seq_nbr")-2000 FROM "ZACK"."Z_LOG" "Z_LOG"))

Tom Kyte
February 06, 2011 - 12:00 pm UTC

... Constant or not, what's more important is of curse a cheaper cost which the 2nd
query seems to give according to the paln. In our case, the N in "last N"
varies based on the user input. ...

I've said it a million times, you cannot compare the cost of two different queries. It just doesn't work in real life.

http://asktom.oracle.com/pls/asktom/f?p=100:11:0::::P11_QUESTION_ID:313416745628


We all have to use our knowledge, our wisdom, our insights into how things work to work them out in our heads. It is obvious to me in looking at the two rather simple plans above - especially after I wrote out in english what they do step by step - which is more efficient.


I can even tell you "why". One of the queries accurately knows "I SHALL RETURN 2,000 RECORDS".

The other query says "I think on average I'll return about 1,121 records"


One of the plans has a dead accurate cost based on dead accurate "guesses" as to the cardinality. Guesses that we KNOW are dead accurate.

One of the plans doesn't. It estimated the cardinality "pretty well", but not as good as the other one. Therefore, any cost associated with it will be impacted by that.



RowID

Ananth, February 08, 2011 - 2:58 pm UTC

Hi Tom,

For ROWID, can we say it this way.
On any day/Time, the ROWID is unique. (if its being derived).

Its only the varying data that resides on that ROWID.


Regards
Ananth
Tom Kyte
February 10, 2011 - 3:55 pm UTC

the rowid is unique within a table, yes.

the rowid is not unique within a database.

And these days the rowid assigned to a row may change over time, it is not constant as it used to be in the olden days.


rowid

Debasis, February 15, 2011 - 7:18 am UTC

Rowid cannot be used to extract specifically first record or last record inserted, because if one record is deleted in between, then there might be chance that rowid can be assigned to a new record inserted. so in that case max(rowid) or min(rowid) is not working properly to extract the first record and last record.

One more instance like while loading data to a table, suppose there are two processes to load the data into table, then rowid assignment will not be in a sequential order. In this case, min(rowid) and max(rowid) will not be suitable to extract the last record and first record.

Thanks,
Debasis

rowid

Debasis, February 15, 2011 - 7:21 am UTC

In my view, the only way to extract the first record or last record by the use of timestamp.




Thanks,
Debasis
Tom Kyte
February 15, 2011 - 8:41 am UTC

and even that doesn't "work" if by "first" or "last" you mean in order of commit.

The application, the developer, the data modeler - they must specify the rules that deem something "first" or "last" and the data model must support attributes that allow that rule to be followed.

How to find first and last record from table

Surendra, May 22, 2011 - 1:33 pm UTC

Hi TOM,
I am Surendra want know about findout first and last record from a table.Can you send reply to this query.


Thanks & Regards
Surendra



Tom Kyte
May 23, 2011 - 11:53 am UTC

tell me what a first and last record would look like?

The concept of first and last do not exist in relational databases - that hasn't changed since I first answered this question (we already stated such) way back when.

to find out first record in a table in oracle

T.venkatesh, May 22, 2011 - 2:32 pm UTC


to find out first record in a table in oracle

T.venkatesh, May 22, 2011 - 2:34 pm UTC

Please send me the answer for it its urgent
Thanks,
T.venkatesh
Tom Kyte
May 23, 2011 - 12:00 pm UTC

if you would have read this page, you would have realized before posting that...


there is no answer. Please think about it, this isn't dbase or a flat file here, it is a relational database - there is no such thing as a first or last record.

RIGHT ANSWER FOR LAST RECORD IN ORACLE

A reader, June 20, 2011 - 2:00 am UTC

SELECT * FROM (SELECT REGN_DT,ROWNUM R, REGN_NO FROM VT_OWNER) WHERE R >(SELECT MAX(ROWNUM) -4 FROM VT_OWNER);

Tom Kyte
June 20, 2011 - 1:06 pm UTC

hahahahahahah

Hope you don't have any production code that counts on that. But only because you are wronger than wrong.

There is NO SUCH THING as a "last record", data goes where ever data feels like going.

There are so many things wrong with your approach, so many misunderstandings (I'm being rather blunt here because if you had read this page, you would know that already)...


first of all, you select "-4" in the above. What is -4 for??? getting max(rownum)-4 and getting everything "greater than that" - what does that do?? It looks like you are trying to get the last 4 records - but whatever.

It still doesn't work.

Because rows do not go to the "end" of a table. They go where ever we feel like putting them.

Here is an example for you - using your query. I used an 8k blocksize, 11.2.0.2 database, with ASSM tablespaces (automatic segment space managed)





ops$tkyte%ORA11GR2> create table t ( x int, y varchar2(4000), z varchar2(4000) );

Table created.

ops$tkyte%ORA11GR2> 
ops$tkyte%ORA11GR2> insert into t values ( 1, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

ops$tkyte%ORA11GR2> insert into t values ( 2, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

ops$tkyte%ORA11GR2> insert into t values ( 3, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

ops$tkyte%ORA11GR2> insert into t values ( 4, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

ops$tkyte%ORA11GR2> insert into t values ( 5, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

ops$tkyte%ORA11GR2> commit;

Commit complete.

ops$tkyte%ORA11GR2> select x from t;

         X
----------
         5
         1
         2
         3
         4

<b>Now, you might already notice a problem here... The first row is coming out AFTER the fifth row - go figure.  We put data where we feel like it...</b>

ops$tkyte%ORA11GR2> delete from t where x = 5;

1 row deleted.

ops$tkyte%ORA11GR2> commit;

Commit complete.

ops$tkyte%ORA11GR2> insert into t values ( 42, rpad( 'x', 4000, 'x' ), rpad( 'x', 2000, 'x' ) );

1 row created.

<b>that is arguably the last row right?  I'm in a single user database here, no one else..</b>

ops$tkyte%ORA11GR2> commit;

Commit complete.

ops$tkyte%ORA11GR2> select x from t;

         X
----------
        42
         1
         2
         3
         4

<b>oh, but wait, we decided in this case (this is a direct cut and paste from sqlplus, nothing up my sleeve here) it should be "first" in a full scan...</b>

ops$tkyte%ORA11GR2> 
ops$tkyte%ORA11GR2> SELECT *
  2    FROM (SELECT x, ROWNUM R FROM t)
  3   WHERE R >(SELECT MAX(ROWNUM) -4 FROM t);

         X          R
---------- ----------
         1          2
         2          3
         3          4
         4          5


Interesting, you returned everything EXCEPT for the last row!!!!


Now, please stop thinking there is a first row, a second row, a last row and so on.

Unless you have some attribute in your data to tell you what is "last", there is going to be no such thing as a last row.

Alexander, June 20, 2011 - 2:59 pm UTC

Why did you include those varchar2(4000) columns? Was that to ensure the row deleted would not be inserted back into the free block and thereby accidentally show ordered results?
Tom Kyte
June 20, 2011 - 3:18 pm UTC

so I could come up with an easy case knowing I could control the blocks on the freelist very simply.

I just had a row per block in my example. A block either

a) had a row
b) was on the free list and had no rows

and nothing in between

It can happen with ANY size row however. I do not NEED to have the varchar's there, it just made it really trivial for me to demonstrate "it does not work that way, it has never worked that way, it will NEVER work that way, there is no such thing as a 'last row'"

how to identify the rowid which one is max(rowid) in table

charan, June 23, 2011 - 1:16 am UTC

Hi Tom,
i have one doubt on rowid concept.how to idntify which rowid is max in the table?
Actualy i have table
sno rowid name values
--- ----- ---- ------
1 AAA rad 10
1 AAB rad 20
1 AAC rad 5

please give advise on this?
Tom Kyte
June 23, 2011 - 7:57 am UTC

why? It is not like it implies that is the newest record or anything. The rowid is just an address, it does not convey any other information short of an address.

MAX() works on rowids, so you can find the max rowid in the same fashion you find the max salary.

how to get duplicate records?

charan, June 23, 2011 - 8:04 am UTC

HI Tom,
i write one querry plz explain that querry........
select * from emp where rowid not in (select max(rowid)from emp group by deptno);

above i posted one table that refers to how it works?plz reslove me.
Tom Kyte
June 23, 2011 - 8:17 am UTC

plz? German postal codes? I'm confused, what do German postal codes have to do with anything? http://zipcodelookup.me/postal-codes-for-german-towns.php


In any case, I'll explain the query.

The subquery "select max(rowid) from emp group by deptno" generates a set of rowids - one for each department found in the emp table. The rowid selected is the "max" rowid for that department - which is not really meaningful, it just ensures at least one and at most one rowid per department will be selected.

Then, the outer query selects all of the data from the emp table where the rowid is NOT IN that set of maximum rowids.

That query returns all of the rows from EMP *except* for one row from each department in EMP - it is a rather "random" row that is dropped out.


It looks like a nonsense query to me, I cannot see how it could ever be useful.

how to delete the duplicate records in table

charan, June 28, 2011 - 12:39 am UTC

hI Tom,

i have a table that table have different duplicate data
how to remove the duplicate records in my table?
how to process please give suggestion

How to Fix

garys, June 28, 2011 - 12:34 pm UTC

MAX should modified in the next release to give an error if you pass it a rowid. Any answer would be nonsense, anyway.

(I realize that doing this would break a lot of existing code, but it would be a Tough Love Blessing in disguise.)
Tom Kyte
June 28, 2011 - 12:59 pm UTC

No it isn't nonsense, you can use between, >, <, = and so on. the concepts exist for that (thankfully - we couldn't do "do it yourself parallelism" without them!)

http://asktom.oracle.com/pls/asktom/asktom.search?p_string=%22do+it+yourself+parallelism%22

http://asktom.oracle.com/pls/asktom/asktom.search?p_string=dbms_parallel_execute


for a rowid to be greater than another rowid implies it is in a file with a bigger number (or the same), a block with a greater number (or the same) and a slot in the block that is bigger.

it can be used in a sensible fashion:

delete from t where rowid NOT IN (select max(rowid) from t group by SOME,KEY,COLUMN(S) );


that would de-duplicate that table using SOME,KEY,COLUMNS as the unique key - it would keep one record per SOME,KEY,COLUMN value....

Enlightening response

garys, June 28, 2011 - 1:11 pm UTC

Thanks for the response about what it means to order rowids. I read about the rowid structure some time back and now it makes (more) sense.

comparing rowids

Sokrates, June 28, 2011 - 2:31 pm UTC

it seems we even can compare rowids of different tables (though I cannot think of a case where this might be of any use):

sql > CREATE SCHEMA AUTHORIZATION sokrates
  2    create table heap1(i int)
  3    create table heap2(i int)
  4  ;

Schema created.

sql > insert
  2  when mod(rownum, 2) = 0 then into heap1 else into heap2
  3  select rownum from dba_objects where rownum<=4;

4 rows created.

sql > select
  2     a.rowid || ' ' ||
  3     case when a.rowid < b.rowid then '<' when a.rowid = b.rowid then '=' when a.rowid > b.rowid then '>' end
  4     || ' ' || b.rowid
  5  from heap1 a, heap2 b;

A.ROWID||''||CASEWHENA.ROWID<B.ROWIDTHE
---------------------------------------
AAAuoUAAEAAAAi9AAA > AAAuoTAAEAAAAi1AAA
AAAuoUAAEAAAAi9AAA > AAAuoTAAEAAAAi1AAB
AAAuoUAAEAAAAi9AAB > AAAuoTAAEAAAAi1AAA
AAAuoUAAEAAAAi9AAB > AAAuoTAAEAAAAi1AAB


However, comparing the rowid of a heap table with one of an iot seems to fail with ORA-01410: invalid ROWID:

sql > CREATE SCHEMA AUTHORIZATION sokrates
  2    create table heap(i int)
  3    create table iot( i int primary key ) organization index
  4  ;

Schema created.

sql > insert
  2  when mod(rownum, 2) = 0 then into heap else into iot
  3  select rownum from dba_objects where rownum<=4;

4 rows created.

sql >   select
  2        a.rowid || ' ' ||
  3        case when a.rowid < b.rowid then '<' when a.rowid = b.rowid then '=' when a.rowid > b.rowid then '>' end
  4        || ' ' || b.rowid
  5  from heap a, iot b;
from heap a, iot b
     *
ERROR at line 5:
ORA-01410: invalid ROWID


why ?
Tom Kyte
June 28, 2011 - 4:10 pm UTC

... (though I cannot think of a case where this might be of any use) ....

clustered tables perhaps - in that situation - two rows can have the same rowid ;) Still not sure what you would use it for though.


And true on the IOT since it really isn't a rowid, it is a logical rowid - the rowid of an IOT contains a rowid hint and the primary key of the row in the IOT.

need help on sql query

Shayam, June 29, 2011 - 11:31 pm UTC

Hi Tom,

CREATE TABLE CMR
(
LOGDATE CHAR(8 BYTE),
LOGTIME CHAR(6 BYTE),
ORDERNO NUMBER(10),
MESSAGE VARCHAR2(1024 BYTE)
)
LOGGING
NOCOMPRESS
NOCACHE
NOPARALLEL
MONITORING;


CREATE TABLE ABC
(
ORDERNO NUMBER(10) NOT NULL,
FILENAME VARCHAR2(64 BYTE),
JOBDATE CHAR(8 BYTE)
)
LOGGING
NOCOMPRESS
NOCACHE
NOPARALLEL
MONITORING;

insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8082,'Maintenance','20110624');
insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8083,'Maintenance','20110623');
insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8084,'Maintenance','20110625');
insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8085,'condition_test','20110622');
insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8086,'condition_test','20110621');
insert into abc(ORDERNO,FILENAME, JOBDATE) Values(8087,'condition_test','20110621');
commit;

Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110624', '090031',8082,'HELD BY OPS');
Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110623', '130231',8083,'HELD BY OPS');
Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110625', '070131',8084,'HELD BY OPS');
Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110622', '030031',8085,'HELD BY OPS');
Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110621', '180031',8086,'HELD BY OPS');
Insert into CMR(LOGDATE, LOGTIME,ORDERNO,MESSAGE) Values('20110621', '020031',8087,'HELD BY OPS');
commit;

SQL Query:
--------
select b.filename, b.jobDATE, to_char(to_date(logdate || logtime, 'YYYY-MM-DD HH24:MI:SS'),
'DD-MM-YY HH24:MI:SS') Held_Time,
TRUNC (MOD ((SYSDATE - TO_DATE (MAX (logdate) || MAX (logtime),'YYYY-MM-DD HH24:MI:SS')) * 24,24))
HR
from cmr a, abc b
where message = 'HELD BY OPS' and a.orderno=b.orderno
group by b.filename, b.jobDATE,a.logdate, a.logtime
having TRUNC (MOD ((SYSDATE - TO_DATE (MAX (logdate) || MAX (logtime),'YYYY-MM-DD HH24:MI:SS')) *
24,24)) >= 3

Result:-
=======
FILENAME JOBDATE HELD_TIME HR
--------------------------------------------------------
Maintenance 20110623 23-06-11 13:02:31 20
condition_test 20110621 21-06-11 18:00:31 15
condition_test 20110621 21-06-11 02:00:31 7
condition_test 20110622 22-06-11 03:00:31 6

Requirement: how we can get latest held time for particular file name which is held by ops more than 3 hour. here i am getting 3 held time for filename "condition_test" .

Could you please help me on this.

Thanks for effort....


Tom Kyte
July 01, 2011 - 8:14 am UTC

do you want three records with the latest held time that exceeds 3 hr's associated with each filename?

or did you want the record that has the latest held time that exceeds 3 hr's?

I don't really understand your data - you'll have to explain it. Your query is quite confusing. I don't see a single primary key nor foreign key, I don't know what is supposed to be unique. It is strange to be grouping by something (logtime) that you also MAX() against. And it is really really strange to be concatenating two fields together "MAX (logdate) || MAX (logtime)" to create a date since the logdate and logtime attributes in that case could come from two different rows!!!


and to use mod??? if the time between sysdate and your max function was 24 hours - the mod would be zero - the held time would be 24 hours, but you wouldn't recognize it as such??

I *think* you might want this, but truly - I am not sure at all.

ops$tkyte%ORA11GR2> select * from (
  2  select b.filename,
  3         b.jobDATE,
  4         to_char(to_date(logdate || logtime, 'YYYY-MM-DD HH24:MI:SS'), 'DD-MM-YY HH24:MI:SS') Held_Time,
  5         (SYSDATE - TO_DATE (logdate || logtime,'YYYY-MM-DD HH24:MI:SS')) * 24 hr,
  6             row_number() over (partition by b.filename order by to_date(logdate || logtime, 'YYYY-MM-DD HH24:MI:SS')) rn
  7    from cmr a, abc b
  8   where message = 'HELD BY OPS'
  9     and  a.orderno=b.orderno
 10     and (SYSDATE - TO_DATE (logdate || logtime,'YYYY-MM-DD HH24:MI:SS')) * 24 >= 3
 11     ) where rn = 1
 12  /

FILENAME                                                         JOBDATE  HELD_TIME                 HR         RN
---------------------------------------------------------------- -------- ----------------- ---------- ----------
Maintenance                                                      20110623 23-06-11 13:02:31 188.188333          1
condition_test                                                   20110621 21-06-11 02:00:31 247.221667          1

need help

Shayam, July 02, 2011 - 2:18 am UTC

Thanks for reply and suggesstion...
i have held and free two times the file 'Maintenance' and 'condition_test' in 2 minute interval.then i run your query..for rn = 1,
it is giving first held time and for rn =2 it is is giving
latest held time for both file.
is there any way to get latest or max rownumber(rn)instead of giving manually 1 and 2 for rn because it can be many no of time hold and free ?
i want no of record with latest held time that exceeds 3 hr's associated with each filename.







Tom Kyte
July 05, 2011 - 7:44 am UTC

you need to explain this better. I don't understand your logic, your model, your data at all - I said as much above.

Start over, create tables, inserts into table, and provide A LOT of text that explains exactly what is what. Like a specification you would give a programmer to write code from

Shayam, July 05, 2011 - 10:26 am UTC

Take a example of emp table.

Query:
SELECT deptno, ename, rn
FROM (SELECT deptno, job, ename, sal,
ROW_NUMBER () OVER (PARTITION BY deptno, job ORDER BY sal,
ename) AS rn
FROM emp)
WHERE rn >= 1

Result:

DEPTNO ENAME RN
10 MILLER 1
10 CLARK 1
10 KING 1
20 FORD 1
20 SCOTT 2
20 SMITH 1
20 ADAMS 2
20 JONES 1
30 JAMES 1
30 BLAKE 1
30 MARTIN 1
30 WARD 2
30 TURNER 3
30 ALLEN 4

In this scenario , we know rn =4 is the max /latest row number in emp table so we will put WHERE rn >= 4 instead of WHERE rn >= 1 in sql query to get max row number row.

Suppose we donot know how many no of row in the table. And we need to take record only for max rownumber row.So how we can modify above sql query to get only max rownumber row.



Tom Kyte
July 05, 2011 - 4:00 pm UTC

this confuses me.


do you want the highest row_number from ALL deptnos/jobs
or
do you only want the highest row_number in any deptno/job

these answer that in turn:

ops$tkyte%ORA11GR2> select * from (
  2  SELECT deptno, ename, rn, max(rn) over() max_rn
  3    FROM (SELECT deptno, job, ename, sal,
  4                 ROW_NUMBER () OVER (PARTITION BY deptno, job ORDER BY sal,
  5                  ename) AS rn
  6            FROM emp)
  7  ) where rn = max_rn
  8  /

    DEPTNO ENAME              RN     MAX_RN
---------- ---------- ---------- ----------
        30 ALLEN               4          4


ops$tkyte%ORA11GR2> 
ops$tkyte%ORA11GR2> select *
  2    FROM (SELECT deptno, job, ename, sal,
  3                 ROW_NUMBER () OVER (PARTITION BY deptno, job ORDER BY sal DESC nulls last,
  4                  ename DESC nulls last) AS rn
  5            FROM emp)
  6  where rn = 1
  7  /

    DEPTNO JOB       ENAME             SAL         RN
---------- --------- ---------- ---------- ----------
        10 CLERK     MILLER           1300          1
           MANAGER   CLARK            2450          1
           PRESIDENT KING             5000          1

        20 ANALYST   SCOTT            3000          1
           CLERK     ADAMS            1100          1
           MANAGER   JONES            2975          1

        30 CLERK     JAMES             950          1
           MANAGER   BLAKE            2850          1
           SALESMAN  ALLEN            1600          1


9 rows selected.

Shayam, July 06, 2011 - 8:00 am UTC

Thanks a million

last row of a table based on priority

A reader, August 06, 2011 - 10:06 am UTC

I have a lookup table that has a composite unique key made up of 3 columns (a, b, c). Unique as any of the columns can be null and I need some help the query below;

select  *
from    ORDERS ord, LOOKUP l
where  (   l.a = ord.a and l.b = ord.b and l.c = ord.c 
        or l.a = ord.a and l.b = ord.b and l.c is null
        or l.a = ord.a and l.b is null and l.c = ord.c
        or l.a is null and l.b = ord.b and l.c = ord.c
        or l.a = ord.a and l.b is null and l.c is null
       )


The above query could return up to 5 rows for each distinct ORDER row - what I actually want to return is 1 ORDER row but the one with the highest LOOKUP "priority", and that "priority" is determined by the order of the WHERE clause above, e.g.

lookup row where l.a = ord.a and l.b = ord.b and l.c = ord.c is priority 1 (highest)
lookup row where l.a = ord.a and l.b = ord.b and l.c is null
is priority 2 (2nd highest)
lookup row where l.a = ord.a and l.b is null and l.c = ord.c
is priority 3 (3rd highest), etc

If you can help me with a SQL query first, that would be great.

Secondly, and I might not be able to change this for our project, do you think we can change the table design to make the query easier?
Tom Kyte
August 06, 2011 - 2:43 pm UTC

Secondly, and I might not be able to change this for our project, do you think we can change the table design to make the query easier?


Yes, I think you should change your data model, this is really sort of whacky - I'd have to understand the "requirement" (you point to data in a really strange manner, I'd have to really understand the motivation here - what real world purpose this solves - I cannot imagine this being useful in business).


how many rows do you expect to return from the outer query? do you use this to return "a row", "10 rows", "all rows" ?

last row of a table based on priority

A reader, August 07, 2011 - 3:52 am UTC

Thanks Tom.

I state "Secondly" as I'm doubtful I'll be able to get the architects to consider changing their design at this stage, so "Firstly" is for me to get the SQL to work with the design we have!

LOOKUP serves as some kind of validation for the business. However the design has been made so that the user does not have to enter every combination of the 3 codes in order for the validation to pass. e.g. if they've entered all 3 codes, then fine, but if the data only matches on 2, then that's fine as well, etc. So they've come up with these 5 different filters (in the WHERE clause).

So for example, the validation check code would be along the lines of;

select  *
from    ORDERS ord
where exists 
(select *
 from LOOKUP l
 where  (   l.a = ord.a and l.b = ord.b and l.c = ord.c 
         or l.a = ord.a and l.b = ord.b and l.c is null
         or l.a = ord.a and l.b is null and l.c = ord.c
         or l.a is null and l.b = ord.b and l.c = ord.c
         or l.a = ord.a and l.b is null and l.c is null
        )
)


Any existence of 5 rows to them passes the VALID check.

However, the client wants to report on ORDERS and also pull in other data from LOOKUP but only returning just 1 row back from LOOKUP, so they've decided that a priority based on the filter test is used (e.g. if l.a = ord.a and l.b = ord.b and l.c = ord.c then that is the row with the highest priority).

I understand the design appears wacky, but we have what we have and the solution I've ended up with so far is outer joining to LOOKUP 5 times in the select statement and a case expression round each report column;

select case when lup1.rowid is not null then lup1.value
            when lup2.rowid is not null then lup2.value...
from   orders o, 
       lookup lup1,lookup lup2,lookup lup3...


Not great, hence I've posted my query to see if there's a better way.

A reader, August 11, 2011 - 3:47 pm UTC

I should think that this query should solve for the first question. Sorry about the formatting, and you could do the two inner queries in one if you wanted, but your ROW_NUMBER function would be very hard to read.


select ord.a, ord.b, ord.c FROM (
select ord.a, ord.b, ord.c,
ROW_NUMBER() OVER (PARTITION BY ord.a, ord.b, ord.c ORDER BY DECODE(sort_key,'YYY',1,'YYN',2,'YNY',3,'NYY',4,'YNN',5,6) ASC) AS rowno
FROM
(select ord.a, ord.b, ord.c,
DECODE(l.a,NULL,'N','Y') || DECODE(l.b,NULL,'N','Y') || DECODE(l.c,NULL,'N','Y') AS sort_key
from ORDERS ord, LOOKUP l
where ( l.a = ord.a and l.b = ord.b and l.c = ord.c
or l.a = ord.a and l.b = ord.b and l.c is null
or l.a = ord.a and l.b is null and l.c = ord.c
or l.a is null and l.b = ord.b and l.c = ord.c
or l.a = ord.a and l.b is null and l.c is null
)))
WHERE rownum = 1;


thanks

A reader, August 11, 2011 - 4:20 pm UTC

thank you reader of Arizona, will give that a try

Selecting last record

Jobz, March 07, 2012 - 1:40 am UTC

Hi Tom,
Tried the method that you mentioned in the example. I got the corrct answer, when I ordered it in ASC order.

Tom Kyte
March 07, 2012 - 6:49 pm UTC

so, if you flip a coin five times and it comes up heads - do you conclude "coins, when flipped, come up heads"?

All I have to do is provide one counter case that demonstrates something isn't true. I don't care how many "positive" cases you have - as long as I have one case, I've shown "it is not true"

how about this,


show your work and I'll tweak it to show that even your test fails.

A Reader, March 26, 2012 - 11:52 pm UTC

Hi Tom,

my requirement is to fetch record of jobs which are running longer than 24 hours.

Description of table tco:
Name Null? Type
----------------------------------------- -------- ----------------------------
JOBNAME NOT NULL VARCHAR2(96)
STRTDATE NOT NULL VARCHAR2(24)
STRTTIME NOT NULL VARCHAR2(18)
ENDDATE NOT NULL VARCHAR2(24)
ENDTIME NOT NULL VARCHAR2(18)
STATUS NOT NULL VARCHAR2(3)

SQL Query used:
select jobname,status,to_char(to_date(max(STRTDATE) || max(STRTTIME), 'YYYY-MM-DD HH24:MI:SS'),'DD-MM-YY HH24:MI:SS') Start_Time,
trunc ((sysdate - to_char(to_date(max(STRTDATE) || max(STRTTIME), 'YYYY-MM-DD HH24:MI:SS'),'DD-MM-YY HH24:MI:SS'))*24,24) hour,
case
when status = 'A'then 'Executing'
else status end "Current_Status",
from tco where status = 'A'
group by jobname,status
having trunc (((sysdate - to_char(to_date(max(STRTDATE) || max(STRTTIME), 'YYYY-MM-DD HH24:MI:SS'),'DD-MM-YY HH24:MI:SS'))*24,24) > 24)

Getting Error:
ERROR at line 1:
ORA-00932: inconsistent datatypes: expected NUMBER got DATE

Can you please help on this.
Tom Kyte
March 27, 2012 - 7:49 am UTC

which rocket scientist decided this would be a brilliant idea?

STRTDATE                                  NOT NULL VARCHAR2(24)
 STRTTIME                                  NOT NULL VARCHAR2(18)
 ENDDATE                                   NOT NULL VARCHAR2(24)
 ENDTIME                                   NOT NULL VARCHAR2(18)


I'll never understand, not in a million years.

...
else status end "Current_Status",                                      
 from tco where status = 'A'    
...


that is *not* the query you used, see that "," there - it does not belong and with it we get:

 12  /
 from tco where status = 'A'
 *
ERROR at line 8:
ORA-00936: missing expression




if I take it out I get



 12  /
'YYYY-MM-DD HH24:MI:SS'),'DD-MM-YY HH24:MI:SS'))*24,24) > 24)
                                                   *
ERROR at line 11:
ORA-00907: missing right parenthesis



so, in short, you didn't bother to give us anything useful to work with :( Just sort of wasted our time.


A test case would have a create table and a select against it that *actually* reproduced the query.


If status = 'A' is active jobs, then I think all you want is a rather "simple" plain query isn't it?

ops$tkyte%ORA11GR2> select *
  2    from tco
  3   where status = 'A' -- running I presume, A for Active?
  4     and to_date( strtdate || strttime, 'yyyy-mm-dd hh24:mi:ss' ) < sysdate-1
  5  ;

no rows selected

A reader, March 30, 2012 - 12:26 am UTC

Hi Tom,

Thanks for valuable replying !!!
i have run the SQL query as you suggested below:
SQL query:-
select jobname,status from tco where status ='A' and jobname like 'SAP%' and to_date( strtdate || strttime, 'yyyy-mm-dd hh24:mi:ss' ) >= sysdate-1
Output:-
JOBNAME STA
--------------------------------------------------------------------------
SAP_SRT_METERING_REORG A
SAP_SRT_SEQ_DELETE_BGRFC_QUEUES A
SAP_SRT_SEQ_DELETE_TERM_SEQ A
SAP_SRT_SEQ_DELETE_TERM_SEQ A
SAP_SRT_SEQ_DELETE_TERM_SEQ A
SAP_SRT_SEQ_DELETE_TERM_SEQ A
SAP_SRT_SEQ_DELETE_TERM_SEQ A

7 rows selected.

but from SAPGUI i am getting 12 jobs are cencelled:

FYR:

Job overview from: 29.03.2012 at: 00:00:00
to: 30.03.2012 at: 23:59:59
Selected job names: SAP*
Selected user names: *

Scheduled Released Ready Active Finished Canceled
Event controlled Event ID:
ABAP program Program name :


Job Spool Job Doc Status Start date Start time Duration(sec.) Delay (sec.)

SAP_SRT_METERING_REORG Canceled 30.03.2012 03:00:42 0 42
SAP_SRT_SEQ_DELETE_BGRFC_QUEUES Canceled 29.03.2012 01:04:30 0 0
SAP_SRT_SEQ_DELETE_BGRFC_QUEUES Canceled 29.03.2012 02:04:31 0 1
SAP_SRT_SEQ_DELETE_BGRFC_QUEUES Canceled 30.03.2012 00:04:40 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 29.03.2012 00:04:30 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 29.03.2012 03:04:31 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 29.03.2012 04:04:31 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 29.03.2012 23:04:40 1 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 30.03.2012 01:04:41 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 30.03.2012 02:04:41 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 30.03.2012 03:04:42 0 0
SAP_SRT_SEQ_DELETE_TERM_SEQ Canceled 30.03.2012 04:04:43 0 0

*Summary 1 43

Could you please help me on this.
Tom Kyte
March 30, 2012 - 7:10 am UTC

so, you haven't bothered to read anything I wrote above?

I'll show you again:


so, in short, you didn't bother to give us anything useful to work with :( Just sort of wasted our time.



what the heck is "SAPGUI"
why do I care about cancelled jobs
where is a schema to work with
where is test data to play with
where is the query you are trying - one that actually is a query, not the stuff from above
where is the definition of all of the inputs
where is the definition of all of the outputs (logic needed to transform inputs into outputs - a full specification, a description)


come on.

How about order by?

Cornell, May 18, 2012 - 12:45 pm UTC

First, as you'd indicated, one needs to define "last". If it's the most recent entry, timestamp's the way to go. Can't use sequence numbers (as in NEXTVAL), as they're not necessarily in order.

I'm probably not using the right jargon, but when a database is clustered, groups of potential nextval values are handed out to each member of the cluster, which uses them as needed. That is, 1..10 might go to one, and 11..20 might go to another. Then three nextval calls on the first, second and first, would yield a sequence of 1,11,2.

Second, about the only way..

You'd indicated that this is the only way to get the "last" (most recent timestamp_column) row:
select * from t where TIMESTAMP_COLUMN = ( select max(timestamp_column) from T )
and rownum = 1;

That query would require going over the entire table to find the MAX, then going over it again to find those that match, if I'm not mistaken.

Wouldn't order by desc, get one row work as well?
select * from (
  select * from t 
   order by TIMESTAMP_COLUMN desc
  )
 where rownum = 1;

This query would sort the table, I don't know how expensive that is in comparison to MAX.

Testing in TOAD shows elapsed times that are fairly close, the second usually a tad faster. Unfortunately, my skills with determining efficiency are currently at the level of checking elapsed time.
Tom Kyte
May 21, 2012 - 8:19 am UTC

That query would require going over the entire table to find the MAX

it would not. Making the reasonable assumption that there was an index on timestamp_column, we'd use an index min/max scan to find the max value (about 3 IO's) and be done with it.


And any semantically equivalent query would suffice - that still fits in with the statement "the only way". You need to get the row having the maximum timestamp. I can come up with an infinite number of queries to "say that", but it is still "the only way".

Your query is technically not semantically equivalent since your query returns at most one row. My query returns all of the "last rows"

and as of the time of writing - it was version 8.0 - which did not have the top-n query optimization


testing in toad for response times from a query that is pretty much trivial (as this is) isn't going to do anything. Toad measures elapsed time, not database time. You are timing the amount of time it takes toad to paint the screen, deal with the data and so on.



And your query would not sort the table, assuming an index on timestamp_column - it would return as quickly as the first query.

Even if it did not have the index, your query would not sort the entire table, it would just have to basically scan the table. Top-N query optimization would make it so we'd read the first row - then we'd read the second row and only keep it if created was larger. And so on. We'd only have one row in memory at any point in time, we do not sort the entire table.



A better approach

Mark, July 13, 2012 - 9:11 am UTC

You absolutely do not have to use "order by" to get the first or last entry in a table. It's all in how you set the table up. If you set the table up with an auto_incrementing primary key, you can use the max of that key to get the lastest entry. Example:

select * from some_table where tables_primary_key = ( select max(table_primary_key) from some_table )

That should give you an almost instant reponse from the database and you won't have to sit there and wait on some "order by" sorting.
Tom Kyte
July 16, 2012 - 3:44 pm UTC

umm,

guess which will do less work:

select * from (select * from t order by indexed_column desc) where rownum = 1;

versus

select * from some_table where tables_primary_key = ( select
max(table_primary_key) from some_table )


Let's see:

big_table%ORA11GR2> set autotrace traceonly
big_table%ORA11GR2> select *
  2    from (select *
  3            from big_table
  4                   order by id desc )
  5   where rownum = 1
  6  /


Execution Plan
----------------------------------------------------------
Plan hash value: 3437067777

----------------------------------------------------------------------------------------------
| Id  | Operation                     | Name         | Rows  | Bytes | Cost (%CPU)| Time     |
----------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT              |              |     1 |   141 |     4   (0)| 00:00:01 |
|*  1 |  COUNT STOPKEY                |              |       |       |            |          |
|   2 |   VIEW                        |              |     1 |   141 |     4   (0)| 00:00:01 |
|   3 |    TABLE ACCESS BY INDEX ROWID| BIG_TABLE    |  1000K|    94M|     4   (0)| 00:00:01 |
|   4 |     INDEX FULL SCAN DESCENDING| BIG_TABLE_PK |     1 |       |     3   (0)| 00:00:01 |
----------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter(ROWNUM=1)


Statistics
----------------------------------------------------------
          0  recursive calls
          0  db block gets
          4  consistent gets
          0  physical reads
          0  redo size
       1345  bytes sent via SQL*Net to client
        420  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
          1  rows processed

big_table%ORA11GR2> select *
  2    from big_table
  3   where id = (select max(id) from big_table)
  4  /


Execution Plan
----------------------------------------------------------
Plan hash value: 2895345854

---------------------------------------------------------------------------------------------
| Id  | Operation                    | Name         | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT             |              |     1 |    99 |     6   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID | BIG_TABLE    |     1 |    99 |     3   (0)| 00:00:01 |
|*  2 |   INDEX UNIQUE SCAN          | BIG_TABLE_PK |     1 |       |     2   (0)| 00:00:01 |
|   3 |    SORT AGGREGATE            |              |     1 |     5 |            |          |
|   4 |     INDEX FULL SCAN (MIN/MAX)| BIG_TABLE_PK |     1 |     5 |     3   (0)| 00:00:01 |
---------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("ID"= (SELECT MAX("ID") FROM "BIG_TABLE" "BIG_TABLE"))


Statistics
----------------------------------------------------------
          0  recursive calls
          0  db block gets
          7  consistent gets
          0  physical reads
          0  redo size
       1345  bytes sent via SQL*Net to client
        420  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
          1  rows processed

big_table%ORA11GR2> set autotrace off


Oh, that would be the one that did that "nasty order by" - which doesn't have to sort.

Which of the two above queries do you believe will execute faster?

The first one that
a) reads an index and then
b) access a row in a table?

or the second one that
a) reads an index to get a max value and then
b) reads an index to get a rowid and then
c) reads a table


I'd rather hit the index once - how about you?



You don't have to "sit there and wait on some order by sorting" - the database is a tad bit more intelligent than that.

before you blast something or suggest you have a vastly superior approach - you should really

a) set up a test case
b) give the example
c) show the numbers

It saves time....

(you could have just read the review right above yours too, we had this discussion)


Or you could have read the original answer which included a few ways to do this... one of which was:

select * from t where TIMESTAMP_COLUMN = ( select max(timestamp_column) from T
)
and rownum = 1;


and the statement "it is the only way" referred to the fact you need some column in your table that identifies a row as being "the last", there is no concept of a last row.

solved my problem

Mubashar Ahmad, October 04, 2017 - 8:24 am UTC

i have a table having no pk and no timestamp. this solution helped me a lot. Thanks Tom.

Thanks for clarifying

Huang Jingheng, August 23, 2018 - 9:17 am UTC

I've been trying to use rownum to do that, its good that I now know that's a wrong track to follow.

Doesn't Fetches last record sometimes

Nisha, March 04, 2021 - 5:37 am UTC

select * from t where TIMESTAMP_COLUMN = ( select max(timestamp_column) from T )
and rownum = 1;

This query is not fetching lastest record sometimes. Usually when there is a lot of insert operation happening simultaneously in t table.Please suggestions any other alternative to fetch latest record using insert timestamp
Chris Saxon
March 04, 2021 - 9:08 am UTC

So how do you know it's not getting the latest row?

Bear in mind it's always possible that two rows have exactly the same insert time. When this happens there's no safe way to find which was last.

If the column has the date data type, this is likely as these are only granular to the second. Timestamps can have fractional seconds. With enough precision, it's less likely but still possible.

Please suggestions any other alternative

Please give more details on why you want to find the latest row.

Often this is because people want to know what values were inserted, e.g. for auto-generated PK values. In which case the returning clause is the way to go:

insert into ... 
returning ...

Chuck Jolley, March 04, 2021 - 9:39 pm UTC

This strikes me as the same problem as with trying to get gapless sequences. You can have one process inserting rows and give have gapless sequence numbers, or you can have multiple processes inserting rows and give up on guaranteed gapless sequences.
In the same way, there really isn't a "latest row" much of the time with multiple processes inserting in the table.
Which one is latest? Last one created in the client, last one inserted, last one committed? It's kind of Schrodinger's row.
Save the timestamp, select where row_number(order by timestamp desc) = 1 and call it the last row. If someone complains, challenge them to prove you are wrong.

Connor McDonald
March 05, 2021 - 4:41 am UTC

Yup.

The moment you run that query, its potentially out of date.

Fetching latest record

Nisha, March 05, 2021 - 2:23 pm UTC

I face this issue for when, I try to fetch the latest record from one table and insert it to other.

Example:

Table:item ( item_id, name, price, insert_timestamp)
Constraint - item_id is unique here.

Table:item_dtl ( item_id, name, quantity insert_timestamp)
Constraint - item_id is unquie here

Sequence is as follow:

Existing entries in table
Table : Item
Item_id|name|price|insert_timestamp

4|Monitor|10|12|3/22/2012 6:26:17.712300000 AM
3|CPU|20|100|3/22/2012 6:26:15.712300000 AM
2|Monitor|30|200|21/1/2012 5:27:15.712300000 AM
1|CPU|40|200|3/21/2012 5:26:15.712300000 AM

Table: item_ dtl
Item_id| name| quantity| insert_timestamp
4|Monitor|10|3/22/2012 6:27:17.712300000 AM
3|CPU|20|3/22/2012 6:27:15.712300000 AM
2|Monitor|30|21/1/2012 5:29:15.712300000 AM
1|CPU|40|3/21/2012 5:28:15.712300000 AM


Script 1 : Monitor Script

Insert into item values(5, 'Monitor',50, sysdate);

Insert into item_dtl(item_id,name, quantity,insert_timestamp) values (( select item_id from item where name=`Monitor' and insert_timestamp = ( select max( insert_timestamp) from item where name = 'Monitor ' ) and rownum = 1),( select price from item where name=`Monitor ' and insert_timestamp = ( select max( insert_timestamp) from item where name = 'Monitor ' ) and rownum = 1),100, sysdate);

Script 2 : CPU Script
Insert into item values (6, 'CPU', 60,sysdate);

Insert into item_dtl(item_id,name, quantity,insert_timestamp) values (( select item_id from item where name=`CPU' and insert_timestamp = ( select max( insert_timestamp) from item where name = 'CPU ' ) and rownum = 1),( select price from item where name=`CPU ' and insert_timestamp = ( select max( insert_timestamp) from item where name = 'CPU ' ) and rownum = 1),100, sysdate);

Similarly, we have 10 script and these are running simultaneously.
I get unique key constraint error when adding records in item_dtl.
Timestamp shows insert happened in item table before running the insert for item_dtl.

NOTE: the script fails only sometime, 20% failure.


I hope the above details are clear. Please let me know what i am doing wrong here.




Chris Saxon
March 05, 2021 - 5:18 pm UTC

So it looks like the returning clause is the way to go here!

Insert into item values(5, 'Monitor',50, sysdate)
returning id, price 
into l_id, l_price;

Insert into item_dtl 
values (l_id, l_price, 100, sysdate);


Similarly, we have 10 script and these are running simultaneously.

It's highly likely that the detail insert in one session is picking up an item inserted in another session from time-to-time. Use the returning clause as shown above to avoid this problem.

Doesn't Fetches last record sometimes

Nisha, March 05, 2021 - 5:37 pm UTC

Thanks Tom for the quick update.
Can a delay of 2s between 'insert into item' and 'insert into item_dtls' also work?
As this has been implemented in two different platforms (i.e)
Item table insertion is happening using Java program
Item_dts insertion is happening in shell script.

If delay doesn't work, then I will definitely restructure my current implementation and use returning as suggested.

Chris Saxon
March 08, 2021 - 2:17 pm UTC

Returning is the only safe way here; adding delays will only make the problem worse!

Thanks Tom!

Carol Benesh, April 09, 2021 - 4:41 pm UTC

You always provide great complete answers. Whenever I google and there is a link 'Ask Tom' I just know this link will explain what I need. Thank you.
Connor McDonald
April 12, 2021 - 5:17 am UTC

Thanks for the feedback