获取表中列的列表,包括数据类型、非空和主键约束的 SQL Server 查询

296

我需要编写一条SQL Server查询,以获取特定表中列的列表、它们关联的数据类型(及其长度)以及它们是否为非空值。我已经完成了这部分内容。

但是现在我还需要在同一个表中,针对某一列,在该列是主键时返回TRUE

我应该如何做到这一点?

我的预期输出是:

Column name | Data type | Length | isnull | Pk

4
请展示您已经有的代码。 - DOK
20个回答

577

为了避免某些列出现重复行,请使用user_type_id而不是system_type_id。

SELECT 
    c.name 'Column Name',
    t.Name 'Data type',
    c.max_length 'Max Length',
    c.precision ,
    c.scale ,
    c.is_nullable,
    ISNULL(i.is_primary_key, 0) 'Primary Key'
FROM    
    sys.columns c
INNER JOIN 
    sys.types t ON c.user_type_id = t.user_type_id
LEFT OUTER JOIN 
    sys.index_columns ic ON ic.object_id = c.object_id AND ic.column_id = c.column_id
LEFT OUTER JOIN 
    sys.indexes i ON ic.object_id = i.object_id AND ic.index_id = i.index_id
WHERE
    c.object_id = OBJECT_ID('YourTableName')

只需将YourTableName替换为您实际的表名 - 适用于SQL Server 2005及以上版本。

如果您使用模式,请将YourTableName替换为YourSchemaName.YourTableName,其中YourSchemaName是实际模式名称,YourTableName是实际表名。


2
这会给nvarchar等类型的列提供错误的长度。它会给出字节长度,是列类型中长度的两倍。 - Andrew Savinykh
16
那些长度并没有错误 - 它确实给出了字节长度 - 这是以字节为单位的最大可能长度... 如果你想计算空间等等,那就是你要获得的长度.... - marc_s
2
对我来说,SQL Server 2012非常好用 :) - Doc Holiday
12
如果涉及同一列的多个索引,此查询将返回重复的列。要修复它,请使用以下内容替换最后两个联接:LEFT OUTER JOIN sys.index_columns ic LEFT OUTER JOIN sys.indexes i ON ic.object_id = i.object_id AND ic.index_id = i.index_id ON ic.object_id = c.object_id AND ic.column_id = c.column_id AND i.is_primary_key=1 - Razvan Socol
2
这个查询对我来说节省了很多时间,因为我的表有60多列。而且它完美地运行了。 - Akshay Chawla
显示剩余24条评论

112

存储过程 sp_columns 返回详细的表信息。

exec sp_columns MyTable

2
执行 sp_pkeys 执行 sp_fkeys - Leonardo Marques de Souza
3
如果使用这个命令,请注意MyTable只是表名,不是模式。若要按模式过滤,请将其作为第二个参数添加:exec sp_columns 'MyTable', 'MySchema' - Daniel

95

您可以使用以下查询:

select COLUMN_NAME, DATA_TYPE, CHARACTER_MAXIMUM_LENGTH, 
       NUMERIC_PRECISION, DATETIME_PRECISION, 
       IS_NULLABLE 
from INFORMATION_SCHEMA.COLUMNS
where TABLE_NAME='TableName'

获取除了Pk信息以外的所有所需元数据。


2
我已经做到了 :) 但我还需要主键 :| - Shrayas
sp_fkeys该存储过程返回指定表的外键信息。它接受三个参数:表架构、表名称和参考表架构。sp_pkeys该存储过程返回指定表的主键信息。它接受三个参数:表架构、表名称和目录架构。 - Leonardo Marques de Souza

25

在SQL 2012中,您可以使用:

EXEC sp_describe_first_result_set N'SELECT * FROM [TableName]'

这将为您提供列名以及它们的属性。


这个方法适用于使用OPENROWSET打开的Excel文件,而许多其他解决方案则不适用。谢谢。 - Uttam

15

试试这个:

select COLUMN_NAME, DATA_TYPE, CHARACTER_MAXIMUM_LENGTH, IS_NULLABLE 
from INFORMATION_SCHEMA.COLUMNS IC
where TABLE_NAME = 'tablename' and COLUMN_NAME = 'columnname'

2
你的答案与Ajadex发布的答案有何不同?两个答案都没有返回主键信息。 - Artemix

13
为确保获得正确的长度,您需要将Unicode类型视为特殊情况。请参见下面的代码。
更多信息请参见:https://msdn.microsoft.com/en-us/library/ms176106.aspx
SELECT 
   c.name 'Column Name',
   t.name,
   t.name +
   CASE WHEN t.name IN ('char', 'varchar','nchar','nvarchar') THEN '('+

             CASE WHEN c.max_length=-1 THEN 'MAX'

                  ELSE CONVERT(VARCHAR(4),

                               CASE WHEN t.name IN ('nchar','nvarchar')

                               THEN  c.max_length/2 ELSE c.max_length END )

                  END +')'

          WHEN t.name IN ('decimal','numeric')

                  THEN '('+ CONVERT(VARCHAR(4),c.precision)+','

                          + CONVERT(VARCHAR(4),c.Scale)+')'

                  ELSE '' END

   as "DDL name",
   c.max_length 'Max Length in Bytes',
   c.precision ,
   c.scale ,
   c.is_nullable,
   ISNULL(i.is_primary_key, 0) 'Primary Key'
FROM    
   sys.columns c
INNER JOIN 
   sys.types t ON c.user_type_id = t.user_type_id
LEFT OUTER JOIN 
   sys.index_columns ic ON ic.object_id = c.object_id AND ic.column_id = c.column_id
LEFT OUTER JOIN 
   sys.indexes i ON ic.object_id = i.object_id AND ic.index_id = i.index_id
