sql-server - SQL将多个表合并为一个问题
问题描述
我在 SSMS 中运行以下查询,您可以看到它显示了来自不同数据库的多个表中的结果。
如何将表格合并为一个?
我尝试将 UNION 和 UNION ALL 添加到其中SELECT @Sql = COALESCE(@Sql + ' UNION ' + CHAR(13) + CHAR(10), '' ) +
,但出现Cannot resolve the collation conflict between "Latin1_General_CI_AS" and "SQL_Latin1_General_CP1_CI_AS" in the UNION operation.
错误
任何想法将不胜感激!
DECLARE @table table(dbname sysname COLLATE Latin1_General_CI_AS)
INSERT INTO @table(dbname)
SELECT NAME COLLATE Latin1_General_CI_AS FROM sys.databases where name like '%AccountsLive'
DECLARE @Sql NVARCHAR(MAX) = NULL;
SELECT @Sql = COALESCE(@Sql + CHAR(13) + CHAR(10), '' ) +
'SELECT c_broker, det_costheader, cuname, ch_name, CONVERT(VARCHAR(11),ch_date_req,103) AS ''Flight_Date'', IIF(OUT.outstanding>0, ''Yes'', ''No'') AS ''Anything_Outstanding?'', sum(c_grossmargin)/2 AS ''Gross_Margin'' FROM '
+ QUOTENAME(dbname) COLLATE Latin1_General_CI_AS + '.dbo.AT_ACS_COMMISSIONS AS COM
LEFT JOIN ' + QUOTENAME(dbname) COLLATE Latin1_General_CI_AS + '.dbo.AT_ACS_PROJECTOUTSTANDING AS OUT
ON COM.det_costheader = OUT.project
WHERE c_broker = ''HKGCGOJT''
GROUP BY det_costheader, c_broker, cuname, ch_name, ch_date_req, outstanding'
FROM @table
exec( @Sql );
解决方案
如果任何列在不同的数据库中有不同的排序规则,那么您需要向COLLATE
它们添加一个。我添加了几个来向您展示它是如何完成的。
笔记:
- 此处不需要表变量
- 使用
STRING_AGG
(orFOR XML
) 聚合,不要使用变量 coalescing,这是不安全的。 - 如果您打算添加参数,则使用
sp_executesql
并将参数传递给动态部分。不要将参数连接到查询中。
DECLARE @Sql NVARCHAR(MAX);
SET @Sql =
(SELECT
STRING_AGG(
N'SELECT
c_broker COLLATE Latin1_General_CI_AS,
det_costheader,
cuname COLLATE Latin1_General_CI_AS,
ch_name,
CONVERT(VARCHAR(11), ch_date_req, 103) AS [Flight_Date],
IIF(OUT.outstanding > 0, ''Yes'', ''No'') AS [Anything_Outstanding?],
sum(c_grossmargin) / 2 AS [Gross_Margin]
FROM ' + QUOTENAME(d.name) + N'.dbo.AT_ACS_COMMISSIONS AS COM
LEFT JOIN ' + QUOTENAME(d.name) + N'.dbo.AT_ACS_PROJECTOUTSTANDING AS OUT
ON COM.det_costheader = OUT.project
WHERE c_broker = ''HKGCGOJT''
GROUP BY det_costheader, c_broker, cuname, ch_name, ch_date_req, outstanding
',
CAST (N'
UNION ALL
' AS nvarchar(max)) )
FROM sys.databases d
where name like '%AccountsLive'
);
exec( @Sql );
如果您在没有 的旧版本 SQL Server 上STRING_AGG
,则需要使用FOR XML
聚合:
DECLARE @Sql NVARCHAR(MAX);
SET @Sql =
STUFF(
(SELECT
NCHAR(10) + N'UNION ALL' + NCHAR(10) +
N'SELECT
c_broker COLLATE Latin1_General_CI_AS,
det_costheader,
cuname COLLATE Latin1_General_CI_AS,
ch_name,
CONVERT(VARCHAR(11), ch_date_req, 103) AS [Flight_Date],
IIF(OUT.outstanding > 0, ''Yes'', ''No'') AS [Anything_Outstanding?],
sum(c_grossmargin) / 2 AS [Gross_Margin]
FROM ' + QUOTENAME(d.name) + N'.dbo.AT_ACS_COMMISSIONS AS COM
LEFT JOIN ' + QUOTENAME(d.name) + N'.dbo.AT_ACS_PROJECTOUTSTANDING AS OUT
ON COM.det_costheader = OUT.project
WHERE c_broker = ''HKGCGOJT''
GROUP BY det_costheader, c_broker, cuname, ch_name, ch_date_req, outstanding
'
FROM sys.databases d
where name like '%AccountsLive'
FOR XML PATH(''), TYPE)
.value('text()[1]','nvarchar(max)'), 1, 11, '');
exec( @Sql );
我建议将其PRINT @Sql
用于测试目的,以确保生成的 SQL 确实有效。
推荐阅读
- java - 我的解决方案中是否存在逻辑流程问题?
- excel - 计算 imagej 或 excel 上每对 31 个数组之间的均方误差
- java - Java 控制台输入长度限制
- active-directory - DirectorySearcher() 和 AuthenticationTypes 枚举
- ios - 如何在 ForEach 中绑定 @State 变量的属性?
- libasound - 来自 libasound 函数 snd_pcm_hw_params 的令人困惑的错误
- mysql - Mysql - 按大于或小于当前时间的日期对记录进行分组 now() mysql
- sql - SQL WHERE + MAX 查询
- php - 分页不适用于 CPT/Taxonomy 的自定义 URL 结构
- typescript - styled-component 组件中的 props 没有类型