Requête du serveur SQL pour obtenir la liste des colonnes d'une table avec les types de données, NOT NULL et les contraintes PRIMARY KEY

229

J'ai besoin d'écrire une requête sur SQL Server pour obtenir la liste des colonnes d'une table particulière, ses types de données associés (avec longueur) et s'ils ne sont pas nuls. Et j'ai réussi à faire beaucoup.

Mais maintenant, je dois aussi obtenir, dans le même tableau, contre une colonne - TRUE si cette colonne est une clé primaire.

Comment puis-je faire cela?

Ma sortie attendue est:

Column name | Data type | Length | isnull | Pk
Shrayas
la source
2
Pourriez-vous montrer le code que vous avez déjà?
DOK

Réponses:

478

Pour éviter les lignes en double pour certaines colonnes, utilisez user_type_id au lieu de 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')

Il suffit de remplacer YourTableName par votre nom de table réel - fonctionne pour SQL Server 2005 et versions ultérieures.

Dans le cas où vous utilisez des schémas, remplacez YourTableName par YourSchemaName.YourTableNameYourSchemaNameest le nom du schéma réel et YourTableNamele nom de la table réelle.

marc_s
la source
2
Cela donne des longueurs incorrectes pour le type de colonnes nvarchar, etc. Il donne une longueur d'octets deux fois plus importante que la longueur du type de colonne.
Andrew Savinykh
14
Ces longueurs ne sont pas fausses - cela donne la longueur d'octets - c'est la longueur maximale possible en octets ... si vous voulez calculer l'espace etc., c'est la longueur que vous voulez obtenir ....
marc_s
2
Fonctionne très bien pour moi SQL Server 2012 :)
Doc Holiday
2
OERE c.object_id = OBJECT_ID ('YourTableName') .... J'avais besoin OERE c.object_id = OBJECT_ID ('MySchema.MyTableName'), puis tout a bien fonctionné.
Ivan
7
Cette requête renvoie des colonnes dupliquées si vous avez plusieurs index impliquant la même colonne. Pour y remédier, remplacez les deux dernières jointures par ce qui suit: 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
95

La procédure stockée sp_columns renvoie des informations détaillées sur la table.

exec sp_columns MyTable
décompilé
la source
2
exec sp_pkeys exec sp_fkeys
Leonardo Marques de Souza
2
requête courte mais fait un gros travail.
Akshay Chawla,
1
belle réponse concise
t_warsop
72

Vous pouvez utiliser la requête:

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

pour obtenir toutes les métadonnées dont vous avez besoin, à l'exception des informations Pk.

Ajadex
la source
2
je l'ai fait :) Mais j'ai aussi besoin du PK: |
Shrayas
msdn.microsoft.com/pt-br/library/ms189813(v=sql.120).aspx sp_fkeys sp_pkeys
Leonardo Marques de Souza
1
C'est génial car cela fonctionne avec les versions SS antérieures à 2005. Merci!
Karl Hoaglund
19

Dans SQL 2012, vous pouvez utiliser:

EXEC sp_describe_first_result_set N'SELECT * FROM [TableName]'

Cela vous donnera les noms des colonnes ainsi que leurs propriétés.

Amruta Kar
la source
13

Essaye ça:

select COLUMN_NAME, DATA_TYPE, CHARACTER_MAXIMUM_LENGTH, IS_NULLABLE 
from INFORMATION_SCHEMA.COLUMNS IC
where TABLE_NAME = 'tablename' and COLUMN_NAME = 'columnname'
khaleel
la source
2
En quoi votre réponse diffère-t-elle de celle publiée par Ajadex? Les deux réponses ne renvoient pas d'informations sur la clé primaire.
Artemix
10

Pour vous assurer d'obtenir la bonne longueur, vous devez considérer les types unicode comme un cas spécial. Voir le code ci-dessous.

Pour plus d'informations, voir: 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')
Développeur Microsoft
la source
1
le nom DDL est SI utile pour le sql dynamique qui crée des tables! Merci!!
George Menoutis
6

