ฉันจะสอบถามค่าในคอลัมน์ SQL Server XML ได้อย่างไร


127

ฉันมี XML ต่อไปนี้ที่เก็บไว้ในคอลัมน์ XML (เรียกว่าRoles) ในฐานข้อมูล SQL Server

<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>

ฉันต้องการแสดงรายการแถวทั้งหมดที่มีบทบาทเฉพาะในแถวนั้น บทบาทนี้ส่งผ่านโดยพารามิเตอร์

คำตอบ:


198
select
  Roles
from
  MyTable
where
  Roles.value('(/root/role)[1]', 'varchar(max)') like 'StringToSearchFor'

หน้าเหล่านี้จะแสดงข้อมูลเพิ่มเติมเกี่ยวกับวิธีการสืบค้น XML ใน T-SQL:

การสืบค้นฟิลด์ XML โดยใช้ t-sql

การแบนข้อมูล XML ใน SQL Server

แก้ไข

หลังจากที่เล่นกับมันนิด ๆ หน่อย ๆ ผมจบลงด้วยแบบสอบถามที่น่าตื่นตาตื่นใจนี้ที่ใช้CROSS ใช้ อันนี้จะค้นหาทุกแถว (บทบาท) สำหรับค่าที่คุณใส่ในนิพจน์ like ของคุณ ...

ด้วยโครงสร้างตารางนี้:

create table MyTable (Roles XML)

insert into MyTable values
('<root>
   <role>Alpha</role>
   <role>Gamma</role>
   <role>Beta</role>
</root>')

เราสามารถสอบถามได้ดังนี้:

select * from 

(select 
       pref.value('(text())[1]', 'varchar(32)') as RoleName
from 
       MyTable CROSS APPLY

       Roles.nodes('/root/role') AS Roles(pref)
)  as Result

where RoleName like '%ga%'

คุณสามารถตรวจสอบ SQL Fiddle ได้ที่นี่: http://sqlfiddle.com/#!18/dc4d2/1/0


5
ตอบทุกคำถามของฉัน[1]คำตอบของคุณทำอะไร?
Bistro

1
คำตอบที่ดีฉันโหวตให้อันนี้ แต่สตริงควรเป็น varchar ฉันเดา
AaA

7
@Bistro การถาม[1]เป็นคำถามที่ดีจริงๆ หมายความว่าคุณเลือกค่าบทบาทแรกจาก XML และนั่นหมายความว่าจะใช้ได้กับการค้นหาAlphaใน xml ตัวอย่างของคุณเท่านั้น Betaมันจะไม่พบแถวหากคุณค้นหา
Mikael Eriksson

1
ในกรณีของฉันฉันต้องค้นหาโหนดที่มีค่าแอตทริบิวต์เฉพาะ คำตอบนี้นำไปสู่การแก้ปัญหาของฉัน ฉันต้องใส่เครื่องหมายคำพูดคู่รอบค่าแอตทริบิวต์
John N

ถ้า XML มีเนมสเปซเราจะสอบถามได้อย่างไร?
FMFF

36
declare @T table(Roles xml)

insert into @T values
('<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>')

declare @Role varchar(10)

set @Role = 'Beta'

select Roles
from @T
where Roles.exist('/root/role/text()[. = sql:variable("@Role")]') = 1

หากคุณต้องการให้แบบสอบถามทำงานตามที่where col like '%Beta%'คุณสามารถใช้ได้contains

declare @T table(Roles xml)

insert into @T values
('<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>')

declare @Role varchar(10)

set @Role = 'et'

select Roles
from @T
where Roles.exist('/root/role/text()[contains(., sql:variable("@Role"))]') = 1

13

ถ้าชื่อฟิลด์ของคุณคือ Roles และชื่อตารางคือ table1 คุณสามารถใช้การค้นหาต่อไปนี้

DECLARE @Role varchar(50);
SELECT * FROM table1
WHERE Roles.exist ('/root/role = sql:variable("@Role")') = 1

นี่เป็นสิ่งที่ดีนี่คือวิธีใดในการค้นหาโดยใช้like? forexample /root/role like ....
Bistro

2
ใช้.value('(/root/role)[1]', 'varchar(max)') like '%yourtext%'แทนexistsเป็น Leniel อธิบาย
AAA

4
คุณลองสิ่งนี้แล้วหรือยัง? พบทุกอย่างไม่ว่าคุณจะใส่@Roleอะไรก็ตาม
Mikael Eriksson

6

ฉันคิดวิธีง่ายๆด้านล่างซึ่งง่ายต่อการจดจำด้วย :-)

