zoukankan      html  css  js  c++  java
  • 金色的 SQL注意事项(1)

    page(1-75)

    最好是没有意义的主键字段,以方便未来的扩展.
    PS:主键,以后标书编码填错须要改的时候,关联表都须要跟着改.假设是一个无意义的自增字段是主键就无此原因.

    主键最好不要设置为联合主键,否则减少效率,不利于扩展
    PS:原文[联合主键能够解决表中没有唯一主键的问题,只是联合主键有例如以下缺点:]
    1.效率低.在进行数据的加入、删除、查找及更新的时候,数据库系统必须处理俩个字段,这样大大减少了数据的处理速度.
    2.使数据库的结构设计变得槽糕.组成联合主键的字段通常都是有业务含义的字段,这与”使用逻辑主键而不是业务主键”的实践相冲突,easy造成系统开发及维护上的麻烦.
    3.使创建指向此表的外键关联关系变得很麻烦,甚至无法创建指向此表的外键关联关系.
    4.加大开发难度.许多开发工具及框架仅仅对单个主键有良好的支持,对于联合主键常常须要进行很复杂的特殊处理.
    考虑到这些缺点,我们应该仅仅在兼容遗留系统等特殊场合才使用联合主键,而在其他场合则应该使用唯一主键.

    字符类型知识点:
    1.大部分数据库中固定长度字符类型名称为char(使用固定长度字符类型保存数据时,因为剩余部分会以空格填充,那么在读取的字段值时就会将后面填充的空格也读出来.
    2.可变长度字符类型一般为varchar
    PS:固定长度字符类型和可变长度字符类型都仅仅能存储基于ACSII的字符,这样对于使用中文、韩文、日文等UniCode字符集的程序来说将会造成存储问题.为了解决问题,我们能够使用国际化可变长度字符类型,这样的类型能够用俩个字节来保存一个字符.这样就能够解决中文、韩文等字符串保存问题了.在大部分数据库中可变长度字符类型名称为nvarchar.
    可是假设字段中没有存放双字节字符的话,尽量不要使用国际化可变长度字符.
    3.固定长度字符类型和可变长度字符类型一般都不能指定过于大的长度,比方长度超过1024是不同意的.超过这个长度的建议使用大字符类型字段

    SQL执行顺序
    WHERE语句在GROUP BY语句之前;SQL会在分组之前计算WHERE语句。
    HAVING语句在GROUP BY语句之后;SQL会在分组之后计算HAVING语句

    对于分组来说,SELECT和GROUP BY列必须匹配。而SELECT语句包括聚合函数时这一规则是一个例外。

    插入语句
    insert into 表名(字段,字段2,字段3) values(‘值’,’值1’,’值2’)
    PS:忽略字段的话,则会依照定义表中的字段顺序进行插入.建议不使用忽略写法.忽略后不easy查看相应的字段和值的相应easy反倒easy出错.

    select * from People –全部
    select name,age from People –部分列
    select max(age) from People –最大值
    select min(age) from People –最小值
    select avg(age) from People –平均值
    select sum(age) from People –求和
    select count(age) from People –统计记录数量

    select * from T_Employee order by FAge asc –排序 asc升序 默认升序可省略
    select * from T_Employee order by FAge desc –降序
    select * from T_Employee order by FAge desc,FSalary desc –多组排序

    二元操作符 or and 左表达式为待匹配的字段,而右表达式为待匹配的通配符表达式.
    select * from T_Employee where FName like ‘erry’ –单字符匹配的通配符”
    select * from T_Employee where FName like ‘%n_’ –多字符匹配的通配符为”%”

    ————–S
    select * from T_Employee where FName like ‘[SJ]%’ –集合匹配 匹配第一个字符为S或者J长度不限的字符串
    select * from T_Tmployee where FName like ‘^[SJ]%’ –上面的表达式取反 即不包括 S或者J开头的长度不限的字符串 等同于以下表达式
    select * from T_Tmployee where NOT(FName like ‘S%’) and NOT(FName like ‘J%’) 通配符过滤是很强大的功能,只是在使用通配符过滤的时候,数据库系统会对全表进行扫描,所以执行
    速度很慢.因此不要过多的使用通配符过滤.在使用其他方式能够实现效果的时候就应该避免使用通配符过滤

    ————–E

    select * from T_Tmployee where FName IS NULL –空值推断 不要使用时 FName = NULL 去判定这样的写法是错误的!
    select * from T_Tmployee where FName IS NOT NULL –推断不为空的值

    –反义运算符 ‘=’ ‘>’ ‘<’ 等于 大于 小于 能够通过 ‘!’ 来取反 ‘!=’ ‘!>’ ‘!<’ 不等于 不大于 不小于
    select * from T_Tmployee where FAge != 22 and FSALARY != 2000 – 检索年龄不等于22谁 工资不等于2000员的员工
    – 不等于运算分 ‘<>’
    ! 运算符 仅仅能执行在MS SQLSERVER 和 DB2 这俩种数据库上,假设要移植到其他数据库上的话 要避免使用这样的方式.
    同义运算符 能够在全部主流数据库上执行.只是因为粗心等原因 很easy将 ‘不大于’ 表示为 ‘<’ ,从而忘记’不大于’是包括 ‘小于’ 和’等于’这俩个意思.easy出错.
    因此推荐 使用 NOT 运算符.来表示 ‘非’的意思 除了’<>’这样的方式之外

    –多值检測 公司要为23,25,28岁员工发福利 检索 姓名 年龄 工号
    –一般性我们的思路是 用or Fage = 23 or Fage = 25 or Fage = 28 数量一旦变多 反倒不好维护
    – SQL 提供了 in 方式. Fage in (23,25,28) in语句仅仅能进行多个离散值的检測.
    –范围值检測 查询某一区间的值 诸如23-27岁. 用In 则穷举 或者 Fage >=23 and Fage <=27
    在sql中 推荐使用 where between 23 and 27 等同于 Fage >=23 and Fage <= 27 并且性能更高.

    慎用 where 1=1 ,使用where 1=1后 数据库无法使用索引等查询优化策略,数据库将被迫对每条数据进行扫描(也就是全表扫描)以比較此行是否满足过滤条件.数据库较大时会比較慢

    建表及案例数据

    USE [NB]
    GO
    /****** 对象:  User [sasa]    脚本日期: 06/25/2015 10:40:21 ******/
    CREATE USER [sasa] FOR LOGIN [sasa] WITH DEFAULT_SCHEMA=[dbo]
    GO
    /****** 对象:  StoredProcedure [dbo].[SP_Select]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    Create Proc [dbo].[SP_Select]       
    @OName varchar(100)      
    As  
    Declare @Str Varchar(1000),  
      @dbname varchar(40)      
    set @dbname=db_name()      
    Set @Str='Select * from '+@dbname+'.dbo.'+@OName      
    Exec (@Str)
    GO
    /****** 对象:  StoredProcedure [dbo].[sp_syscolumns]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    CREATE PROC [dbo].[sp_syscolumns]   --Exec sp_syscolumns 'eemployee'
    @Object NVARCHAR(1000)
    As  
    /*    
    Function:取得一个对象中的全部列的项目(主要针对表)   
    Remark:  Create By Deam L 2013/4/7
    */
    Begin
        Set nocount on
    
        Declare @Name NVARCHAR(1000)
        Select  @Name=Isnull(@Name+',','')+name From syscolumns Where id=object_id(@Object)
        Print   @Name
    
        Set nocount off
    END
    GO
    /****** 对象:  StoredProcedure [dbo].[前言]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    -- =============================================
    -- Author:      <CXP,,>
    -- Create date: <2014-10-8 09:11:56,,>
    -- Description: <获取OA系统当前申请的採购流程,,>
    -- =============================================
    CREATE PROCEDURE [dbo].[前言]
    AS
    GO
    /****** 对象:  StoredProcedure [dbo].[c_CreateSqlBaseTable]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    
    CREATE PROCEDURE [dbo].[c_CreateSqlBaseTable]
    AS
    BEGIN
        --T_Person为记录人员的数据表 当中主键字段FName为人员姓名,FAge为年龄,FRemark为备注信息,
        --T_Debt为债务信息.当中主键为FNumber为债务编号,FAmount为欠债金额,FPerson为欠债人姓名,
        --FPerson与T_person中FName字段建立了外键关联关系
        CREATE TABLE T_Person(FName VARCHAR(20),FAge INT,FRemark VARCHAR(20),primary KEY(FName))
        CREATE TABLE T_Debt(FNumber VARCHAR(20),FAmount NUMERIC(10,2) NOT NULL
        ,FPerson VARCHAR(20),PRIMARY KEY(FNumber),FOREIGN KEY(FPerson) REFERENCES T_Person(FName))
    
        --插入范例数据
        INSERT INTO T_Person(FName,FAge,FRemark) VALUES('Tom',18,'USA')
        INSERT INTO T_Person(FName,FAge,FRemark) VALUES('Jim',20,'USA')
        INSERT INTO T_Person(FName,FAge,FRemark) VALUES('Lili',22,'China')
        INSERT INTO T_Person(FName,FAge,FRemark) VALUES('XiaoWang',17,'China')
        INSERT INTO T_Person(FName,FAge,FRemark) VALUES('Kimisushi',18,'Korea')
        INSERT INTO T_Person(FAge,FName) VALUES(22,'LXF')
        INSERT INTO T_Person VALUES('lurenl',23,'China') --不推荐此写法,easy出错
    
        INSERT INTO T_Debt(FNumber,FAmount,FPerson) VALUES('1',300,'Jim')
        INSERT INTO T_Debt(FNumber,FAmount,FPerson) VALUES('2',300,'Jim')
        INSERT INTO T_Debt(FNumber,FAmount,FPerson) VALUES('3',100,'Tom')
    END
    GO
    /****** 对象:  Table [dbo].[T_Person]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    SET ANSI_PADDING ON
    GO
    CREATE TABLE [dbo].[T_Person](
        [FName] [varchar](20) NOT NULL,
        [FAge] [int] NULL,
        [FRemark] [varchar](20) NULL,
    PRIMARY KEY CLUSTERED 
    (
        [FName] ASC
    )WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
    ) ON [PRIMARY]
    GO
    SET ANSI_PADDING OFF
    GO
    /****** 对象:  Table [dbo].[T_Debt]    脚本日期: 06/25/2015 10:40:21 ******/
    SET ANSI_NULLS ON
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    SET ANSI_PADDING ON
    GO
    CREATE TABLE [dbo].[T_Debt](
        [FNumber] [varchar](20) NOT NULL,
        [FAmount] [numeric](10, 2) NOT NULL,
        [FPerson] [varchar](20) NULL,
    PRIMARY KEY CLUSTERED 
    (
        [FNumber] ASC
    )WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
    ) ON [PRIMARY]
    GO
    SET ANSI_PADDING OFF
    GO
    /****** 对象:  ForeignKey [FK__T_Debt__FPerson__07020F21]    脚本日期: 06/25/2015 10:40:21 ******/
    ALTER TABLE [dbo].[T_Debt]  WITH CHECK ADD FOREIGN KEY([FPerson])
    REFERENCES [dbo].[T_Person] ([FName])
    GO

    版权声明:本文博主原创文章。博客,未经同意不得转载。

  • 相关阅读:
    maven继承父工程统一版本号
    shiro权限控制参考
    动态查询列表页面的分页
    SVN服务器更改ip地址后怎么办
    cookie记住密码功能
    分享小插件的问题
    阿里云短信验证
    从svn上更新maven项目时,所有文件变成包的形式
    Maven工具
    Mybatis的dao层传递单参出现的问题
  • 原文地址:https://www.cnblogs.com/blfshiye/p/4884792.html
Copyright © 2011-2022 走看看