En développant la réponse d'Alex, vous pouvez le faire pour obtenir la contrainte 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'

Je dois avoir manqué que vous vouliez un indicateur pour déterminer si la colonne donnée faisait partie du PK au lieu du nom de la contrainte PK. Pour cela, vous utiliseriez:

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'
Thomas
la source
bien. Cela ne me donne pas le résultat requis :(
Shrayas
5

wite le nom de la table dans l'éditeur de requête sélectionnez le nom et appuyez sur Alt + F1 et il apportera toutes les informations de la table.

Abu Zafor
la source
Il demande une requête, mais vous avez raison de cette façon vous permet de voir toutes les informations.
Rafa Barragan
mais reste; super soigné :-)
netfed
4
SELECT COLUMN_NAME, IS_NULLABLE, DATA_TYPE, CHARACTER_MAXIMUM_LENGTH FROM information_schema.columns WHERE table_name = '<name_of_table_or_view>'

Courir SELECT * l'instruction ci-dessus pour voir ce que retourne information_schema.columns.

Cette question a été répondue précédemment - https://stackoverflow.com/a/11268456/6169225

Marquistador
la source
si cette question a déjà été répondue, marquez le message comme un doublon .
Martijn Pieters
4

Je suis un peu surpris, personne n'a mentionné

sp_help 'mytable'
Mario Levesque
la source
3
IF EXISTS (SELECT 1 FROM INFORMATION_SCHEMA.TABLES 
     WHERE TABLE_TYPE = 'BASE TABLE' AND TABLE_NAME = 'Table')
      BEGIN
        SELECT COLS.COLUMN_NAME, COLS.DATA_TYPE, COLS.CHARACTER_MAXIMUM_LENGTH, 
              (SELECT 'Yes' FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS TC JOIN INFORMATION_SCHEMA.KEY_COLUMN_USAGE KCU
                              ON COLS.TABLE_NAME = TC.TABLE_NAME 
                             AND TC.CONSTRAINT_TYPE = 'PRIMARY KEY'
                             AND KCU.TABLE_NAME = TC.TABLE_NAME
                             AND KCU.CONSTRAINT_NAME = TC.CONSTRAINT_NAME
                             AND KCU.COLUMN_NAME = COLS.COLUMN_NAME) AS KeyX
        FROM INFORMATION_SCHEMA.COLUMNS COLS WHERE TABLE_NAME = 'Table' ORDER BY KeyX DESC, COLUMN_NAME
      END
Pete vM
la source
3

Jeter une autre réponse dans le ring, cela vous donnera ces colonnes et plus:

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;
JustinStolle
la source
2
select
      c.name as [column name], 
      t.name as [type name],
      tbl.name as [table name]
from sys.columns c
         inner join sys.types t 
      on c.system_type_id = t.system_type_id 
         inner join sys.tables tbl
      on c.object_id = tbl.object_id
where
      c.object_id = OBJECT_ID('YourTableName1') 
          and 
      t.name like '%YourSearchDataType%'
union
(select
      c.name as [column name], 
      t.name as [type name],
      tbl.name as [table name]
from sys.columns c
         inner join sys.types t 
      on c.system_type_id = t.system_type_id 
         inner join sys.tables tbl
      on c.object_id = tbl.object_id
where
      c.object_id = OBJECT_ID('YourTableName2') 
          and 
      t.name like '%YourSearchDataType%')
union
(select
      c.name as [column name], 
      t.name as [type name],
      tbl.name as [table name]
from sys.columns c
         inner join sys.types t 
      on c.system_type_id = t.system_type_id 
         inner join sys.tables tbl
      on c.object_id = tbl.object_id
where
      c.object_id = OBJECT_ID('YourTableName3') 
          and 
      t.name like '%YourSearchDataType%')
order by tbl.name

Pour rechercher quelle colonne se trouve dans quelle table en fonction de votre type de données de recherche pour trois tables différentes dans une base de données. Cette requête est extensible aux tables «n».

