2010-04-06 39 views
5

Tôi đang tạo một quy trình được lưu trữ mà tôi muốn chuyển như biến danh sách các dấu phẩy được phân cách bằng dấu phẩy. Tôi muốn sử dụng Id vào một tuyên bố chọn, một cái gì đó như:SQL - Thủ tục được lưu trữ với Chọn Tuyên bố bằng cách sử dụng IN (@Variable_CommaDelimitedListOfIDS)

Create Procedure up_TEST 
@Ids VARCHAR(MAX) 
AS 
SELECT * FROM ATable a 
WHERE a.Id IN(@Ids) 

Rõ ràng tôi nhận được lỗi mà @Ids là một varchar và không phải là một INT nhưng làm thế nào tôi có thể chuyển đổi các danh sách dấu phẩy phân cách?

Tôi đang sử dụng SQL Server 2008

Cảm ơn

Trả lời

2

Đối với những trường hợp tôi sử dụng chức năng này bảng, mà bạn có thể thích ứng với nhu cầu của bạn:

CREATE FUNCTION dbo.f_params_to_list (@par VARCHAR(500)) 
returns @result TABLE (value VARCHAR(30)) 
AS 
begin 
    DECLARE @TempList table 
      (
      value VARCHAR(30) 
     ) 

    DECLARE @Value varchar(30), @Pos int 

    SET @par = LTRIM(RTRIM(@par))+ ',' 
    SET @Pos = CHARINDEX(',', @par, 1) 

    IF REPLACE(@par, ',', '') <> '' 
    BEGIN 
      WHILE @Pos > 0 
      BEGIN 
       SET @Value = LTRIM(RTRIM(LEFT(@par, @Pos - 1))) 
       IF @Value <> '' 
       BEGIN 
        INSERT INTO @TempList (value) VALUES (@Value) --Use Appropriate conversion 
       END 
       SET @par = RIGHT(@par, LEN(@par) - @Pos) 
       SET @Pos = CHARINDEX(',', @par, 1) 

      END 
    END  
    INSERT @result 
    SELECT value 
     FROM @TempList 
    RETURN 
END  

Trong thủ tục lưu trữ của bạn, bạn sẽ sử dụng nó như thế này:

Create Procedure up_TEST 
@Ids VARCHAR(MAX) 
AS 
SELECT * FROM ATable a 
WHERE a.Id IN(SELECT value FROM dbo.f_params_to_list(@Ids)) 
0

Một cách có thể được phân chia và chèn id vào một bảng tempory và hơn SQL sử dụng tại khoản ...

1

Nếu bạn gặp sự cố này, bạn nên đọc các bài viết của Sommarskog.

http://www.sommarskog.se/index.html

Tôi khuyên bạn nên:

  • Mảng và Danh sách trong Sql Server
  • Lời nguyền và phước lành của SQL động.
4

Vì bạn đang sử dụng SQL Server 2008, hãy xem table-valued parameters.

+0

@TGnat, người đàn ông tuyệt vời, cảm ơn mẹo! +1 –

+0

+1 - nếu bạn muốn xem so sánh hiệu suất của các kỹ thuật khác nhau mà bạn có thể thực hiện (thông số có giá trị so với CSV so với TABLE so với XML), tôi đã viết blog về điều này cách đây không lâu: http://www.adathedev.co.uk /2010/02/sql-server-2008-table-valued-parameters.html – AdaTheDev

+1

@AdaTheDev, dựa trên liên kết của bạn, tại sao tạo loại thông số bảng mới cho CustomerID, như 'CREATE TYPE CustomerIDTableType AS TABLE (ID INTEGER PRIMARY KEY) ; 'nó có thể tái sử dụng nhiều hơn để chỉ làm điều này:' TẠO LOẠI LOẠI IntTableType AS TABLE (ID INTEGER PRIMARY KEY); 'và sử dụng nó để chuyển vào bất kỳ danh sách các số nguyên –

2

Sử dụng tham số giá trị bảng (mới trong SQl Server 2008). Đặt nó lên bằng cách tạo ra các loại tham số bảng thực tế:

CREATE TYPE IntTableType AS TABLE (ID INTEGER PRIMARY KEY) 

thủ tục của bạn sau đó sẽ là:

