2009-10-06 73 views
14

我想編寫一個SQL Server 2005存儲過程,它將選擇並返回用戶表中的用戶記錄,以獲取作爲參數傳遞給存儲過程的一些用戶標識。SQL:在存儲過程中的子句:如何傳遞值

如何做到這一點?

我可以將用戶ID作爲以逗號分隔的字符串傳遞。因此,我可以使用

select * 
from users 
where userid in (userids) 

例如, :我想選擇記錄爲id的5,6,7,8,9

如何編寫存儲過程?

+0

這是一個常見的問題。我在這裏有一個選項:http://stackoverflow.com/questions/977021/can-a-stored-procedure-have-dynamic-parameters-to-be-used-in-an-in-clause/977114#977114 – ScottE 2009-10-06 12:36:27

回答

17

對於SQL Server 2005中執行,檢查出厄蘭Sommarskog的優秀Arrays and Lists in SQL Server 2005文章這表明一些技巧如何處理SQL Server 2005中的列表和數組(他還有另一篇SQL Server 2000文章)。

如果能升級到SQL Server 2008中,可以使用被稱爲「表值參數」的新功能:

首先,創建一個用戶定義的表型

CREATE TYPE dbo.MyUserIDs AS TABLE (UserID INT NOT NULL) 

其次,使用該表格類型作爲參數:

CREATE PROC proc_GetUsers @UserIDTable MyUserIDs READONLY 
AS 
SELECT * FROM dbo.Users 
    WHERE userid IN (SELECT UserID FROM @UserIDTable) 

查看詳情here

Marc

7

你可以使用動態sql。通過變量傳遞中陳述到SQL SP和它連接成的SQL查詢和使用sp_execute SQL

create procedure myproc(@clause varchar(100)) as 
begin 
    exec sp_executesql 'select * from users where userid in (' + @clause +')' 
end 
+8

工作,但動態SQL有它的問題(安全,性能等) – 2009-10-06 11:58:26

3

假設T-SQL,你可以使用這個很好的函數(返回一個表)。

DROP FUNCTION sp_ConvertStringToTable 
GO 
CREATE FUNCTION sp_ConvertStringToTable(@list ntext) 
     RETURNS @tbl TABLE (Position INT IDENTITY(1, 1) NOT NULL, 
          Value INT NOT NULL) AS 
    BEGIN 
     DECLARE @pos  int, 
       @textpos int, 
       @chunklen smallint, 
       @str  nvarchar(4000), 
       @tmpstr nvarchar(4000), 
       @leftover nvarchar(4000) 

     SET @textpos = 1 
     SET @leftover = '' 
     WHILE @textpos <= datalength(@list)/2 
     BEGIN 
     SET @chunklen = 4000 - datalength(@leftover)/2 
     SET @tmpstr = ltrim(@leftover + substring(@list, @textpos, @chunklen)) 
     SET @textpos = @textpos + @chunklen 

     SET @pos = charindex(' ', @tmpstr) 
     WHILE @pos > 0 
     BEGIN 
      SET @str = substring(@tmpstr, 1, @pos - 1) 
      INSERT @tbl (Value) VALUES(convert(int, @str)) 
      SET @tmpstr = ltrim(substring(@tmpstr, @pos + 1, len(@tmpstr))) 
      SET @pos = charindex(' ', @tmpstr) 
     END 

     SET @leftover = @tmpstr 
     END 

     IF ltrim(rtrim(@leftover)) <> '' 
     INSERT @tbl (Value) VALUES(convert(int, @leftover)) 

     RETURN 
    END 
GO 

這樣:

SELECT * FROM Users 
WHERE userid IN 
(SELECT Value FROM sp_ConvertStringToTable('1 2 3')) 

您可以更改存儲的功能與逗號分隔字符串代替空格隔開的人一起工作。

如果你不想/不能使用存儲函數,你可以在需要的地方在存儲過程中包含它的代碼。

編輯:這是比字符串連接更令人難以置信的性能。

+0

你有這樣的統計數據?我會有興趣閱讀它們 – 2009-10-06 12:31:56

+0

這很慢,檢查了這一點:http://stackoverflow.com/questions/1456192/comparing-a-column-to-a-list-of-values-in-t-sql/1456404#1456404 – 2009-10-06 20:18:55

+0

哇!謝謝KM,做了一些測試,並將其替換爲現在的項目! 無論如何,我的「這是更快」的意思是指動態SQL。 – 2009-10-07 12:22:17

4

看到my previous answer to this

這是最好的來源:

http://www.sommarskog.se/arrays-in-sql.html

創建拆分功能,使用它像:

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

I prefer the number table approach

對於這種實現方法具d工作,你需要做的這一個時間表設置:

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) 

一旦表中設置了數字,創建此功能:

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 

您現在可以輕鬆地拆分CSV字符串轉換爲表和加入它:

select * from dbo.FN_ListToTable(',','1,2,3,,,4,5,6777,,,') 

OUTPUT:

ListValue 
----------------------- 
1 
2 
3 
4 
5 
6777 

(6 row(s) affected) 

你可以在一個CSV字符串傳遞到一個程序和過程僅排在給定的ID:

SELECT 
    y.* 
    FROM YourTable y 
     INNER JOIN dbo.FN_ListToTable(',',@GivenCSV) s ON y.ID=s.ListValue 
+0

我承認我已經使用了幾年了,但它回來咬我的屁股。這個塊的多語句版本(這裏:http://www.sommarskog.se/arrays-in-sql-2005.html#tblnum)確實快了兩倍。當涉及的表包含接近一百萬行時,問題才明顯。所以在使用@ KM的解決方案之前請考慮這一點。 – pkExec 2016-01-11 09:11:45

8

只要使用它像這樣將工作

Create procedure sp_DoctorList 
@userid varchar(100) 
as 
begin 
exec ('select * from doctor where userid in ('+ @userid +')') 
end 
+0

感謝您的建議隊友!只是** exec **部分在我的程序中爲我工作'insert into #temp_hourly exec('從ZONE選擇ZONE_ID,其中('+ @ZoneId +')組中的ZONE_ID由ZONE_ID')' – 2016-03-16 12:17:08

-1

你也可以用代替FIND_IN_SET 。見下面

create procedure myproc(IN in_user_ids varchar(100)) 
begin 
    select * from users where FIND_IN_SET(userid, in_user_ids); 
end 
+0

'FIND_IN_SET'不是SQL標準函數,並且它在Microsoft ** SQL Server **中不可用**(OP詢問) – 2015-06-15 12:07:37

0

查詢試試這個這個工作對我來說

DECLARE @InClause NVARCHAR(100) 
SET @InClause = 'tom,dick,harry' 
DECLARE @SafeInClause NVARCHAR(100) 
SET @SafeInClause = ',' + @InClause + ',' 
SELECT * FROM myTable WHERE PATINDEX(',' + myColumn + ',', @SafeInClause) > 0 
相關問題