如何删除具有不同ID的多行数据?

56

我想要做这样的事情:

DELETE FROM table WHERE id IN (SELECT ....)

我该如何做到这一点?


什么数据库?在大多数情况下,你描述的应该可以正常工作。 - Thilo
9
您需要翻译的内容是:“DELETE FROM table WHERE id IN (1,2,3,4)DELETE FROM table WHERE id IN (SELECT id FROM table2 WHERE X=Y)”,请问您具体想了解什么信息? - Tim Schmelter
可能是 如何在SQL中删除id =(x到y)的多行 的重复问题。 - dodexahedron
这个问题实际上不值得点赞,因为提问者显然甚至没有尝试过或者懒得去谷歌搜索。更让人痛心的是,它是一个重复的问题,与https://dev59.com/22Qo5IYBdhLWcg3wdPO8?rq=1相同。 - dodexahedron
它是三年前提出的,怎么可能是重复的呢? - ave4496
显示剩余4条评论
4个回答

112

如果您需要选择ID:

 DELETE FROM table WHERE id IN (SELECT id FROM somewhere_else)
如果您已经知道它们(且不超过数千个):
 DELETE FROM table WHERE id IN (?,?,?,?,?,?,?,?)

5
如果数量达到了数千,那会对性能产生很大影响吗?我试图删除一些重复项,但有些地方有2-3个重复项。对于其他条目,我可能有几百个或几千个重复项。 - Zeliax
刚刚确认了这个 DELETE FROM user-access WHERE user-id=5 AND article-id IN (14,7,8) 操作,并想补充说明一下,数据库并不在意 article-id 为 8 的行不存在,它只是执行了操作。 - clockw0rk

2
< p>免责声明:以下建议根据情况可能会增加负担。该功能仅经过MSSQL 2008 R2测试,但似乎与其他版本兼容

如果您想使用许多ID进行此操作,您可以使用函数创建一个临时表,在其中您将能够从选择中删除

查询可能如下所示:

-- not tested
-- @ids will contain a varchar with your ids e.g.'9 12 27 37'
DELETE FROM table WHERE id IN (SELECT i.number FROM iter_intlist_to_tbl(@ids))

这是一个函数:

ALTER FUNCTION iter_intlist_to_tbl (@list nvarchar(MAX))
   RETURNS @tbl TABLE (listpos int IDENTITY(1, 1) NOT NULL,
                       number  int NOT NULL) AS

   -- funktion gefunden auf http://www.sommarskog.se/arrays-in-sql-2005.html
   -- dient zum übergeben einer liste von elementen

BEGIN
    -- Deklaration der Variablen
    DECLARE @startpos int,
            @endpos   int,
            @textpos  int,
            @chunklen smallint,
            @str      nvarchar(4000),
            @tmpstr   nvarchar(4000),
            @leftover nvarchar(4000)

    -- Startwerte festlegen
   SET @textpos = 1
   SET @leftover = ''

   -- Loop 1
    WHILE @textpos <= datalength(@list) / 2
    BEGIN

        --
        SET @chunklen = 4000 - datalength(@leftover) / 2 --datalength() gibt die anzahl der bytes zurück (mit Leerzeichen)

        --
        SET @tmpstr = ltrim(@leftover + substring(@list, @textpos, @chunklen))--SUBSTRING ( @string ,start , length ) | ltrim(@string) abschneiden aller Leerzeichen am Begin des Strings

        --hochzählen der TestPosition
        SET @textpos = @textpos + @chunklen

        --start position 0 setzen
        SET @startpos = 0

        -- end position bekommt den charindex wo ein [LEERZEICHEN] gefunden wird
        SET @endpos = charindex(' ' COLLATE Slovenian_BIN2, @tmpstr)--charindex(searchChar,Wo,Startposition)

        -- Loop 2
        WHILE @endpos > 0
        BEGIN
            --str ist der string welcher zwischen den [LEERZEICHEN] steht
            SET @str = substring(@tmpstr, @startpos + 1, @endpos - @startpos - 1) 

            --wenn @str nicht leer ist wird er zu int Convertiert und @tbl unter der Spalte 'number' hinzugefügt
            IF @str <> ''
                INSERT @tbl (number) VALUES(convert(int, @str))-- convert(Ziel-Type,Value)

            -- start wird auf das letzte bekannte end gesetzt
            SET @startpos = @endpos

            -- end position bekommt den charindex wo ein [LEERZEICHEN] gefunden wird
            SET @endpos = charindex(' ' COLLATE Slovenian_BIN2, @tmpstr, @startpos + 1)
        END
        -- Loop 2

        -- dient dafür den letzten teil des strings zu selektieren
        SET @leftover = right(@tmpstr, datalength(@tmpstr) / 2 - @startpos)--right(@string,anzahl der Zeichen) bsp.: right("abcdef",3) => "def"
    END
    -- Loop 1

    --wenn @leftover nach dem entfernen aller [LEERZEICHEN] nicht leer ist wird er zu int Convertiert und @tbl unter der Spalte 'number' hinzugefügt
    IF ltrim(rtrim(@leftover)) <> ''
        INSERT @tbl (number) VALUES(convert(int, @leftover))

    RETURN
END


    -- ############################ WICHTIG ############################
    -- das is ein Beispiel wie man die Funktion benutzt
    --
    --CREATE    PROCEDURE get_product_names_iter 
    --      @ids varchar(50) AS
    --SELECT    P.ProductName, P.ProductID
    --FROM      Northwind.Products P
    --JOIN      iter_intlist_to_tbl(@ids) i ON P.ProductID = i.number
    --go
    --EXEC get_product_names_iter '9 12 27 37'
    --
    -- Funktion gefunden auf http://www.sommarskog.se/arrays-in-sql-2005.html
    -- dient zum übergeben einer Liste von Id's
    -- ############################ WICHTIG ############################

2
Delete from BA_CITY_MASTER where CITY_NAME in (select CITY_NAME from BA_CITY_MASTER group by CITY_NAME having count(CITY_NAME)>1);

你应该通过选择代码并按Ctrl-K进行格式化。此外,最好提供一些解释,或许建议一下问题中的代码有什么问题(如果有的话)。 - Tom Fenech
1
这是一个很好的解决方案:DELETE FROM [TableName] WHERE [ColumnName] IN (SELECT [ColumnName] FROM [TableName] GROUP BY [ColumnName] HAVING COUNT([ColumnName])>1); - dakab

-4
  • 您可以制作这个。

    CREATE PROC [dbo].[sp_DELETE_MULTI_ROW]
    @CODE XML ,@ERRFLAG CHAR(1) = '0' OUTPUT

AS

SET NOCOUNT ON
SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED

DELETE tb_SampleTest WHERE CODE IN( SELECT Item.value('.', 'VARCHAR(20)') FROM @CODE.nodes('RecordList/ID') AS x(Item) )

IF @@ROWCOUNT = 0 SET @ERRFLAG = 200

SET NOCOUNT OFF

  • <'RecordList'><'ID'>1<'/ID'><'ID'>2<'/ID'><'/RecordList'>

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接