MSSQL之十一 数据库高级编程总结

网友投稿 806 2022-05-29

视频课 https://edu.csdn.net/course/play/7940

一.  SQL Server数据库命名规范

数据库命名规范:

1. 数据库名:

1.1)用产品或项目的名字命名;

1.2)Pascal Case,如AdventureWork;

1.3)避免使用特殊字符,如数字,下划线,空格之类;

1.4)避免使用缩写

2. 表名

2.1)使用复数,Pascal Case,而复数只加在最后一个单词上如:Products,Users,UserRoles

2.2)避免使用特殊字符,如数字,下划线,空格之类;

2.3)避免使用缩写

3. 列名

3.1) 使用Pascal Case

3.2) 避免和表名重复,避免数据类型前缀如: Int

3.3) 避免使用缩写或者特殊字符

4. 存储过程

4.1)用动词加表名描述操作类型

4.2)使用前缀:sp+{“Insert”, “Update”, “Delete”,“Get”, “Validate”,...}

5.  视图

5.1)参考表名规则

5.2)用"vw"做前缀

6. 触发器

6.1)使用"trg"前缀

6.2) 使用操作类型+表名,如:trg_ProductsInsert

7. 索引

7.1)使用格式如:idx_{表名}_{索引列名}_{Unique/NonUnique}_{Cluster/NonCluster}

8. 主键

8.1) 使用格式如:pk_{表名}_{主键列名}

9. 外键

9.1) 使用格式如:fk_{主表名}_{主表的列名}_{引用表名}_{引用表的列名}

10. default

10.1) 使用格式如:df_{表名}_{列名}

11. 约束

11.1) 使用格式如:ck_{表名}_{列名}

12. 变量

12.1) 参照列名规则

二. 数据库备份

备份处理的存储过程

set ANSI_NULLS ON

set QUOTED_IDENTIFIER ON

go

/*--备份所有数据库

备份的文件名为数据库名+日期+.bak

将所有的用户数据库(或指定的数据库列表)

备分到指定的目录下.

/*--调用示例

--备份所有用户数据库

exec   p_backupdb  @bkpath='D:\',@dbname=''

--备份指定数据库

exec   p_backupdb   @bkpath=D:\',@dbname='数据库名称'

--*/

create  proc   [dbo].[p_backupdb]

@bkpath   nvarchar(260)='D:\', --备份文件的存放目录,不指定则使用SQL默认的备份目录

@dbname   nvarchar(4000)='' --要备份的数据库名称列表,不指定则备份所有用户数据库

as

declare   @sql   varchar(8000)

DECLARE @strdate NVARCHAR(200)

set @strdate = convert(NVARCHAR(10),getdate(),120)

set @strdate = REPLACE(@strdate, '-' , '')

--检查参数

if   isnull(@bkpath,'')=''

begin

select   @bkpath=rtrim(reverse(filename))  from   master..sysfiles   where  name='master'

