Bạn cần một cái gì đó theo hình thức:
[column].modify('delete (//i[.=(1, 2)])') -- like SQL IN
-- or
[column].modify('delete (//i[.=1 or .=2])')
-- or
[column].modify('delete (//i[.=1], //i[.=2])')
-- or
[column].modify('delete (//i[contains("|1|2|",concat("|",.,"|"))])')
XQuery không hỗ trợ loại xml SQL trong SQL2005, và sửa đổi phương pháp duy nhất chấp nhận xâu (không biến cho phép).
Dưới đây là một hack xấu xí w/chứa hàm:
declare @table table ([column] xml)
insert @table ([column]) values ('<r><i>1</i><i>2</i><i>3</i></r>')
declare @parameter xml
set @parameter = '<r><i>1</i><i>2</i></r>'
-- build a pipe-delimited string
declare @in nvarchar(max)
set @in = convert(nvarchar(max),
@parameter.query('for $i in (/r/i) return concat(string($i),"|")')
)
set @in = '|'+replace(@in,'| ','|')
update @table set [column].modify ('
delete (//i[contains(sql:variable("@in"),concat("|",.,"|"))])
')
select * from @table
Đây là một w/SQL động:
-- replace table variable with temp table to get around variable scoping
if object_id('tempdb..#table') is not null drop table #table
create table #table ([column] xml)
insert #table ([column]) values ('<r><i>1</i><i>2</i><i>3</i></r>')
declare @parameter xml
set @parameter = '<r><i>1</i><i>2</i></r>'
-- we need dymamic SQL because the XML modify method only permits string literals
declare @sql nvarchar(max)
set @sql = convert(nvarchar(max),
@parameter.query('for $i in (/r/i) return concat(string($i),",")')
)
set @sql = substring(@sql,1,len(@sql)-1)
set @sql = 'update #table set [column].modify(''delete (//i[.=('[email protected]+')])'')'
print @sql
exec (@sql)
select * from #table
nếu bạn đang cập nhật biến xml chứ không phải là một cột, sử dụng sp_executesql và thông số đầu ra:
declare @xml xml
set @xml = '<r><i>1</i><i>2</i><i>3</i></r>'
declare @parameter xml
set @parameter = '<r><i>1</i><i>2</i></r>'
declare @sql nvarchar(max)
set @sql = convert(nvarchar(max),
@parameter.query('for $i in (/r/i) return concat(string($i),",")')
)
set @sql = substring(@sql,1,len(@sql)-1)
set @sql = 'set @xml.modify(''delete (//i[.=('[email protected]+')])'')'
exec sp_executesql @sql, N'@xml xml output', @xml output
select @xml
Phương pháp thay thế bằng cách sử dụng con trỏ để lặp qua xóa val ues, có lẽ kém hiệu quả do nhiều cập nhật:
declare @table table ([column] xml)
insert @table ([column]) values ('<r><i>1</i><i>2</i><i>3</i></r>')
declare @parameter xml
set @parameter = '<r><i>1</i><i>2</i></r>'
/*
-- unfortunately, this doesn't work:
update t set [column].modify('delete (//i[.=sql:column("p.i")])')
from @table t, (
select i.value('.', 'nvarchar')
from @parameter.nodes('//i') a (i)
) p (i)
select * from @table
*/
-- so we have to use a cursor
declare @cursor cursor
set @cursor = cursor for
select i.value('.', 'varchar') as i
from @parameter.nodes('//i') a (i)
declare @i int
open @cursor
while 1=1 begin
fetch next from @cursor into @i
if @@fetch_status <> 0 break
update @table set [column].modify('delete (//i[.=sql:variable("@i")])')
end
select * from @table
Thông tin thêm về những hạn chế của các biến XML trong SQL2005 ở đây:
http://blogs.msdn.com/denisruc/archive/2006/05/17/600250.aspx
Có ai có một cách tốt hơn?