我有一个表格,它定义了 XML 表单的布局。它<Control ...>
内部的节点具有 Id (GUID) 和 DataType (char) 等属性...
此 XML 数据用于在运行时创建表单,当保存数据时,将其写入 XML 中称为元素<Field ...>
的具有以下属性的元素:名称(与控制节点 ID 中的 GUID 匹配)和数据(保存之前的值)输入控制)。
我们有一个问题,当您 .ToString 一个 Date 对象时,它使用机器的日期格式设置。所以日期可以以任何可能的格式保存。我已更新代码以始终以YYYY/MM/DD
格式保存日期,但现在我需要将数据库中的现有数据更新为YYYY/MM/DD
. 由于我们无法知道日期的保存格式,因此我们假设它保存为MM/DD/YYYY
.
所以现在我的问题是尝试更新 SQL Server 中的 XML。使用 xquery 和 CROSS APPLY .nodes( ) 东西,我可以编写一个查询来查找所有格式不正确的日期,但我不知道如何更新它们。
此 SQL 脚本将创建表变量并用一些测试数据填充它们,并具有将返回错误日期值的查询。注释掉的最后一部分是我如何尝试将它们更新为新YYYY/MM/DD
格式,但是如果您取消注释,您会发现它不起作用。
有人有想法么?
PS 这只能在 SQL 脚本中完成。我知道在 .NET 中编写一个读取所有 xml 数据并更新属性然后将数据保存回数据库的函数非常容易,但在这种情况下这是不可能的,因为:原因。
DECLARE @Form AS Table
(
FormID INT,
FormDataXML XML
);
DECLARE @ScreenData AS Table
(
ScreenDataID INT,
FormID INT,
ScreenDataXML XML
);
DECLARE @ControlsToUpdate AS TABLE
(
FormID INT,
ControlID char(36),
DataType CHAR(1)
);
INSERT INTO @Form
VALUES (1, '<Form><Control Id="00000000-0000-0000-0000-000000000000" DataType="D" /><Control Id="00000000-0000-0000-0000-000000000001" DataType="N" /></Form>');
INSERT INTO @Form
VALUES (2, '<Form><Control Id="00000000-0000-0000-0000-000000000002" DataType="D" /><Control Id="00000000-0000-0000-0000-000000000003" DataType="D" /></Form>');
INSERT INTO @ScreenData
VALUES (1, 1, '<ScreenData><Field Name="00000000-0000-0000-0000-000000000000" Data="01/31/2012" /><Field Name="00000000-0000-0000-0000-000000000001" Data="1234.56" /></ScreenData>');
INSERT INTO @ScreenData
VALUES (2, 1, '<ScreenData><Field Name="00000000-0000-0000-0000-000000000000" Data="02/28/2013" /><Field Name="00000000-0000-0000-0000-000000000001" Data="0" /></ScreenData>');
INSERT INTO @ScreenData
VALUES (3, 2, '<ScreenData><Field Name="00000000-0000-0000-0000-000000000002" Data="03/31/2013" /><Field Name="00000000-0000-0000-0000-000000000003" Data="04/30/2013" /></ScreenData>');
INSERT INTO @ScreenData
VALUES (4, 2, '<ScreenData><Field Name="00000000-0000-0000-0000-000000000002" Data="2013/05/31" /><Field Name="00000000-0000-0000-0000-000000000003" Data="2013/06/30" /></ScreenData>');
--Data treated as scheduled items
INSERT INTO @ScreenData
VALUES (5, 2, '<ScreenData><Item><Field Name="00000000-0000-0000-0000-000000000002" Data="01/01/2012" /><Field Name="00000000-0000-0000-0000-000000000003" Data="02/02/2012" /></Item><Item><Field Name="00000000-0000-0000-0000-000000000002" Data="03/03/2012" /><Field Name="00000000-0000-0000-0000-000000000003" Data="04/04/2012" /></Item></ScreenData>')
INSERT INTO @ControlsToUpdate
SELECT FormID,
data.control.value('@Id', 'char(36)'),
data.control.value('@DataType', 'char(1)')
FROM @Form
CROSS APPLY FormDataXML.nodes('//Control') data(control)
WHERE data.control.value('@DataType', 'char(1)') = 'D';
--This will display the ScreenDataID, FormID, ControlID, and current Date value for dates that are in the old mm/dd/yyyy format
SELECT d.ScreenDataID, d.FormID, c.ControlID,
data.field.value('@Data', 'char(10)') as Date
FROM @ScreenData d
CROSS APPLY d.ScreenDataXML.nodes('//Field') as data(field)
INNER JOIN @ControlsToUpdate c ON c.ControlID = data.field.value('@Name', 'CHAR(36)')
AND d.FormID = c.FormID
WHERE data.field.value('@Data', 'char(10)')
LIKE '[01][0123456789]/[0123][0123456789]/[12][0123456789][0123456789][0123456789]';
--UPDATE d
--SET data.field.modify('replace value of (/@Data) with "' +
-- SUBSTRING(data.field.value('@Data', 'char(10)'), 7, 4) + '/' +
-- SUBSTRING(data.field.value('@Data', 'char(10)'), 1, 2) + '/' +
-- SUBSTRING(data.field.value('@Data', 'char(10)'), 4, 2) + '"')
--FROM @ScreenData d
--CROSS APPLY d.ScreenDataXML.nodes('//Field') as data(field)
--INNER JOIN @ControlsToUpdate c ON c.Id = data.field.value('@Name', 'CHAR(36)')
-- AND d.FormID = c.FormID
--WHERE data.field.value('@Data', 'varchar(MAX)')
--LIKE '[01][0123456789]/[0123][0123456789]/[12][0123456789][0123456789][0123456789]';
GO
--Edit-- @ScreenData 中的 xml 也可以包含<Item>
包含节点的<Field>
节点。发生这种情况时,您将拥有多个<Field>
具有相同 Name 属性值的节点。当屏幕具有项目列表视图时就是这种情况,您将拥有多个项目,每个项目都引用相同的控件但有自己的值。ScreenDataID 5 显示了这一点。