我想编写一个SQL Server 2005存储过程,它将选择并返回用户表中的用户记录,以获取作为参数传递给存储过程的一些用户标识。SQL:在存储过程中的子句:如何传递值
如何做到这一点?
我可以将用户ID作为以逗号分隔的字符串传递。因此,我可以使用
select *
from users
where userid in (userids)
例如, :我想选择记录为id的5,6,7,8,9
如何编写存储过程?
我想编写一个SQL Server 2005存储过程,它将选择并返回用户表中的用户记录,以获取作为参数传递给存储过程的一些用户标识。SQL:在存储过程中的子句:如何传递值
如何做到这一点?
我可以将用户ID作为以逗号分隔的字符串传递。因此,我可以使用
select *
from users
where userid in (userids)
例如, :我想选择记录为id的5,6,7,8,9
如何编写存储过程?
对于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
你可以使用动态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
工作,但动态SQL有它的问题(安全,性能等) – 2009-10-06 11:58:26
假设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'))
您可以更改存储的功能与逗号分隔字符串代替空格隔开的人一起工作。
如果你不想/不能使用存储函数,你可以在需要的地方在存储过程中包含它的代码。
编辑:这是比字符串连接更令人难以置信的性能。
你有这样的统计数据?我会有兴趣阅读它们 – 2009-10-06 12:31:56
这很慢,检查了这一点:http://stackoverflow.com/questions/1456192/comparing-a-column-to-a-list-of-values-in-t-sql/1456404#1456404 – 2009-10-06 20:18:55
哇!谢谢KM,做了一些测试,并将其替换为现在的项目! 无论如何,我的“这是更快”的意思是指动态SQL。 – 2009-10-07 12:22:17
这是最好的来源:
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
我承认我已经使用了几年了,但它回来咬我的屁股。这个块的多语句版本(这里:http://www.sommarskog.se/arrays-in-sql-2005.html#tblnum)确实快了两倍。当涉及的表包含接近一百万行时,问题才明显。所以在使用@ KM的解决方案之前请考虑这一点。 – pkExec 2016-01-11 09:11:45
只要使用它像这样将工作
Create procedure sp_DoctorList
@userid varchar(100)
as
begin
exec ('select * from doctor where userid in ('+ @userid +')')
end
感谢您的建议队友!只是** exec **部分在我的程序中为我工作'insert into #temp_hourly exec('从ZONE选择ZONE_ID,其中('+ @ZoneId +')组中的ZONE_ID由ZONE_ID')' – 2016-03-16 12:17:08
你也可以用代替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
'FIND_IN_SET'不是SQL标准函数,并且它在Microsoft ** SQL Server **中不可用**(OP询问) – 2015-06-15 12:07:37
查询试试这个这个工作对我来说
DECLARE @InClause NVARCHAR(100)
SET @InClause = 'tom,dick,harry'
DECLARE @SafeInClause NVARCHAR(100)
SET @SafeInClause = ',' + @InClause + ','
SELECT * FROM myTable WHERE PATINDEX(',' + myColumn + ',', @SafeInClause) > 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