Create Procedure up_TEST 
    @Ids IntTableType READONLY 
AS 

SELECT * 
    FROM ATable a 
    WHERE a.Id IN (SELECT ID FROM @Ids) 

RETURN 0 
GO 

nếu bạn không thể sử dụng các thông số giá trị bảng, xem: "Arrays and Lists in SQL Server 2008 Using Table-Valued Parameters" by Erland Sommarskog, sau đó có rất nhiều cách để tách chuỗi trong SQL Server. Bài viết này bao gồm các ưu và nhược điểm của chỉ là về mọi phương pháp:

"Arrays and Lists in SQL Server 2005 and Beyond, When Table Value Parameters Do Not Cut it" by Erland Sommarskog

Bạn cần phải tạo một chức năng phân chia. Đây là cách một chức năng phân chia có thể được sử dụng:

SELECT 
    * 
    FROM YourTable        y 
    INNER JOIN dbo.yourSplitFunction(@Parameter) s ON y.ID=s.Value 

I prefer the number table approach to split a string in TSQL nhưng có rất nhiều cách để tách chuỗi trong SQL Server, xem liên kết trước đó, điều này giải thích các ưu và nhược điểm của mỗi người.

Đối với phương pháp số Bảng để làm việc, bạn cần làm thiết lập bảng này một thời gian, mà sẽ tạo ra một bảng Numbers chứa hàng từ 1 đến 10,000:

SELECT TOP 10000 IDENTITY(int,1,1) AS Number 
    INTO Numbers 
    FROM sys.objects s1 
    CROSS JOIN sys.objects s2 
ALTER TABLE Numbers ADD CONSTRAINT PK_Numbers PRIMARY KEY CLUSTERED (Number) 

Khi bảng số được thiết lập , tạo ra chức năng phân chia này:

CREATE FUNCTION [dbo].[FN_ListToTable] 
(
    @SplitOn char(1)  --REQUIRED, the character to split the @List string on 
    ,@List  varchar(8000)--REQUIRED, the list to split apart 
) 
RETURNS TABLE 
AS 
RETURN 
(

    ---------------- 
    --SINGLE QUERY-- --this will not return empty rows 
    ---------------- 
    SELECT 
     ListValue 
     FROM (SELECT 
        LTRIM(RTRIM(SUBSTRING(List2, number+1, CHARINDEX(@SplitOn, List2, number+1)-number - 1))) AS ListValue 
        FROM (
          SELECT @SplitOn + @List + @SplitOn AS List2 
         ) AS dt 
         INNER JOIN Numbers n ON n.Number < LEN(dt.List2) 
        WHERE SUBSTRING(List2, number, 1) = @SplitOn 
      ) dt2 
     WHERE ListValue IS NOT NULL AND ListValue!='' 

); 
GO 

Bạn có thể dễ dàng chia một chuỗi CSV vào một bảng và tham gia vào nó:

Create Procedure up_TEST 
@Ids VARCHAR(MAX) 
AS 
SELECT * FROM ATable a 
WHERE a.Id IN (SELECT ListValue FROM dbo.FN_ListToTable(',',@Ids)) 
1

Mọi người dường như đang làm điều gì đó như hôm nay!

Hãy xem http://www.sqlteam.com/article/parsing-csv-values-into-multiple-rows - đây là cách xử lý biến đầu vào được phân tách để cung cấp tập hợp kết quả có chứa các mục chia tách bằng cách tham gia vào bảng số/đếm (đó là điều tốt để có trong cơ sở dữ liệu bạn chưa có). Điều này sau đó cung cấp cho bạn một tập hợp kết quả đơn giản mà bạn có thể tham gia chống lại như bình thường.

1
CREATE PROCEDURE PROCEDURENAME 
@Id Numeric(18,0) 
AS 
BEGIN 
    SELECT * FROM tableName 
    WHERE tableName.Id IN (@Id) 
END 
+0

Chào mừng bạn đến với stackoverflow. Tôi nghĩ bạn đã hiểu sai câu hỏi. Ở trên sẽ không làm việc với một danh sách * nhiều * id (Đây cũng là một câu hỏi cũ đã được trả lời :). – Leigh

Các vấn đề liên quan