select * from  
(select cast (xmlCol as varchar(max)) texty
 from myTable (NOLOCK) 
) a 
where texty like '%MySearchText%'

1
เราไม่ควรค้นหาด้วยการปรับแต่งสตริงเนื่องจากจะส่งผลให้การค้นหาช้าเกินไป
Malcolm Salvador

5

คุณสามารถทำสิ่งต่อไปนี้

declare @role varchar(100) = 'Alpha'
select * from xmltable where convert(varchar(max),xmlfield) like '%<role>'+@role+'</role>%'

เห็นได้ชัดว่านี่เป็นการแฮ็กเล็กน้อยและฉันจะไม่แนะนำให้ใช้วิธีแก้ปัญหาอย่างเป็นทางการ อย่างไรก็ตามฉันพบว่าเทคนิคนี้มีประโยชน์มากเมื่อทำการสืบค้น adhoc บนคอลัมน์ XML ใน SQL Server Management Studio สำหรับ SQL Server 2012


2

เคล็ดลับที่เป็นประโยชน์ ค้นหาค่าในคอลัมน์ SQL Server XML (XML พร้อมเนมสเปซ)

เช่น

Table [dbo].[Log_XML] contains columns Parametrs (xml),TimeEdit (datetime)

เช่น XML ใน Parametrs:

<ns0:Record xmlns:ns0="http://Integration"> 
<MATERIAL>10</MATERIAL> 
<BATCH>A1</BATCH> 
</ns0:Record>

เช่นแบบสอบถาม:

select
 Parametrs,TimeEdit
from
 [dbo].[Log_XML]
where
 Parametrs.value('(//*:Record/BATCH)[1]', 'varchar(max)') like '%A1%'
 ORDER BY TimeEdit DESC

1

ฉันใช้คำสั่งด้านล่างเพื่อดึงค่าใน XML ในตาราง Sql

with xmlnamespaces(default 'http://test.com/2008/06/23/HL.OnlineContract.ValueObjects')
select * from (
select
            OnlineContractID,
            DistributorID,
            SponsorID,
    [RequestXML].value(N'/OnlineContractDS[1]/Properties[1]/Name[1]', 'nvarchar(30)') as [Name]
   ,[RequestXML].value(N'/OnlineContractDS[1]/Properties[1]/Value[1]', 'nvarchar(30)') as [Value]
     ,[RequestXML].value(N'/OnlineContractDS[1]/Locale[1]', 'nvarchar(30)') as [Locale]
from [OnlineContract]) as olc
where olc.Name like '%EMAIL%' and olc.Value like '%EMAIL%' and olc.Locale='UK EN'

จะเกิดอะไรขึ้นถ้า XML ไม่มีนิยาม Namespace
Muflix

0

คุณสามารถค้นหาทั้งแท็กหรือเฉพาะค่าที่ระบุก็ได้ ที่นี่ฉันใช้สัญลักษณ์แทนสำหรับเนมสเปซ xml

declare @myDoc xml
set @myDoc = 
'<Root xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://stackoverflow.com">
    <Child>my value</Child>
 </Root>'

select @myDoc.query('/*:Root/*:Child') -- whole tag
select @myDoc.value('(/*:Root/*:Child)[1]', 'varchar(255)') -- only value
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.