SQL Server如何查找表名或列名中包含空格的表和列

最近发现一个数据库中的某个表有个字段名后面包含了一个空格,这个空格引起了一些小问题,一般出现这种情况,是因为创建对象时,使用双引号或双括号的时候,由于粗心或手误多了一个空格,如下简单案例所示:

USE TEST;
GO
 
--表TEST_COLUMN中两个字段都包含有空格
CREATE TABLE TEST_COLUMN 
(
    "ID  "    INT IDENTITY (1,1),
    [Name ]   VARCHAR(32),
    [Normal]  VARCHAR(32)
);
GO
 
--表[TEST_TABLE ]中包含空格, 里面对应三个字段,一个前面包含空格(后面详细阐述),一个字段中间包含空格,一个字段后面包含空格。
CREATE TABLE [TEST_TABLE ]
(
 
    [ F_NAME]        NVARCHAR(32),
    [M NAME]         NVARCHAR(32),
    [L_NAME ]        NVARCHAR(32)
)
GO

那么要如何找出表名或字段名包含空格的相关信息呢? 不管是常规方法还是正则表达式,这个都会效率不高。我们可以用一个取巧的方法,就是通过字段的字符数和字节数的规律来判断,如果没有包含空格,那么列名的字节数和字符数满足下面规律(表名也是如此):

DATALENGTH(name) = 2* LEN(name)

SELECT  name ,
        DATALENGTH(name) AS NAME_BYTES ,
        LEN(name)         AS NAME_CHARACTER
FROM    sys.columns
WHERE   object_id = OBJECT_ID(‘TEST_COLUMN‘);