WHERE
   c.object_id = OBJECT_ID('YourTableName')

2
DDL名称对于创建动态SQL的表格非常有用!谢谢! - George Menoutis

11

我有点惊讶,没有人提到

sp_help 'mytable'

对我不起作用。 - Ac Hybl
@Ac Hybl,你把表名放在单引号里了吗?sp_help对我来说已经使用了20多年。 - Mario Levesque
谢谢关注,但是是的。我刚刚再试了一次,使用单引号时出现了语法错误。这可能不是你方法的问题。这可能是与Integrify后端有关的问题。我无法直接访问DB。我通过公司的工作流管理平台发送查询。 - Ac Hybl

7

在Alex的回答上进行扩展,您可以执行以下操作以获取PK约束

Select C.COLUMN_NAME, C.DATA_TYPE, C.CHARACTER_MAXIMUM_LENGTH, C.NUMERIC_PRECISION, C.IS_NULLABLE, TC.CONSTRAINT_NAME
From INFORMATION_SCHEMA.COLUMNS As C
    Left Join INFORMATION_SCHEMA.TABLE_CONSTRAINTS As TC
      On TC.TABLE_SCHEMA = C.TABLE_SCHEMA
          And TC.TABLE_NAME = C.TABLE_NAME
          And TC.CONSTRAINT_TYPE = 'PRIMARY KEY'
Where C.TABLE_NAME = 'Table'

我可能错过了您想要一个标志来确定给定的列是否是主键的一部分,而不是PK约束的名称。对于这个问题,您可以使用:

Select C.COLUMN_NAME, C.DATA_TYPE, C.CHARACTER_MAXIMUM_LENGTH
    , C.NUMERIC_PRECISION, C.NUMERIC_SCALE
    , C.IS_NULLABLE
    , Case When Z.CONSTRAINT_NAME Is Null Then 0 Else 1 End As IsPartOfPrimaryKey
From INFORMATION_SCHEMA.COLUMNS As C
    Outer Apply (
                Select CCU.CONSTRAINT_NAME
                From INFORMATION_SCHEMA.TABLE_CONSTRAINTS As TC
                    Join INFORMATION_SCHEMA.CONSTRAINT_COLUMN_USAGE As CCU
                        On CCU.CONSTRAINT_NAME = TC.CONSTRAINT_NAME
                Where TC.TABLE_SCHEMA = C.TABLE_SCHEMA
                    And TC.TABLE_NAME = C.TABLE_NAME
                    And TC.CONSTRAINT_TYPE = 'PRIMARY KEY'
                    And CCU.COLUMN_NAME = C.COLUMN_NAME
                ) As Z
Where C.TABLE_NAME = 'Table'

好吧,它没有给我所需的结果 :( - Shrayas

6
SELECT COLUMN_NAME, IS_NULLABLE, DATA_TYPE, CHARACTER_MAXIMUM_LENGTH FROM information_schema.columns WHERE table_name = '<name_of_table_or_view>'

在上述语句中运行SELECT *以查看information_schema.columns返回的信息。
此问题已经得到回答-https://stackoverflow.com/a/11268456/6169225

如果这个问题已经被回答,请将帖子标记为重复。 - Martijn Pieters
使用'information_schema.columns',我们可以查看所有表列的信息,包括查看表名和模式。 - Ram

6
抛砖引玉,以下代码将为您提供这些列以及更多内容:
SELECT col.TABLE_CATALOG AS [Database]
     , col.TABLE_SCHEMA AS Owner
     , col.TABLE_NAME AS TableName
     , col.COLUMN_NAME AS ColumnName
     , col.ORDINAL_POSITION AS OrdinalPosition
     , col.COLUMN_DEFAULT AS DefaultSetting
     , col.DATA_TYPE AS DataType
     , col.CHARACTER_MAXIMUM_LENGTH AS MaxLength
     , col.DATETIME_PRECISION AS DatePrecision
     , CAST(CASE col.IS_NULLABLE
                WHEN 'NO' THEN 0
                ELSE 1
            END AS bit)AS IsNullable
     , COLUMNPROPERTY(OBJECT_ID('[' + col.TABLE_SCHEMA + '].[' + col.TABLE_NAME + ']'), col.COLUMN_NAME, 'IsIdentity')AS IsIdentity
     , COLUMNPROPERTY(OBJECT_ID('[' + col.TABLE_SCHEMA + '].[' + col.TABLE_NAME + ']'), col.COLUMN_NAME, 'IsComputed')AS IsComputed
     , CAST(ISNULL(pk.is_primary_key, 0)AS bit)AS IsPrimaryKey
  FROM INFORMATION_SCHEMA.COLUMNS AS col
       LEFT JOIN(SELECT SCHEMA_NAME(o.schema_id)AS TABLE_SCHEMA
                      , o.name AS TABLE_NAME
                      , c.name AS COLUMN_NAME
                      , i.is_primary_key
                   FROM sys.indexes AS i JOIN sys.index_columns AS ic ON i.object_id = ic.object_id
                                                                     AND i.index_id = ic.index_id
                                         JOIN sys.objects AS o ON i.object_id = o.object_id
                                         LEFT JOIN sys.columns AS c ON ic.object_id = c.object_id
                                                                   AND c.column_id = ic.column_id
                  WHERE i.is_primary_key = 1)AS pk ON col.TABLE_NAME = pk.TABLE_NAME
                                                  AND col.TABLE_SCHEMA = pk.TABLE_SCHEMA
                                                  AND col.COLUMN_NAME = pk.COLUMN_NAME
 WHERE col.TABLE_NAME = 'YourTableName'
   AND col.TABLE_SCHEMA = 'dbo'
 ORDER BY col.TABLE_NAME, col.ORDINAL_POSITION;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接