- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
到目前为止,我可以使用以下 SQL 查询提取数据库表的列表:
SELECT
DISTINCT
TABLE_SCHEMA,
TABLE_NAME
FROM
INFORMATION_SCHEMA.COLUMNS
在每个表中,第一列名为“年份
”。这些值从年份“2011
”到年份“2017
”:
CREATE TABLE foo (
[Year] int,
AnotherColumn varchar(50),
...
)
CREATE TABLE bar (
[Year] int,
SomeOtherColumn guid,
...
)
CREATE TABLE ...
现在,我需要计算每个表中不同年份的行数,并按以下格式显示输出:
| TABLE_SCHEMA | TABLE_NAME | 2011 | 2012 | ... | 2017 |
|:-------------|-----------:|:-------------------:|:-------------------:|:----|:-------------------:|
| SCHEMA | foo | no. of rows of 2011 | no. of rows of 2012 | ... | no. of rows of 2017 |
| SCHEMA | bar | no. of rows of 2011 | no. of rows of 2012 | ... | no. of rows of 2017 |
| SCHEMA | ... | no. of rows of 2011 | no. of rows of 2012 | ... | no. of rows of 2017 |
有人有什么建议吗?非常感谢!
最佳答案
虽然每个 SQL 实现都提供某种形式的值参数化,但不存在这样的工具来参数化对象标识符(例如表名、列名等)——这意味着您必须求助于 Dynamic-SQL,这会引入其自身的风险(即SQL 注入(inject))。
对于您的具体问题,我们可以首先尝试在不使用 Dynamic-SQL 的情况下解决它,假设要查询一组已知且固定的表,然后我们可以将其转换为 Dynamic-SQL,希望以安全的方式:
SELECT
'Table1' AS TableName
[Year],
COUNT(*) AS YearRowCount
FROM
Table1
GROUP BY
[Year]
UNION ALL
SELECT
'Table2' AS TableName
[Year],
COUNT(*) AS YearRowCount
FROM
Table2
GROUP BY
[Year]
UNION ALL
...
希望您在这里看到了一种模式。
到目前为止,该查询将为我们提供以下形式的结果:
TableName Year YearRowCount
'Table1' 2017 1234
'Table1' 2016 2345
'Table1' 2015 3456
'Table1' 2014 1234
'Table1' 2013 1234
'Table1' 2011 1234
'Table2' 2017 1234
'Table2' 2016 2345
'Table2' 2015 3456
'Table2' 2013 1234
'Table2' 2012 1234
'Table2' 2011 1234
...
然后我们可以使用 PIVOT
将行转置为列。不幸的是,PIVOT
(和 UNPIVOT
)确实要求您显式命名要转置的每一列 - 但如果它们有一个 PIVOT ALL
那就太好了> 功能或其他东西)。
SELECT
tableName,
YearRowCount,
[2011], [2012], [2013], [2014], [2015], [2016], [2017]
FROM
(
-- our UNION query goes here --
)
PIVOT
(
SUM( YearRowCount )
FOR [Year] IN ( 2011, 2012, 2013, 2014, 2015, 2016, 2017 )
)
现在我们知道内部查询的模式和围绕它的 PIVOT 语句,我们可以使其动态化。
有 3 种方法可以在“针对每一行...”的基础上生成动态 SQL。第一种是使用 CURSOR
,第二种是使用某种 T-SQL 循环(WHILE
等) - 这两种方法都采用迭代 方法 - 但还有第三个版本,它功能更强大,语法更简单。我将演示这种函数式方法。
此外,我们可以通过使用(滥用)用作 sprintf
实现的 FORMATMESSAGE
函数来避免手动字符串连接的丑陋部分。要使用 FORMATMESSAGE
格式化字符串需要 SQL Server 2016 或更高版本(尽管据我所知,兼容性级别不需要为 130
)。如果您运行的是早期版本,则需要使用 CONCAT
或 'foo' + @var + 'bar'
式连接。
我还使用此答案中描述的 COALESCE( [aggregate] + [separator], '' ) + [value]
技巧:https://stackoverflow.com/a/194887/159145 - 这是连接(聚合)行的一种方法值(value)观,虽然感觉有点丑。请记住,SQL 主要关注无序元组数据集(即表)的关系代数,它通常不涵盖 View 级别的问题,例如排序或聚合排序数据 - 这就是连接。
DECLARE @unionTemplate varchar(1024) = '
SELECT
''%s.%s'' AS TableName
[Year],
COUNT(*) AS YearRowCount
FROM
[%s].[%s]
GROUP BY
[Year]
'
DECLARE @unionSeparator varchar(20) = '
UNION ALL
'
DECLARE @unionQuery varchar(max)
SELECT
@unionQuery = COALESCE( @unionQuery + @unionSeparator, '' ) + FORMATMESSAGE( @unionTemplate, SCHEMA_NAME, TABLE_NAME, SCHEMA_NAME, TABLE_NAME )
FROM
INFORMATION_SCHEMA.TABLES
ORDER BY
SCHEMA_NAME,
TABLE_NAME
无论如何,这个查询将生成存储在@unionQuery
中的查询,所以现在我们只需要编写它......
DECLARE @pivotQuery varchar(max) = '
SELECT
tableName,
YearRowCount,
[2011], [2012], [2013], [2014], [2015], [2016], [2017]
FROM
(
%s
)
PIVOT
(
SUM( YearRowCount )
FOR [Year] IN ( 2011, 2012, 2013, 2014, 2015, 2016, 2017 )
)'
SET @pivotQuery = FORMATMESSAGE( @pivotQuery, @unionQuery )
...并执行它(EXEC sp_executesql
优于古老的 EXEC()
) - 另请注意 EXEC()
是与 EXEC
不同!
EXEC sp_executesql @pivotQuery
哒哒!
这些未经测试,但如果您需要在 2016 年 (v13.0) 之前的 SQL Server 版本上运行,请尝试使用这些 FORMATMESSAGE
替代方案:
DECLARE @unionQuery nvarchar(max)
SELECT
@unionQuery =
COALESCE( @unionQuery + ' UNION ALL ', '' ) +
CONCAT(
'SELECT ''',
SCHEMA_NAME, '.', TABLE_NAME, '[Year],
COUNT(*) AS YearRowCount
FROM
[', SCHEMA_NAME, '].[', TABLE_NAME, ']
GROUP BY
[Year]
'
)
FROM
INFORMATION_SCHEMA.TABLES
ORDER BY
SCHEMA_NAME,
TABLE_NAME
由于@pivotQuery
仅插入一次,因此可以使用REPLACE
插入内部@unionQuery
,但切勿在以下情况下这样做:处理用户提供的值,因为您容易遭受类似 SQL 注入(inject)的攻击:
SET @pivotQuery = REPLACE( @pivotQuery, '%s', @unionQuery )
关于SQL查询对所有表中的列进行计数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46107854/
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 9 年前。 Improve
我有点卡在 JavaScript 逻辑上来完成这个任务。 基本上 如果我给出一个数字(比如 30) 我想在两边都显示 5。 所以 25 26 27 28 29 30 31 32 33 34 35 这部
我编写的程序有问题。我无法获得输入字符串的正确字数,但我获得了正确的最长字符数。我不知道为什么,但这是我的代码。我正在做的是将一个字符串传递给一个函数,该函数将字符串中的所有字母大写。然后,该函数逐个
我有功能 public ArrayList vyberNahodnaPismena() { String[] seznamPismen = {"A", "Á", "B", "C", "Č",
这可以在 PGSQL 中完成吗?我有一个我创建的 View ,其中主机名、ip 和数据中心来自一个表,ifdesc 和 if stats 来自另一个表。 View 输出如下所示: hostname |
我想要一组来自订单文件的数据,这些数据可以为我提供客户编号、订单编号、产品、数量、价格以及每个订单的订单详细信息文件中的行数。我在最后一部分遇到问题。 Select Header.CustNo, He
我有属于街道的房子。一个用户可以买几套房子。我如何知道用户是否拥有整条街道? street table with columns (id/name) house table with columns
我有一套有 200 万个主题标签。然而,只有大约 200k 是不同的值。我想知道哪些主题标签在我的数据中重复得更多。 我用它来查找每个主题标签在我的数据集上重复了多少次: db.hashtags.ag
我有如下文件: { "_id" : "someuniqueeventid", "event" : "event_type_1", "date" : ISODate("2014-
我有以下三个相互关联的表: 主持人(有多个 session ) session (有多个进程) 过程 表结构如下: 主机表 - id, name session 表 - id, host_id, na
我需要根据 2 个字段对行进行计数以进行分组。 动物(一) id group_id strain_id death_date death_cause status --
我有一个 LINQ 语句,我正在努力改正,所以可能这一切都错了。我的目标是查询一个表并加入另一个表以获取计数。 地点 标识、显示 ProfilePlaces ID、PlaceID、通话、聆听 基本上P
我无法编写 Countifs 来完成我想要的。我每个月都会运行一份 claim 报告,其中包含大量按列组织的数据,并每月将其导出到 Excel 中。在一个单独的选项卡上,我有引用此数据复制到的选项卡的
我有一些数据采用此 sqlfilddle 中描述的格式:http://sqlfiddle.com/#!4/b9cdf/2 基本上,一个包含用户 ID 和事件发生时间的表。我想做的是根据用户发生事件的时
我有以下 SQL 语句: SELECT [l.LeagueId] AS LeagueId, [l.LeagueName] AS NAME, [lp.PositionId] FROM
我试图找出一个值在列中出现的平均次数,根据另一列对其进行分组,然后对其进行计算。 我有 3 张 table ,有点像这样 DVD ID | NAME 1 | 1 2 | 1 3
我有一个非常简单的 SQL 问题。我有一个包含以下列的数据库表: 零件号 销售类型(为简单起见,称之为销售类型 1、2、3、4、5) 我希望编写一个包含以下三列的查询: 零件号 Sales Type
我创建了以下存储过程,用于计算选定位置的特定范围之间每天的记录数: [dbo].[getRecordsCount] @LOCATION as INT, @BEGIN as datetime, @END
我有一个包含一组列的表,其中一个是日期列。 我需要计算该列的值引用同一个月的次数。如果一个月内,该计数的总和超过 3,则返回。 例如: ____________________ | DATE |
看XXX数据如下: lala XXX = EL String [XXX] | TXT String | MMS String 为此,XXX数据yppz是由 lala
我是一名优秀的程序员,十分优秀!