一. 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
--
--执行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
但以上的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,以此类推)