SQL Server如何查找表名或列名中包含空格的表和列
这篇文章给大家分享的是有关SQL Server如何查找表名或列名中包含空格的表和列的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。
如下案例所示:
USETEST;GO--表TEST_COLUMN中两个字段都包含有空格CREATETABLETEST_COLUMN("ID"INTIDENTITY(1,1),[Name]VARCHAR(32),[Normal]VARCHAR(32));GO--表[TEST_TABLE]中包含空格,里面对应三个字段,一个前面包含空格(后面详细阐述),一个字段中间包含空格,一个字段后面包含空格。CREATETABLE[TEST_TABLE]([F_NAME]NVARCHAR(32),[MNAME]NVARCHAR(32),[L_NAME]NVARCHAR(32))GO
实现方法:
那么要如何找出表名或字段名包含空格的相关信息呢? 不管是常规方法还是正则表达式,这个都会效率不高。我们可以用一个取巧的方法,就是通过字段的字符数和字节数的规律来判断,如果没有包含空格,那么列名的字节数和字符数满足下面规律(表名也是如此):
DATALENGTH(name)=2*LEN(name)
SELECTname,DATALENGTH(name)ASNAME_BYTES,LEN(name)ASNAME_CHARACTERFROMsys.columnsWHEREobject_id=OBJECT_ID('TEST_COLUMN');clip_image001
原理是这样的,保存这些元数据的字段类型为sysname ,其实这个系统数据类型,用于定义表列、变量以及存储过程的参数,是nvarchar(128)的同义词。所以一个字母占2个字节。那么我们安装这个规律写了一个脚本来检查数据中那些表名或字段名包含空格。方便巡检。如下测试所示
IFOBJECT_ID('tempdb.dbo.#TabColums')ISNOTNULLDROPTABLEdbo.#TabColums;CREATETABLE#TabColums(object_idINT,column_idINT)INSERTINTO#TabColumsSELECTobject_id,column_idFROMsys.columnsWHEREDATALENGTH(name)!=LEN(name)*2SELECTTL.nameASTableName,C.NameASFieldName,T.NameASDataType,DATALENGTH(C.name)ASCOLUMN_DATALENGTH,LEN(C.name)ASCOLUMN_LENGTH,CASEWHENC.Max_Length=-1THEN'Max'ELSECAST(C.Max_LengthASVARCHAR)ENDASMax_Length,CASEWHENC.is_nullable=0THEN'×'ELSEN'√'ENDASIs_Nullable,C.is_identity,ISNULL(M.text,'')ASDefaultValue,ISNULL(P.value,'')ASFieldCommentFROMsys.columnsCINNERJOINsys.typesTONC.system_type_id=T.user_type_idLEFTJOINdbo.syscommentsMONM.id=C.default_object_idLEFTJOINsys.extended_propertiesPONP.major_id=C.object_idANDC.column_id=P.minor_idINNERJOINsys.tablesTLONTL.object_id=C.object_idINNERJOIN#TabColumsTCONC.object_id=TC.object_idANDc.column_id=TC.column_idORDERBYC.Column_IdASC
那么为什么表名TEST_TABLE的三个字段里面,前面包含空格与与中间包含空格都识别不出来呢?这个与数据库的LEN函数有关系,LEN函数返回指定字符串表达式的字符数,其中
不包含尾随空格。所以这个脚本是无法排查表名或字段名前面包含空格的。如果要排查这种情况,就需要使用下面SQL脚本(中间包含空格在此略过,这个不符合命名规则):
SELECT*FROMsys.columnsWHERENAMELIKE'%'--字段前面包含空格。
其实到了这一步,还没有完,如果一个实例,里面有十几个数据库,那么使用上面这个脚本,我要切换数据库,执行十几次,对于我这种懒人来说,我觉得无法忍受的。那么必须写
一个脚本,将所有数据库全部检查完。本来想用sys.sp_MSforeachdb,但是这个内部存储过程有一些限制,遂写了下面脚本。
DECLARE@db_nameNVARCHAR(32);DECLARE@sql_textNVARCHAR(MAX);DECLARE@dbTABLE(database_nameNVARCHAR(64));IFOBJECT_ID('tempdb.dbo.#TabColums')ISNOTNULLDROPTABLEdbo.#TabColums;CREATETABLE#TabColums(object_idINT,column_idINT);INSERTINTO@dbSELECTnameFROMsys.databasesWHEREstate_desc='ONLINE'ANDdatabase_id!=2;WHILE(1=1)BEGINSELECTTOP1@db_name=database_nameFROM@dbORDERBY1;IF@@ROWCOUNT=0RETURN;SET@sql_text=N'USE'+@db_name+';TRUNCATETABLE#TabColums;INSERTINTO#TabColumsSELECTobject_id,column_idFROMsys.columnsWHEREDATALENGTH(name)!=LEN(name)*2;SELECT'''+@db_name+'''ASDatabaseName,TL.nameASTableName,C.nameASFieldName,T.nameASDataType,DATALENGTH(C.name)ASCOLUMN_DATALENGTH,LEN(C.name)ASCOLUMN_LENGTH,CASEWHENC.max_length=-1THEN''Max''ELSECAST(C.max_lengthASVARCHAR)ENDASMax_Length,CASEWHENC.is_nullable=0THEN''×''ELSE''√''ENDASIs_Nullable,C.is_identity,ISNULL(M.text,'''')ASDefaultValue,ISNULL(P.value,'''')ASFieldCommentFROMsys.columnsCINNERJOINsys.typesTONC.system_type_id=T.user_type_idLEFTJOINdbo.syscommentsMONM.id=C.default_object_idLEFTJOINsys.extended_propertiesPONP.major_id=C.object_idANDC.column_id=P.minor_idINNERJOINsys.tablesTLONTL.object_id=C.object_idINNERJOIN#TabColumsTCONC.object_id=TC.object_idANDC.column_id=TC.column_idORDERBYC.column_idASC;';PRINT(@sql_text);EXECUTE(@sql_text);DELETEFROM@dbWHEREdatabase_name=@db_name;ENDTRUNCATETABLE#TabColums;DROPTABLE#TabColums;
另外,对应表名而言,可以使用下面脚本。在此略过,不做过多介绍!
DECLARE@db_nameNVARCHAR(32);DECLARE@sql_textNVARCHAR(MAX);DECLARE@dbTABLE(database_nameNVARCHAR(64));INSERTINTO@dbSELECTnameFROMsys.databasesWHEREstate_desc='ONLINE'ANDdatabase_id!=2;WHILE(1=1)BEGINSELECTTOP1@db_name=database_nameFROM@dbORDERBY1;IF@@ROWCOUNT=0RETURN;SET@sql_text=N'USE'+@db_name+';SELECT'''+@db_name+'''asdatabase_name,name,DATALENGTH(name)astable_name_bytes,LEN(name)astable_name_character,type_desc,create_date,modify_dateFROMsys.tablesWHEREDATALENGTH(name)!=LEN(name)*2;';PRINT(@sql_text);EXECUTE(@sql_text);DELETEFROM@dbWHEREdatabase_name=@db_name;END
感谢各位的阅读!关于“SQL Server如何查找表名或列名中包含空格的表和列”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。