Delete all but top n from database table in SQL

Sql

Sql Problem Overview


What's the best way to delete all rows from a table in sql but to keep n number of rows on the top?

Sql Solutions


Solution 1 - Sql

DELETE FROM Table WHERE ID NOT IN (SELECT TOP 10 ID FROM Table)

Edit:

Chris brings up a good performance hit since the TOP 10 query would be run for each row. If this is a one time thing, then it may not be as big of a deal, but if it is a common thing, then I did look closer at it.

Solution 2 - Sql

I would select ID column(s) the set of rows that you want to keep into a temp table or table variable. Then delete all the rows that do not exist in the temp table. The syntax mentioned by another user:

DELETE FROM Table WHERE ID NOT IN (SELECT TOP 10 ID FROM Table)

Has a potential problem. The "SELECT TOP 10" query will be executed for each row in the table, which could be a huge performance hit. You want to avoid making the same query over and over again.

This syntax should work, based what you listed as your original SQL statement:

create table #nuke(NukeID int)

insert into #nuke(Nuke) select top 1000 id from article

delete article where not exists (select 1 from nuke where Nukeid = id)

drop table #nuke

Solution 3 - Sql

Future reference for those of use who don't use MS SQL.

In PostgreSQL use ORDER BY and LIMIT instead of TOP.

DELETE FROM table
WHERE id NOT IN (SELECT id FROM table ORDER BY id LIMIT n);

MySQL -- well...

> Error -- This version of MySQL does not yet support 'LIMIT & > IN/ALL/ANY/SOME subquery'

Not yet I guess.

Solution 4 - Sql

Here is how I did it. This method is faster and simpler:

Delete all but top n from database table in MS SQL using OFFSET command

WITH CTE AS
    (
    SELECT  ID
    FROM    dbo.TableName
    ORDER BY ID DESC
    OFFSET 11 ROWS
    )
DELETE CTE;

Replace ID with column by which you want to sort. Replace number after OFFSET with number of rows which you want to keep. Choose DESC or ASC - whatever suits your case.

Solution 5 - Sql

I think using a virtual table would be much better than an IN-clause or temp table.

DELETE 
    Product
FROM
    Product
    LEFT OUTER JOIN
    (
	    SELECT TOP 10
	    	Product.id
	    FROM
	    	Product
	) TopProducts ON Product.id = TopProducts.id
WHERE
	TopProducts.id IS NULL

Solution 6 - Sql

This really is going to be language specific, but I would likely use something like the following for SQL server.

declare @n int
SET @n = SELECT Count(*) FROM dTABLE;
DELETE TOP (@n - 10 ) FROM dTable

if you don't care about the exact number of rows, there is always

DELETE TOP 90 PERCENT FROM dTABLE;

Solution 7 - Sql

I don't know about other flavors but MySQL DELETE allows LIMIT.

If you could order things so that the n rows you want to keep are at the bottom, then you could do a DELETE FROM table LIMIT tablecount-n.

Edit

Oooo. I think I like Cory Foy's answer better, assuming it works in your case. My way feels a little clunky by comparison.

Solution 8 - Sql

I would solve it using the technique below. The example expect an article table with an id on each row.

Delete article where id not in (select top 1000 id from article)

Edit: Too slow to answer my own question ...

Solution 9 - Sql

Refactored?

Delete a From Table a Inner Join (
    Select Top (Select Count(tableID) From Table) - 10) 
        From Table Order By tableID Desc
) b On b.tableID = A.tableID

edit: tried them both in the query analyzer, current answer is fasted (damn order by...)

Solution 10 - Sql

Better way would be to insert the rows you DO want into another table, drop the original table and then rename the new table so it has the same name as the old table

Solution 11 - Sql

I've got a trick to avoid executing the TOP expression for every row. We can combine TOP with MAX to get the MaxId we want to keep. Then we just delete everything greater than MaxId.

-- Declare Variable to hold the highest id we want to keep.	
DECLARE @MaxId as int = (
SELECT MAX(temp.ID)
FROM (SELECT TOP 10 ID FROM table ORDER BY ID ASC) temp
)

-- Delete anything greater than MaxId. If MaxId is null, there is nothing to delete.
IF @MaxId IS NOT NULL
	DELETE FROM table WHERE ID > @MaxId

Note: It is important to use ORDER BY when declaring MaxId to ensure proper results are queried.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionRiriView Question on Stackoverflow
Solution 1 - SqlCory FoyView Answer on Stackoverflow
Solution 2 - SqlChris MillerView Answer on Stackoverflow
Solution 3 - SqlSimurrView Answer on Stackoverflow
Solution 4 - SqlHrvojeView Answer on Stackoverflow
Solution 5 - SqlTim WilsonView Answer on Stackoverflow
Solution 6 - SqlNoahView Answer on Stackoverflow
Solution 7 - SqlMark BiekView Answer on Stackoverflow
Solution 8 - SqlRiriView Answer on Stackoverflow
Solution 9 - SqlShawnView Answer on Stackoverflow
Solution 10 - SqlSQLMenaceView Answer on Stackoverflow
Solution 11 - SqlclamchodaView Answer on Stackoverflow