tags:

views:

2447

answers:

7

Is it possible to return the last row of a table in MS SQL Server. I am using an auto increment field for the ID and i want to get the last one just added to join it with something else. Any idea?

Here's the code:

const string QUERY = @"INSERT INTO Questions (ID, Question, Answer, CategoryID, Permission) " 
                   + @"VALUES (@ID, @Question, @Answer, @CategoryID, @Permission) "; 

using (var cmd = new SqlCommand(QUERY, conn)) 
{ 
    cmd.Parameters.AddWithValue("@Question", question); 
    cmd.Parameters.AddWithValue("@Answer", answer); 
    cmd.Parameters.AddWithValue("@CategoryID", lastEdited);
    cmd.Parameters.AddWithValue("@Permission", categoryID);
    cmd.ExecuteNonQuery(); 
}
+3  A: 
select top 1 * from yourtable order by id desc
Philippe Leybaert
Be careful. This looks right at first glance, but if you read the question closely there are problems with it.
Joel Coehoorn
True, but the question is not entirely clear. "joining with" could also mean finding related records to the last one in the table.
Philippe Leybaert
-1 this will return the last insert by any user, not the current's user last insert
Remus Rusanu
@Remus Rusanu: Nowhere in the question was there any mention that it should return the current user's last insert. But if you think my answer is wrong, your downvote was deserved I guess.
Philippe Leybaert
He says the he wants to join 'the one just added' with something else after an insert, like in typical master-detail inserts. It would be highly unusual to want to get a random row inserted by some other user.
Remus Rusanu
@Remus - the OP edited his question, initially he said he wanted to find the last added row in a table and join against it, he wasn't very specific. I think this was a simple answer to an initially simple question.
Stan R.
@Stan: I can't remove my -1 now unfortunately
Remus Rusanu
+9  A: 

use

  • scope_identity() returns the last identity value generated in this session and this scope
  • ident_current() returns the last identity value generated for a particular table in any session and any scope

    select ident_current( 'yourTableName' )

will return the last identity created by a different session.

Most of the time you should use scope_identity() right after an insert statement like so.

--insert statement
SET @id = CAST(SCOPE_IDENTITY() AS INT)
Stan R.
.........booo :(
Stan R.
Stan I didn't downvote- but it is OK to delete answers!
RichardOD
just edit your answer to stop the downvotes. hint: No one actually provided the tsql for using scope_identity().
dotjoe
i edited my answer, to reflect more of what the user is asking for.
Stan R.
+1 and i didn't even know about ident_current().
dotjoe
+22  A: 

Not safe - could have multiple inserts going on at the same time and the last row you'd get might not be yours. You're better off using SCOPE_IDENTITY() to get the last key assigned for your transaction.

n8wrl
Excellent point. When I posted I just read "gimme the last row". I ignored his intention to use it to create a foreign key reference.
Randolpho
+1 for providing the correct solution
dotjoe
+1: Correct if the client doesn't start a new transaction.
Philippe Leybaert
+1 very good point..now i wish people wouldn't downvote me because i misread the question.
Stan R.
+1. I was surprised (and worried) yours was the only answer that used this.
RichardOD
but scope identity gets me back a column of null values i want the last one not the count?
Ahmad Farid
A: 

With a simple select you can do something like this:

SELECT *
FROM table_name
WHERE IDColumn=(SELECT max(IDColum) FROM table_name)
Doliveras
unsafe in a concurrent environment......
marc_s
+16  A: 
Joel Coehoorn
+1 for mentioning scope_identity() is almost always the right one to use.
RichardOD
im surprised nobody mentioned identity_current, I posted some links and examples
Stan R.
but scope identity gets me back a column of null values i want the last one not the count?just added: the previous lines of codes were the ones that added this new row
Ahmad Farid
Don't include scope_identity() in the query. You run it after adding your new row and save the result in a variable, which you can then use to make the 'join', if that's even still required. Gimme a moment and I'll have an edit that demonstrates this.
Joel Coehoorn
TSQl code line 5:SELECT @NewOrderID=scope_identit()Should be:SELECT @NewOrderID=scope_identity()letter y is missing.
HollyStyles
I also had table spelled wrong, but that's what I get for typing directly into the reply window. Both are fixed now.
Joel Coehoorn
const string QUERY = @"INSERT INTO Questions (ID, Question, Answer, CategoryID, Permission) " + @"VALUES (@ID, @Question, @Answer, @CategoryID, @Permission) "; using (var cmd = new SqlCommand(QUERY, conn)) { cmd.Parameters.AddWithValue("@Question", question); cmd.Parameters.AddWithValue("@Answer", answer); cmd.Parameters.AddWithValue("@CategoryID", lastEdited); cmd.Parameters.AddWithValue("@Permission", categoryID); cmd.ExecuteNonQuery();
Ahmad Farid
can you check my code and tell me plz?
Ahmad Farid
thanks man for that full support. i tried ur code but i got this exception!! Specified cast is not valid.Description: An unhandled exception occurred during the execution of the current web request. Please review the stack trace for more information about the error and where it originated in the code.Exception Details: System.InvalidCastException: Specified cast is not valid.
Ahmad Farid
ok i tried it like that:NewQuestionID = Convert.ToInt32(cmd.ExecuteScalar()); instead and it worked !! weirdthank you very much man
Ahmad Farid
+1  A: 

I'm not sure of your version of SQL Server, but look for the OUTPUT clause of ther INSERT statement. You can capture a set of rows with this clause

KM
+1  A: 

Since the questioner is using .NET, here's a modified example of how to do it. (I removed ID from the insert list since it's autoincrement--the original example would fail. I also assume ID is an SQL int, not a bigint.)

        const string QUERY = @"INSERT INTO Questions (Question, Answer, CategoryID, Permission) "
                           + @"VALUES (@Question, @Answer, @CategoryID, @Permission);"
                           + @"SELECT @ID = SCOPE_IDENTITY();";

        using (var cmd = new SqlCommand(QUERY, conn)) 
        { 
            cmd.Parameters.AddWithValue("@Question", question); 
            cmd.Parameters.AddWithValue("@Answer", answer); 
            cmd.Parameters.AddWithValue("@CategoryID", lastEdited);
            cmd.Parameters.AddWithValue("@Permission", categoryID);
            cmd.Parameters.Add("@ID", System.Data.SqlDbType.Int).Direction = ParameterDirection.Output;
            cmd.ExecuteNonQuery();

            int id = (int)cmd.Parameters["@ID"].Value;
        }

EDITED: I also suggest considering LINQ to SQL instead of hand-coding SqlCommand objects--it's much better (faster to code, easier to use) for many common scenarios.

Ben M