select  @bkpath=substring(@bkpath,charindex('\',@bkpath)+1,4000)

,@bkpath=reverse(substring(@bkpath,charindex('\',@bkpath),4000))+'BACKUP\'

end

else   if   right(@bkpath,1)<>'\'  set   @bkpath=@bkpath+'\'

--得到要备份的数据库列表

if   isnull(@dbname,'')=''

declare   tb   cursor   local  for

select   name   from  master..sysdatabases   where   name  not   in('master','tempdb','model','msdb')

else

declare   tb   cursor   local  for

select   name   from  master..sysdatabases

where   name   not  in('master','tempdb','model','msdb')  and(name  like '%'+@dbname+'%')

--备份处理

open   tb

fetch   next   from   tb  into   @dbname

while   @@fetch_status=0

begin

set   @sql='backup   database  '+@dbname

+'   to   disk='''+@bkpath+@dbname+'_'+@strdate

+'.bak''   with   format'

exec(@sql)

fetch   next   from   tb  into   @dbname

end

close   tb

deallocate   tb

go

二.      Sql Server 2005的分页存储过程

CREATEPROCEDURE     [dbo].[TopPageList]

@strTable      varchar(200),   --表名 ("@strTable", "myUser");

@strColumn     varchar(50),    --按该列来进行分页("@strColumn", "UserId");

@strOrderColumn  varchar(50),    --排序字段order by XXX desc

@intOrder     int,--排序的顺序 0 升序 1降序

@strColumnlist  varchar(150) , --要查询出的字段列表,*表示全部字段 cmd.Parameters.Add("@strColumnlist", "*");

@strWhere      varchar(800)='',--查询条件cmd.Parameters.Add("@strWhere", "");

@intPageSize    int, --每页记录数  cmd.Parameters.Add("@intPageSize", 15);

@intPageNum    int,           --指定页  cmd.Parameters.Add("@intPageNum", 5);

--  @intPageCount   int   OUTPUT  , --总页数  SqlParameter paramPageCount =cmd.Parameters.Add("@intPageCount", SqlDbType.Int);

-- paramPageCount.Direction = ParameterDirection.Output;

@itemCount      int  OUTPUT

--   @doCount bit = 0, -- 返回, 非值则返回记录总数

AS

--设置相应的空格

--设置DESC ASC

if @intOrder=0  --0升序

set @strOrderColumn=' order by '+@strOrderColumn

else           --降序

set @strOrderColumn=' order by '+@strOrderColumn +' desc '

DECLARE   @sql    nvarchar(2000) --用于构造SQL语句

DECLARE   @where1 varchar(800)   --构造条件语句

DECLARE   @where2 varchar(800)   --构造条件语句

IF   @strWhere   is   null  or   rtrim(@strWhere)=''

-- 为了避免SQL关键字与字段、表名等连在一起,首先为传入的变量添加空格

BEGIN  --没有查询条件

SET   @where1=' WHERE '

SET   @where2=' '

END

ELSE

BEGIN  --有查询条件

SET   @where1=' WHERE  ('+@strWhere+')  AND '

SET   @where2=' WHERE  ('+@strWhere+') '

END

------构造SQL语句,计算总页数。计算公式为总页数= Ceiling ( 记录个数/ 页大小)

--计算总项数

SET   @sql='SELECT   @itemCount=COUNT('+@strColumn+') from  '+@strTable +@where2

print(@sql)

EXEC sp_executesql  @sql,N'@itemCount  int  OUTPUT',@itemCount   OUTPUT

--   1:直接计算    2:自己写个分页控件里面设置一下也可以~!

-- set @intPageCount  =floor(cast(@itemCount as float)/@intPageSize)

--   if @intPageCount

--   set @intPageCount =@intPageCount +1

--

--执行SQL语句,计算总页数,并将其放入@intPageCount变量中

--将总页数放到查询返回记录集的第一个字段前,此语句可省略

SET  @strColumnlist=' '+ Cast(@itemCount as varchar(30)) + ' asitemCount,' +' '+ @strColumnlist

--+ Cast(@intPageCount as varchar(30)) + ' as PageCount,'

SET @sql='SELECT TOP '+ CAST(@intPageSize   AS  varchar)  +  @strColumnlist +

' FROM ' + @strTable + @where1 + '  '+

@strColumn + ' not in  '+

'  (SELECT TOP '+ CAST(@intPageSize*(@intPageNum - 1)  AS varchar) + ' ' +

@strColumn + ' FROM '+ @strTable+@where2+@strOrderColumn+')  ' +@strOrderColumn

print(@sql)

--ELSE

--     begin   --构造降序的SQL---针对2个表的时候会出现聚合函数的异常--适合单个表格的数据库分页操作

--     SET @sql='SELECT TOP '+CAST(@intPageSize   AS   varchar)  + @strColumnlist +

--              ' FROM ' + @strTable + @where1 + '  '+

--             @strColumn + '<(SELECT MIN('+@strColumn+') '+

--             ' FROM (SELECT TOP '+ CAST(@intPageSize*(@intPageNum - 1)  AS varchar) + ' ' +

--             @strColumn + ' FROM '+ @strTable+@where2+@strOrderColumn+')  as tblTmp)' +@strOrderColumn

--    print(@sql)

--     end

IF   @intPageNum=1--第一页

SET   @sql='SELECT  TOP   '+CAST(@intPageSize   AS   varchar) +@strColumnlist + ' FROM'+@strTable+

@where2+@strOrderColumn

--END

--PRINT   @sql

print(@sql)

exec(@sql)

public static void BindingContent(string strTable, string strColumn, stringstrOrderColumn, int intOrder, string strColumnlist, string strWhere,IChangePageStored changePage)

{

SqlParameter[] paras=new SqlParameter[9];

paras[0] =new SqlParameter("@strTable" ,SqlDbType.VarChar);

paras[0].Value = strTable;

paras[1] =new SqlParameter("@strColumn", SqlDbType.VarChar);

paras[1].Value = strColumn;

paras[2] =new SqlParameter("@strOrderColumn", SqlDbType.VarChar);

paras[2].Value = strOrderColumn;

paras[3] =new SqlParameter("@strColumnlist", SqlDbType.VarChar);

paras[3].Value = strColumnlist;

paras[4] =new SqlParameter("@intOrder", SqlDbType.Int);

paras[4].Value = intOrder;

paras[5] =new SqlParameter("@strWhere", SqlDbType.VarChar);

paras[5].Value = strWhere;

paras[6] =new SqlParameter("@intPageSize", SqlDbType.Int);

paras[6].Value = changePage.PageSize;

paras[7] =new SqlParameter("@intPageNum", SqlDbType.Int);

paras[7].Value = changePage.CurrentPage  ;

//   paras[8] = newSqlParameter("@intPageCount", SqlDbType.Int);

//  paras[8].Direction = ParameterDirection.Output;

paras[8] =new SqlParameter("@itemCount", SqlDbType.Int);

paras[8].Direction = ParameterDirection.Output;

DataSet ds =DBTool.ExecuteDataset(CommandType.StoredProcedure, "TopPageList",paras);

/* @intPageCount   int   OUTPUT  , --总页数  SqlParameterparamPageCount = cmd.Parameters.Add("@intPageCount", SqlDbType.Int);

-- paramPageCount.Direction = ParameterDirection.Output;

@strTable = N'zhq_in_content c  INNER JOIN zhp_in_columns  m ONc.columns_id=m.columns_id',

@strColumn = N'c.content_id',

@strOrderColumn = N'c.createdate',

@intOrder = 1,

@strColumnlist = N'*',

@strWhere = N'c.status=0 AND c.del=0',

@intPageSize = 20,

@intPageNum =100,*/

changePage.DataSource = ds.Tables[0].DefaultView;// 设置分页控件的数据

if (ds !=null && ds.Tables[0].Rows.Count > 0)

{

changePage.RecordCount =int.Parse(ds.Tables[0].Rows[0]["itemCount"].ToString());

// changePage.PageCount =int.Parse(ds.Tables[0].Rows[0]["PageCount"].ToString());

}

if(changePage.DataUI.GetType().BaseType.Name == "BaseDataList")

{

changePage.DataUI.DataSource = changePage.DataSource;//设置数据源控件的数据

changePage.DataUI.DataBind();

}

}

四.SQLServer异构数据库之间数据的导入导出

本文讨论了如何通过Transact-SQL以及系统函数OPENDATASOURCE和OPENROWSET在同构和异构数据库之间进行数据的导入导出,并给出了详细的例子以供参考。

1. 在SQL Server数据库之间进行数据导入导出

(1).使用SELECT INTO导出数据

在SQL Server中使用最广泛的就是通过SELECTINTO语句导出数据,SELECT INTO语句同时具备两个功能:根据SELECT后跟的字段以及INTO后面跟的表名建立空表(如果SELECT后是*, 空表的结构和FROM所指的表的结构相同);将SELECT查出的数据插入到这个空表中。在使用SELECT INTO语句时,INTO后跟的表必须在数据库不存在,否则出错,下面是一个使用SELECT INTO的例子。

假设有一个表table1,字段为f1(int)、f2(varchar(50))。

SELECT * INTO table2 FROM table1

这条SQL语的在建立table2表后,将table1的数据全部插入到table1中的,还可以将*改为f1或f2以便向适当的字段中插入数据。

SELECT INTO不仅可以在同一个数据中建立表,也可以在不同的SQL Server数据库中建立表。

USE db1

SELECT * INTO db2.dbo.table2 FROM table1

以上语句在数据库db2中建立了一个所有者是dbo的表table2,在向db2建表时当前登录的用户必须有在db2建表的权限才能建立table2。使用SELECT INTO要注意的一点是SELECT INTO不可以和COMPUTE一起使用,因为COMPUTE返回的是一组记录集,这将会引起二意性(即不知道根据哪个表建立空表)。

(2).使用INSERTINTO和 UPDATE插入和更新数据

SELECT INTO只能将数据复制到一个空表中,而INSERT INTO可以将一个表或视图中的数据插入到另外一个表中。

INSERT INTO table1 SELECT * FROM table2

或 INSERT INTO db2.dbo.table1 SELECT * FROMtable2

MSSQL之十一 数据库高级编程总结

但以上的INSERT INTO语句可能会产生一个主键冲突错误(如果table1中的某个字段是主键,恰巧table2中的这个字段有的值和table1的这个字段的值相同)。因此,上面的语句可以修改为

INSERT INTO table1   -- 假设字段f1为主键

SELECT * FROM table2 WHERE NOT EXISTS(SELECTtable1.f1 FROM table1 WHERE table1.f1=table2.f1 )

以上语句的功能是将table2中f1在table1中不存在的记录插入到table1中。

要想更新table1可以使用UPDATE语句

UPDATE table1 SET table1.f1=table2.f1,table1.f2=table2.f2 FROM table2 WHERE table1.f1=table2.f1

将以上两条INSERT INTO和UPDATE语句组合起来在一起运行,就可以实现记录在table1中不存在时插入,存在时更新的功能,但要注意要将UPDATE放在 INSERT INTO前面,否则UPDATE更新的记录数将是table1和table2记录数的总和。

2. 使用OPENDATASOURCE和OPENROWSET在不同类型的数据库之间导入导出数据

在异构的数据库之间进行数据传输,可以使用SQL Server提供的两个系统函数OPENDATASOURCE和OPENROWSET。

OPENDATASOURCE可以打开任何支持OLE DB的数据库,并且可以将OPENDATASOURCE做为SELECT、UPDATE、INSERT和DELETE后所跟的表名。如

SELECT * FROM OPENDATASOURCE('SQLOLEDB', 'DataSource=192.168.18.252;User ID=sa;Password=test').pubs.dbo.authors

这条语句的功能是查询192.168.18.252这台机器中SQL Server数据库pubs中的authors表。从这条语句可以看出,OPENDATASOURCE有两个参数,第一个参数是 provider_name,表示用于访问数据源的 OLE DB 提供程序的 PROGID 的名称。provider_name 的数据类型为 char,没有默认值。第二个参数是连接字符串,根据OLE DB Provider不同而不同(如果不清楚自己所使用的OLE DBProvider的连接字符串,可以使用delphi、visualstudio等开发工具中的ADO控件自动生成相应的连接字符串)。

OPENROWSET函数和OPENDATASOURCE函数类似,只是它可以在打开数据库的同时对数据库中的表进行查询,如以下语句

OPENROWSET('MSDASQL.1', 'Driver=Microsoft VisualFoxPro Driver; SourceDB=c:\db; SourceType=DBF', SELECT * FROM [b.dbf])

最后一个参数查询foxpro表b.dbf,读者可以通过where条件对b.dbf进行过滤。如果将INSERT INTO、SELECT INTO和OPENDATASOURCE或OPENROWSET一起使用,就可以使SQL Server数据库和其它类型的数据库之间进行数据导入导出。下面介绍如何使用这两个函数在SQL Server数据库和其它类型的数据库之间进行数据导入导出。

(1).SQLServer数据库和SQL Server数据库之间的数据导入导出。

导入数据

SELECT   * INTOauthors1 FROMOPENDATASOURCE( 'SQLOLEDB', 'Data Source=192.168.18.252;UserID=sa;Password=abc').pubs.dbo.authors

导出数据

INSERT INTO OPENDATASOURCE('SQLOLEDB','DataSource=192.168.18.252;User ID=sa;Password=abc').test.dbo.authors select * frompubs.dbo.authors

在这条语句中OPENDATASOURCE(...)可以理解为SQL Server的一个服务,.pubs.dbo.authors是这个服务管理的一个数据库的一个表authors。使用INSERT INTO时OPENDATASOURCE(...)后跟的表必须存在。

也可以将以上的OPENDATASOURCE换成OPENROWSET

INSERT INTO OPENROWSET('SQLOLEDB','192.168.18.252;sa;abc',select * from test.dbo.kk) SELECT * FROM pubs.dbo.authors

使用OPENROWSET要注意一点,192.168.18.252;sa;abc中间是";",而不是","。OPENDATASOURCE和OPENROWSET都不接受参数变量。

(2).SQL Server数据库和Access数据库之间的数据导入导出。

导入数据

SELECT * INTO access FROM OPENDATASOURCE( 'Microsoft.Jet.OLEDB.4.0','Provider=Microsoft.Jet.OLEDB.4.0;Data Source=c:\data.mdb;Persist SecurityInfo=False')...table1

或者使用OPENROWSET

SELECT * FROM  OPENROWSET('Microsoft.Jet.OLEDB.4.0', 'c:\data.mdb;admin;',SELECT * FROMtable1)

导出数据

INSERT INTOOPENDATASOURCE('Microsoft.Jet.OLEDB.4.0','Provider=Microsoft.Jet.OLEDB.4.0;DataSource=c:\data.mdb;Persist Security Info=False')...table1 SELECT * FROM access

打开access数据库的OLE DBProvider叫Microsoft.Jet.OLEDB.4.0,需要注意的是操作非SQL Server数据库在OPENDATASOURCE(...)后面引用数据库中的表时使用"...”,而不是“.”。

(3).SQL Server数据库和文本文件之间的数据导入导出。

导入数据

SELECT * INTO text1 FROMOPENDATASOURCE('MICROSOFT.JET.OLEDB.4.0','Text;DATABASE=c:\')...[data#txt]

导出数据

INSERT INTOOPENDATASOURCE('MICROSOFT.JET.OLEDB.4.0','Text;DATABASE=c:\')...[data#txt]SELECT * FROM text1

或者使用OPENROWSET

INSERT INTOOPENROWSET('MICROSOFT.JET.OLEDB.4.0','Text;DATABASE=c:\, [data#txt]') SELECT *FROM text1

如果要插入部分字段,可使用

INSERT INTOOPENROWSET('MICROSOFT.JET.OLEDB.4.0','Text;DATABASE=c:\, SELECT aa FROM[data#txt]') SELECT aa FROM text1

这条SQL语句的功能是将c盘根目录的data.txt文件导入到text1表中,在这里文件名中的“.”要使用“#”代替。在向文本导出时,不仅文本文件要存在,而且第一行必须和要导出表的字段一至。

(4).SQL Server数据库和dbase数据库之间的数据导入导出。

导入数据

SELECT * INTO dbase FROMOPENROWSET('MICROSOFT.JET.OLEDB.4.0 ', 'dBase III;HDR=NO;IMEX=2;DATABASE=C:\',SELECT* FROM [b.dbf])

导出数据

INSERT INTO OPENROWSET('MICROSOFT.JET.OLEDB.4.0', dBase III;HDR=NO;IMEX=2;DATABASE=C:\,SELECT * FROM [b.dbf]) SELECT * FROMdbase

OPENROWSET(...)中的b.dbf使用[...]括起来,是为了当dbf文件名有空格等字符时不会出错,如果没有这些特殊字符,可以将[...]去掉

(5).SQL Server数据库和foxpro数据库之间的数据导入导出。

导入数据

SELECT * INTO foxpro FROMOPENROWSET('MSDASQL.1',   'Driver=Microsoft Visual FoxProDriver;SourceDB=c:\; SourceType=DBF, 'SELECT * FROM [a.dbf])

导出数据

INSERT INTO OPENROWSET('MSDASQL.1' ,'Driver=Microsoft Visual FoxPro Driver;  SourceDB=c:\db;SourceType=DBF,'SELECT * FROM a.dbf) SELECT * FROM foxpro

在此处a.dbf不能使用[...]括起来,否则出错(这是由driver决定的)。

(6).SQL Server数据库和excel文件之间的数据导入导出

导入数据

SELECT * INTO excel FROMOPENDATASOURCE(MICROSOFT.JET.OLEDB.4.0,Excel 5.0;DATABASE=c:\book1.xls)...[Sheet1$]

导出数据

INSERT INTOOPENDATASOURCE(MICROSOFT.JET.OLEDB.4.0,Excel 5.0;DATABASE=c:\book1.xls)...[Sheet1$] SELECT * FROM excel

在book1.xls的Sheet1中必须有和excel表相对应的字段,否则会出错。

以上讨论了几种常用的数据库和SQL Server数据库之间如何使用Transact-SQL进行数据导入导出。在SQL Server中还提供了将其它类型的数据库注册到SQL Server中的功能,这样就可以和使用SQL Server数据库表一样使用这些被注册数据库中的表了。

EXEC sp_addlinkedserver access,OLE DB Providerfor Jet, Microsoft.Jet.OLEDB.4.0, c:\data.mdb

以上SQL使用存储过程sp_addlinkedserver注册了一个access数据库,我们可以在SQL Server中使用如下语句查询在data.mdb中的table1。

SELECT * FROM access...table1

这样就可很方便地查询access数据库中的表了,如果要导入table1,可以使用SELECT * INTO table2 FROMaccess...table1。如果想删除注册的数据库连接,使用如下语句。

EXEC sp_dropserver access

使用Transact-SQL不仅可以向SQLServer数据库导入导出数据,而且还可以使任意两种类型数据库之间互相导入导出数据。以access和excel为例进行说明。

INSERT INTOOPENDATASOURCE(MICROSOFT.JET.OLEDB.4.0,Excel 5.0;DATABASE=c:\book1.xls)...[Sheet1$] SELECT * FROM OPENROWSET(Microsoft.Jet.OLEDB.4.0,c:\data.mdb;admin;,SELECT * FROM table1)

以上SQL语句将access数据库的table1表的数据插入到excel文件book1.xls中的Sheet1表单中。

使用Transact-SQL进行数据的导入导出,可以很方便地将这些Transact-SQL语句放到客户端程序中(如delphi、c#等),从而可以很容易地编写自已的数据库导入导出工具。

五.无限级分类的数据库设计方案

第一种方案:

表为两张,一张分类表,一张信息表。

表1:

`ID` int(10),

`cID` tinyint(3) ,

`title` varchar(255),

表2:

`cID` tinyint(3) ,

`parentID` tinyint(3),

`order` tinyint(3) ,

`name` varchar(255),

这样可以根据cID = parentID来判断上一级内容,运用递归至最顶层 。

第二种方案:

设置parentID为varchar类型,将父类id都集中在这个字段里,用符号隔开,比如:1,3,6

这样可以比较容易得到各上级分类的ID,而且在查询分类下的信息的时候,可以使用如:Select * From information Where cID Like "1,3%"。不过在添加分类和转移分类的时候操作将非常麻烦。

以上两种方案地址:http://search.phpres.com/phpres-top2007,98552.html

第三种方案:

每级分类递增两位数字,这样,每级分类的数目限定在100个之间,分类方法主要为编码法;

示例:

一级分类:01,02,03

二级分类:0101,0102,0103,0201,0202........

三级分类:010101,010102,010103,010104..........

数据库查询时使用 like '01%'就可得到一级分类01下的所有子分类,非常方便!

如果要列出所有分类的树型结构,只需用一条语句select * from pro_class order bycode,再稍微处理一下就可。(其中,pro_class为产品分类表,code为类别编码)。

设计的数据库结构如下:

id:                   类别id,主键

classname:         类名

classcode:          类别编码

parent:             父id

left_child:          最左孩子id(或第一个孩子)

right_sibling:      右兄弟id

layer:                层级(第一级类别为1,第2级类别2,以此类推)

SQL 数据库

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:30 岁的码农人生 ——人生至暗时,你依然能窥见光明
下一篇:公式树开源库分析
相关文章