gpt4 book ai didi

sql-server - CTE 和 FOR XML 生成嵌套的 XML

转载 作者:数据小太阳 更新时间:2023-10-29 01:41:22 26 4
gpt4 key购买 nike

我在数据库中有一个邻接表,想通过 SQL SP 将 XML 格式的数据传递给客户端。我正在尝试使用 CTE 和 FOR XML,但我没有让 XML 节点嵌套。

仅供引用,这将代表站点地图。

表结构:

CREATE TABLE [dbo].[PageHierarchy](
[ModuleId] [int] NOT NULL,
[PageId] [int] IDENTITY(1,1) NOT NULL,
[ParentPageId] [int] NULL,
[PageUrl] [nvarchar](100) NULL,
[PageTitle] [nvarchar](50) NOT NULL,
[PageOrder] [int] NULL)

以及 CTE 的开端:

;WITH cte AS
(
select * from PageHierarchy where ParentPageId is null
union all
select child.* from PageHierarchy child inner join cte parent on parent.PageId = child.ParentPageId
)
SELECT ModuleId, PageId, ParentPageId, PageUrl, PageTitle, PageOrder FROM cte
group by ModuleId, PageId, ParentPageId, PageUrl, PageTitle, PageOrder
order by PageOrder
for xml auto, root ('bob')

生成如下所示的 XML:

<bob>
<cte ModuleId="1" PageId="1" PageUrl="~/Admin/" PageTitle="Administration" PageOrder="1000" />
<cte ModuleId="1" PageId="4" ParentPageId="1" PageTitle="Manage Users" PageOrder="1030" />
<cte ModuleId="1" PageId="5" ParentPageId="4" PageUrl="~/Admin/AddUser" PageTitle="Add Users" PageOrder="1040" />
<cte ModuleId="1" PageId="8" ParentPageId="4" PageUrl="~/Admin/EditUser" PageTitle="Edit/Search User" PageOrder="1070" />
</bob>

当我想要的是如下所示的 XML 时:

<bob>
<cte ModuleId="1" PageId="1" PageUrl="~/Admin/" PageTitle="Administration" PageOrder="1000" />
<cte ModuleId="1" PageId="4" ParentPageId="1" PageTitle="Manage Users" PageOrder="1030" >
<cte ModuleId="1" PageId="5" ParentPageId="4" PageUrl="~/Admin/AddUser" PageTitle="Add Users" PageOrder="1040" />
<cte ModuleId="1" PageId="8" ParentPageId="4" PageUrl="~/Admin/EditUser" PageTitle="Edit/Search User" PageOrder="1070" />
</cte>
</bob>

我猜问题不在于 CTE,而在于选择,但我不知道从哪里开始修复它。另外,我不知道嵌套的深度,所以我假设我需要它来支持至少 10 层的深度。

编辑 1:
我想我越来越接近......看着this页面,我创建了一个 UDF,但仍然存在一些问题:

CREATE FUNCTION PageHierarchyNode(@PageId int)
RETURNS XML
WITH RETURNS NULL ON NULL INPUT
BEGIN RETURN
(SELECT ModuleId AS "@ModuleId", PageId AS "@PageId",
ParentPageId AS "@ParentPageId", PageUrl AS "@PageUrl",
PageTitle AS "@PageTitle", PageOrder AS "@PageOrder",
CASE WHEN ParentPageId=@PageId
THEN dbo.PageHierarchyNode(PageId)
END
FROM dbo.PageHierarchy WHERE ParentPageId=@PageId
FOR XML PATH('Page'), TYPE)
END

以及调用UDF的SQL

SELECT ModuleId AS "@ModuleId", PageId AS "@PageId",
ParentPageId AS "@ParentPageId", PageUrl AS "@PageUrl",
PageTitle AS "@PageTitle", PageOrder AS "@PageOrder",
dbo.PageHierarchyNode(PageId)
FROM PageHierarchy
FOR XML PATH('Page'), ROOT('SiteMap'), TYPE

这将为我嵌套 XML,但它会复制节点,这不是我想要的......

编辑 2:

我只需要向调用 UDF 的 SELECT 添加一个 WHERE 子句:

...
WHERE ParentPageId IS NULL

最佳答案

原来我根本不想要 CTE,只是我递归调用的 UDF

CREATE FUNCTION PageHierarchyNode(@PageId int)
RETURNS XML
WITH RETURNS NULL ON NULL INPUT
BEGIN RETURN
(SELECT ModuleId AS "@ModuleId", PageId AS "@PageId",
ParentPageId AS "@ParentPageId", PageUrl AS "@PageUrl",
PageTitle AS "@PageTitle", PageOrder AS "@PageOrder",
CASE WHEN ParentPageId=@PageId
THEN dbo.PageHierarchyNode(PageId)
END
FROM dbo.PageHierarchy WHERE ParentPageId=@PageId
FOR XML PATH('Page'), TYPE)
END

使用调用 UDF 的 SQL 作为

SELECT ModuleId AS "@ModuleId", PageId AS "@PageId",
ParentPageId AS "@ParentPageId", PageUrl AS "@PageUrl",
PageTitle AS "@PageTitle", PageOrder AS "@PageOrder",
dbo.PageHierarchyNode(PageId)
FROM PageHierarchy
WHERE ParentPageId IS NULL
FOR XML PATH('Page'), ROOT('SiteMap'), TYPE

关于sql-server - CTE 和 FOR XML 生成嵌套的 XML,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14765937/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com