mtinyavuz
la source
2

Trouver le résultat de la combinaison pour le type de données et la longueur et peut être annulé sous la forme de "NULL" et "Non nul" Utilisez la requête ci-dessous.

SELECT c.name AS 'Column Name',
       t.name + '(' + cast(c.max_length as varchar(50)) + ')' As 'DataType',
       case 
         WHEN  c.is_nullable = 0 then 'null' else 'not null'
         END AS 'Constraint'
  FROM sys.columns c
  JOIN sys.types t
    ON c.user_type_id = t.user_type_id
 WHERE c.object_id    = Object_id('TableName')

vous trouverez le résultat comme indiqué ci-dessous.

entrez la description de l'image ici

Je vous remercie.

Ankit Mori
la source
1
votre condition de Contrainte devrait être l'inverse.
Allen
0
SELECT  
   T.NAME AS [TABLE NAME]
   ,C.NAME AS [COLUMN NAME]
   ,P.NAME AS [DATA TYPE]
   ,P.MAX_LENGTH AS [Max_SIZE]
   ,C.[max_length] AS [ActualSizeUsed]
   ,CAST(P.PRECISION AS VARCHAR) +'/'+ CAST(P.SCALE AS VARCHAR) AS [PRECISION/SCALE]
FROM SYS.OBJECTS AS T
JOIN SYS.COLUMNS AS C
    ON T.OBJECT_ID = C.OBJECT_ID
JOIN SYS.TYPES AS P
    ON C.SYSTEM_TYPE_ID = P.SYSTEM_TYPE_ID
    AND C.[user_type_id] = P.[user_type_id]
WHERE T.TYPE_DESC='USER_TABLE'
  AND T.name = 'InventoryStatus'
ORDER BY 2
Rajiv Singh
la source
1
Veuillez utiliser l'indentation au lieu du balisage en ligne et ajouter des explications à votre réponse.
Toxantron
Pourquoi commander par 2?
reggaeguitar
0

entrez la description de l'image ici

Requête: EXEC SP_DESCRIBE_FIRST_RESULT_SET N'SELECT ANNUAL_INCOME FROM [BSLID2C]. [DBO]. [EMPLOYEE] '

REMARQUE: DANS CERTAINS IDE AVANT SELECT N FONCTIONNE OU, DANS CERTAINS IDE SANS N FONCTIONNE

Abdullah Pariyani
la source
0

Il n'y a pas de clé primaire ici, mais cela peut aider d'autres utilisateurs qui souhaitent simplement avoir un nom de table avec un nom de champ et des propriétés de champ de base

USE [**YourDB**]
GO
SELECT tbl.name, fld.[Column Name],fld.[Constraint],fld.DataType 
FROM sys.all_objects as tbl left join 
(SELECT c.OBJECT_ID,  c.name AS 'Column Name',
       t.name + '(' + cast(c.max_length as varchar(50)) + ')' As 'DataType',
       case 
         WHEN  c.is_nullable = 0 then 'null' else 'not null'
         END AS 'Constraint'
  FROM sys.columns c
  JOIN sys.types t
    ON c.user_type_id = t.user_type_id
) as fld on tbl.OBJECT_ID = fld.OBJECT_ID
WHERE ( tbl.[type]='U' and tbl.[is_ms_shipped] = 0)
ORDER BY tbl.[name],fld.[Column Name]
GO
Suhail Abdul Rehman Chougule
la source
-1

Je viens de faire marc_s "présentation prête":

SELECT 
    c.name 'Column Name',
    t.name 'Data type',
    IIF(t.name = 'nvarchar', c.max_length / 2, c.max_length) 'Max Length',
    c.precision 'Precision',
    c.scale 'Scale',
    IIF(c.is_nullable = 0, 'No', 'Yes') 'Nullable',
    IIF(ISNULL(i.is_primary_key, 0) = 0, 'No', 'Yes') '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')
krs
la source