原理是这样的,保存这些元数据的字段类型为sysname ,其实这个系统数据类型,用于定义表列、变量以及存储过程的参数,是nvarchar(128)的同义词。所以一个字母占2个字节。那么我们安装这个规律写了一个脚本来检查数据中那些表名或字段名包含空格。方便巡检。如下测试所示

 
IF OBJECT_ID(‘tempdb.dbo.#TabColums‘) IS NOT NULL
    DROP TABLE dbo.#TabColums;
 
CREATE TABLE #TabColums
(
    object_id            INT ,
    column_id            INT
)
 
INSERT INTO #TabColums
SELECT  object_id ,
        column_id
FROM    sys.columns
WHERE   DATALENGTH(name) != LEN(name) * 2
 
 
SELECT 
    TL.name AS TableName,
    C.Name AS FieldName,
    T.Name AS DataType,
    DATALENGTH(C.name) AS COLUMN_DATALENGTH,
    LEN(C.name) AS COLUMN_LENGTH,
    CASE WHEN C.Max_Length = -1 THEN ‘Max‘ ELSE CAST(C.Max_Length AS VARCHAR) END AS Max_Length,
    CASE WHEN C.is_nullable = 0 THEN ‘ב  ELSE N‘√‘ END AS Is_Nullable,
    C.is_identity,
    ISNULL(M.text, ‘‘)  AS  DefaultValue,
    ISNULL(P.value, ‘‘) AS FieldComment
    
FROM sys.columns  C
INNER JOIN  sys.types T ON C.system_type_id = T.user_type_id
LEFT  JOIN dbo.syscomments M ON M.id = C.default_object_id
LEFT  JOIN sys.extended_properties P ON P.major_id = C.object_id AND C.column_id = P.minor_id 
INNER JOIN sys.tables TL ON TL.object_id = C.object_id
INNER JOIN #TabColums TC ON C.object_id = TC.object_id AND c.column_id = TC.column_id
ORDER BY C.Column_Id ASC

那么为什么表名TEST_TABLE的三个字段里面,前面包含空格与与中间包含空格都识别不出来呢?这个与数据库的LEN函数有关系,LEN函数返回指定字符串表达式的字符数,其中

不包含尾随空格。所以这个脚本是无法排查表名或字段名前面包含空格的。如果要排查这种情况,就需要使用下面SQL脚本(中间包含空格在此略过,这个不符合命名规则):

SELECT * FROM sys.columns WHERE NAME LIKE ‘ %‘  --字段前面包含空格。

其实到了这一步,还没有完,如果一个实例,里面有十几个数据库,那么使用上面这个脚本,我要切换数据库,执行十几次,对于我这种懒人来说,我觉得无法忍受的。那么必须写

一个脚本,将所有数据库全部检查完。本来想用sys.sp_MSforeachdb,但是这个内部存储过程有一些限制,遂写了下面脚本。

DECLARE @db_name  NVARCHAR(32);
DECLARE @sql_text NVARCHAR(MAX);
 
DECLARE @db TABLE 
(
    database_name  NVARCHAR(64)
);
 
IF OBJECT_ID(‘tempdb.dbo.#TabColums‘) IS NOT NULL
 
    DROP TABLE dbo.#TabColums;
 
CREATE TABLE #TabColums
(
    object_id            INT ,
    column_id            INT
);
 
 
INSERT INTO @db
SELECT name FROM sys.databases WHERE state_desc=‘ONLINE‘ AND database_id !=2;
 
 
WHILE (1=1)
BEGIN
    SELECT TOP 1 @db_name = database_name FROM @db ORDER BY 1;
    
    IF @@ROWCOUNT = 0 RETURN;
 
    SET @sql_text =N‘USE ‘ + @db_name +‘;
                     TRUNCATE TABLE #TabColums;
 
    
                    INSERT INTO #TabColums
                    SELECT  object_id ,
                            column_id
                    FROM    sys.columns
                    WHERE   DATALENGTH(name) != LEN(name) * 2;
                
                    SELECT  ‘‘‘ + @db_name  + ‘‘‘ AS DatabaseName,
                            TL.name AS TableName ,
                            C.name AS FieldName ,
                            T.name AS DataType ,
                            DATALENGTH(C.name) AS COLUMN_DATALENGTH ,
                            LEN(C.name) AS COLUMN_LENGTH ,
                            CASE WHEN C.max_length = -1 THEN ‘‘Max‘‘
                                    ELSE CAST(C.max_length AS VARCHAR)
                            END AS Max_Length ,
                            CASE WHEN C.is_nullable = 0 THEN ‘‘ב‘
                                    ELSE ‘‘√‘‘
                            END AS Is_Nullable ,
                            C.is_identity ,
                            ISNULL(M.text, ‘‘‘‘) AS DefaultValue ,
                            ISNULL(P.value, ‘‘‘‘) AS FieldComment
                    FROM    sys.columns C
                            INNER JOIN sys.types T ON C.system_type_id = T.user_type_id
                            LEFT  JOIN dbo.syscomments M ON M.id = C.default_object_id
                            LEFT  JOIN sys.extended_properties P ON P.major_id = C.object_id
                                                                    AND C.column_id = P.minor_id
                            INNER JOIN sys.tables TL ON TL.object_id = C.object_id
                            INNER JOIN #TabColums TC ON C.object_id = TC.object_id
                                                        AND C.column_id = TC.column_id
                    ORDER BY C.column_id ASC;‘;
        PRINT(@sql_text);
 
        EXECUTE(@sql_text);
 
        DELETE FROM @db WHERE [email protected]_name;
 
END
 
TRUNCATE TABLE #TabColums;
DROP TABLE #TabColums;

另外,对应表名而言,可以使用下面脚本。在此略过,不做过多介绍!

DECLARE @db_name  NVARCHAR(32);
DECLARE @sql_text NVARCHAR(MAX);
 
DECLARE @db TABLE 
(
    database_name  NVARCHAR(64)
);
 
 
 
INSERT INTO @db
SELECT name FROM sys.databases WHERE state_desc=‘ONLINE‘ AND database_id !=2;
 
 
WHILE (1=1)
BEGIN
    SELECT TOP 1 @db_name = database_name FROM @db ORDER BY 1;
    
    IF @@ROWCOUNT = 0 RETURN;
 
    SET @sql_text =N‘USE ‘ + @db_name +‘;
 
                                    
                    SELECT ‘‘‘ + @db_name  + ‘‘‘  as database_name, name, 
                         DATALENGTH(name) as table_name_bytes,
                         LEN(name)          as table_name_character,
                         type_desc,create_date,modify_date 
                    FROM sys.tables
                    WHERE   DATALENGTH(name) != LEN(name) * 2;
                   ‘;
        PRINT(@sql_text);
 
        EXECUTE(@sql_text);
 
        DELETE FROM @db WHERE [email protected]_name;
 
END

原文地址:https://www.cnblogs.com/sjqq/p/9757507.html

时间: 2024-08-07 13:01:19

SQL Server如何查找表名或列名中包含空格的表和列的相关文章

SQL获取数据库名,表名,列名,说明等信息

由于最近工作涉及SQL语句较多,对一些不常见的SQL函数.及存储过程下面进行整理和描述,供大家分享: /************************************************************************************************************** 1.获取所有用户名: islogin='1' :表示帐户 islogin='0' :表示角色 status='2' :表示用户帐户 status='0' :表示系统帐户 **

PL/SQL Developer_如何快速获得表名或列名的文本形式

原创作品,出自 "深蓝的blog" 博客,欢迎转载,转载时请务必注明以下出处,否则追究版权法律责任. 深蓝的blog:http://blog.csdn.net/huangyanlong/article/details/41513537 实验:使用PL/SQL Developer软件获得表名.列名的文本形式 操作过程: 例1:以"逗号"格式获取"用户表名"的文本 使用scott登陆到PL/SQL Developer,在All objects框处找到

Sql Server用户名和登录名的关系总结

Sql Server用户名和登录名的关系总结 以前经常被Sql Server中的用户名和登录名搞迷糊,因为用sa(登录名)就搞定一切东西了,当然这会存在一些安全隐患.网上的文章也貌似讲得很好,但还是不明白.今天决心把这个问题弄明白.mashallah, 看了N多文章加实验后终于弄明白了(学东西最重要的还是要亲自动手试试),现将学到的分享一下,看是否对你有帮助. 登录名:服务器方的一个实体,使用一个登录名只能进入服务器,但是不能让用户访问服务器中的数据库资源.每个登录名的定义存放在master数据

获得目标服务器中所有数据库名、表名、列名

1 /// <summary> 2 /// 获得目标服务器所有数据库名 3 /// </summary> 4 /// <param name="serverName"></param> 5 /// <param name="userName"></param> 6 /// <param name="password"></param> 7 public

SQL SERVER 2008 服务器登录名、角色、数据库用户、角色、架构的关系

sql server登录名.服务器角色.数据库用户.数据库角色.架构区别联系 1.一个数据库用户可以对应多个架构(架构是表容器).架构里面包含的是数据库表. 2.一个数据库角色有可能涉及多个架构.数据库角色对应的是权限. 3.一个用户对应一个数据库角色. 4.登录名与数据库用户在服务器级别是一对多的:在数据库级别是一对一的. 服务器登录名:指有权限登录到某服务器的用户: 服务器角色:指一组固定的服务器用户,默认有9组: 登录名一定属于某些角色,默认为public 服务器角色不容许更改 登录后也不

PowerDesigner16 生成的备注脚本,在sql server 2008 中报“对象名 &#39;sysproperties&#39; 无效”的错误的解决办法

主要是在建模时我们对表.列增加了些说明注释,而Sql2005之后系统表sysproperties已废弃删除而改用sys.extended_properties所致. 1.修改Table TableComment 部分 菜单Database -> Edit Current DBMS 窗体 General 选项卡 下 Script -> Objects -> Table –> TableComment 在开头部分替换如下: [if exists (select 1           

PostgreSQL-JDBC疑似bug:部分接口参数的表名、列名必须全部小写

项目从Oracle移植到PostgreSQL(9.4版)后,这几天又出现故障,经跟踪定位,确定原因是调用PgDatabaseMetaData.getPrimaryKeys()接口返回了空集. 众所周知,大多数情况下SQL语句对表名.列名都是大小写不敏感(据本人经验,linux平台的MySql默认对表名区分大小写,可算是个例外).对应的,各数据库JDBC也理应对大小写不敏感,但实际情况是:PostgreSQL的JDBC的部分接口只认全部小写的表名.列名,而对全大写.大小写混合的情况都不支持. Po

mybatis动态传入表名、列名

原文:http://luoyu-ds.iteye.com/blog/1517607 要实现动态传入表名.列名,需要做如下修改 添加属性statementType="STATEMENT" (可省略) 同时sql里的属有变量取值都改成${xxxx},而不是#{xxx} <delete id="deleteTableData" parameterType="java.util.Map" statementType="STATEMENT&

SQL SERVER 2008 服务器登录名、角色,数据库用户、角色、架构知识点总结

SQL SERVER 2008 服务器登录名.角色,数据库用户.角色.架构的关系可以用下面一张图表示: 用例过程如下: 新建服务器登录名: 登录名角色选定: 新建数据库DB1: 然后新建数据库架构Schema1 再新建数据库用户名  User1 服务器登录名属于某组服务器角色:服务器登录名需要于数据库的用户映射后才拥有操作数据库的权限数据库用户属于某组数据库角色以获取操作数据库的权限数据库角色拥有对应的数据库架构,数据库用户可以通过角色直接拥有架构 服务器角色 public sysadmin--