4.4 数据表的创建和管理

4.4.1 数据表结构的创建

数据表是数据库中的一个数据对象,主要存储各种类型的数据。创建数据表,首先要规划数据内容,定义数据结构。基本表的创建定义中包含了若干列的定义和若干完整性约束。在SQL Server 2000中,每个数据库中最多可以创建200万个表,用户创建数据库表时,最多可以定义1024列,也就是可以定义1024个字段。

SQL Server 2000提供了两种方法创建数据库表:第一种方法是利用对象资源管理器(Enterprise Manager)创建表,另一种方法是利用Transact-SQL语句中的CREATE命令创建表。

1.CREATE TABLE 语句

语法格式:

CREATE TABLE <表名>
(
    <字段名> <数据类型>[列级完整性约束条件]
    [,<字段名> <数据类型>[列级完整性约束条件] ...]
    [,<表级完整性约束条件>]
)

参数说明:

①<表名>是所要定义的基本表的名字。一个表可以由一个或多个属性组成。

②<字段名>一般取有实际意义的名字。

③<数据类型>可以是前面介绍的数据类型。

④在SQL Server 2000中有如下几种完整性约束条件:空值约束(NULL or NOT NULL)、主键约束(PRIMARY KEY CONSTRAINT)、唯一性约束(UNIQUE CONSTRAINT)、检查约束(CHECK CONSTRAINT)、缺省约束(DEFAULT CONSTRAINT)、外部键约束(FOREIGN KEY CONSTRAINT)等。另外还有用规则对象和默认值对象实现约束。因规则的功能可以用CHECK约束实现,默认值的功能也可以用缺省约束实现,故下面对其不赘述。

2.关于创建表时运用约束的说明

(1)空值约束(NULL或NOT NULL)

空值NULL约束决定属性值是否允许为空值(NULL)。NULL表示没有输入任何内容,它不是零和空白。不允许为空值则用NOT NULL表示。

(2)主键约束(PRIMARY KEY CONSTRAINT)

主键约束要求主键属性取值必须唯一,一个表只能包含一个主键约束。如果没有在主键约束中指定CLUSTERED或NONCLUSTERED,并且没有为UNIQUE约束指定聚集索引,则将对该主键约束用CLUSTERED。

主键约束SQL的语法形式如下:

[CONSTRAINT 约束名] PRIMARY KEY [CLUSTERED|NONCLUSTERED](列名[,…n])

(3)唯一性约束(UNIQUE CONSTRAINT)

唯一性约束用于指定一个或者多个列的组合的值具有唯一性,以防止在列中输入重复的值。当使用唯一性约束时,需要考虑以下因素:

①使用唯一性约束的字段允许为空值;

②一个表中可以允许有多个唯一性约束;

③可以把唯一性约束定义在多个字段上;

④唯一性约束用于强制在指定字段上创建一个唯一性索引;

⑤在默认情况下,创建的索引类型为非聚簇索引。

创建唯一性约束的SQL语句如下:

[CONSTRAINT 约束名] UNIQUE [CLUSTERED|NONCLUSTERED](列名[,...n])

(4)检查约束(CHECK CONSTRAINT)

使用检查约束时,应该注意以下几点:

①一个列级检查约束只能与限制的字段有关,一个表级检查约束只能与限制的表中字段有关;

②一个表中可以定义多个检查约束;

③在每个CREATE TABLE语句中,每个字段只能定义一个检查约束;

④在多个字段上定义检查约束,必须将检查约束定义为表级约束;

⑤检查约束中不能包含子查询。

创建检查约束的SQL语法格式如下:

CONSTRAINT CONSTRAINT_name
CHECK [NOT FOR REPLICATION] (logical_expression)

(5)默认约束(DEFAULT CONSTRAINT)

使用默认约束时,应该注意以下几点:

①每个字段只能定义一个默认约束;

②如果定义的默认值长于其对应字段的允许长度,那么输入到表中的默认值将被截断;

③不能加入到带有IDENTITY属性或数据类型为timestamp的字段上;

④如果字段定义为用户定义的数据类型,并且该数据类型绑定到这个字段上,则不允许该字段有默认约束。

(6)外部键约束

外部键约束用于强制参照完整性,提供单个字段或者多个字段的参照完整性。当使用外部键约束时,应该考虑以下几个因素:

①外部键约束提供了字段参照完整性;

②外部键从句中的字段数目和每个字段指定的数据类型必须和REFERENCES从句中的字段相匹配;

③外部键约束不能自动创建索引,需要用户手动创建;

④一个表中最多可以有31个外部键约束;

⑤在临时表中,不能使用外部键约束;

⑥主键和外部键的数据类型必须严格匹配。

⑦如果需要级联修改和删除,要使用ON UPDATE CASCADE ON DELETE CASCADE

外部键约束SQL的语法形式如下:

[CONSTRAINT 约束名] FOREIGN KEY (外键列名)REFERENCES 参照表(参照列名)
[ON UPDATE CASCADE ON DELETE CASCADE]

3.数据表结构创建实例

【例4-1】创建数据库“教学管理”的数据表,包括学生表、课程表、教师表、开课表和选课表,数据表结构如表1-3、表1-4、表1-5、表1-6和表1-7所示,各表的完整性约束如表1-8、表1-9、表1-10、表1-11、表1-12所示。

方法一:使用对象资源管理器创建。

(1)第一步 设置学生表的结构。

①在树状目录中找到要建表的数据库。

②在该数据库上单击鼠标右键,在弹出的快捷菜单中选择“新建”→“表”命令,如图4-4所示。出现创建数据表结构的表设计器窗口,如图4-5所示。

图4-4 新建数据表

图4-5 利用表设计器创建表结构

③表设计器的上半部分有一个表格,在这个表格中输入列的属性,表格的每一行对应一列。对每一列都需要进行设置,其中前3项是必须在建表时给出的。

  • 列名:也称为属性名,可以直接输入。
  • 数据类型:数据类型是一个下拉列表框,其中包括了所有的系统数据类型和数据库中的用户自定义数据类型。
  • 长度:如果选择的数据类型需要长度,则指定长度。
  • 允许空:单击鼠标,可以切换是否允许为空值的状态,勾选说明允许为空值,空白说明不允许为空值,默认状态下是允许为空值的。
  • 如果该列有默认值,可在列属性的“默认值或绑定”处输入。例如,性别的默认值是“男”。

④设置完成后,单击工具栏上的“保存”按钮。在出现的选择名称对话框中输入表名“学生表”。

⑤单击“确定”按钮退出。

(2)第二步 设置学生表完整性约束。

①展开对象资源管理器的数据库“教学管理”,单击“表”结点,选定学生表,单击右键,在快捷菜单上单击“修改”,出现如图4-5所示的创建表结构窗口。

②设置主键约束(PRIMARY KEY),将光标移到需要设置主键的“学号”字段,单击右键,出现下拉菜单,如图4-6所示。然后选择“设置主键”,“学号”列名左侧出现“钥匙”图标。

图4-6 表设计器下拉快捷菜单

如果要设置多属性作为主键,可以按住Ctrl键,用鼠标左键依次单击要选定的列,选定多列后,单击右键出现下拉菜单,如图4-6所示。然后选择“设置主键”。取消主键设置的方法是,选定主键字段,单击右键出现下拉菜单,然后选择“移除主键”。

设置主键后,系统自动建立一个索引。

③在图4-5所示的表设计器上右击,出现下拉快捷菜单,如图4-6所示。

④选择“索引/键”选项卡,如图4-7所示,系统设置“学生表.学号”属性为主键,因此自动在表中建立一个根据学号值的大小升序排列的索引,主键索引名为“PK_学生表”。

⑤设置检查约束(CHECK),学生表定义了三个CHECK约束,第一个约束是学号,第二个约束是身份证号,第三个约束是移动电话。

单击图4-6所示的表设计器下拉快捷菜单中的“CHECK约束”,出现如图4-8所示的“CHECK约束”属性对话框,在其中新建约束。

图4-7 “索引/键”选项卡

图4-8 “CHECK约束”属性对话框

建立学号的约束:单击“添加”按钮,系统自动给定一个约束名,可在“标识-(名称)”处改名为“CK_学生表_学号”,然后在“常规-表达式”行单击,出现约束表达式文本框,输入“学号 LIKE 'S[0-9][0-9][0-9][0-9][0-9][0-9]'”。

建立身份证约束:单击“添加”按钮,系统自动给定一个约束名,同上,改名为“CK_学生表_身份证号”,然后在约束表达式文本框中输入“身份证号 LIKE '[0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9][0−9] [0−9][0−9][0−9][0−9]'”。

建立移动电话的约束与上类似。(考虑到后续数据的模拟性,身份证号和移动电话约束可暂不定义)。

(3)第三步 创建课程表结构,设置相应的约束,过程同学生表。

(4)第四步 创建教师表结构,设置相应的约束。

(5)第五步 创建开课表结构,设置相应的约束。

(6)第六步 创建选课表结构,设置相应的约束。

(7)第七步 设置各表之间的关系,建立相应的外键约束。

“教学管理”数据库中有5个表。其中,教师表、课程表和开课表有联系,开课表中的“课号”和“工号”是关于课程表和教师表的外键。同理,学生表、开课表和选课表有联系,选课表中的“学号”和“开课号”是关于学生表和开课表的外键。

设置步骤如下:

①明确开课表和教师表关于工号的参照关系。开课表中的属性工号参照教师表中的属性工号的值,则开课表为外键表,开课表.工号为外键,教师表为主键表,教师表.工号是主键。

②单击图4-6所示表设计器下拉快捷菜单中的“关系”,出现“外键关系”对话框,单击“添加”按钮,系统自动给出一个关系名,然后在“常规-表和列规范”行单击,出现如图4-9所示的教师表“关系”选项卡。在“关系名”处修改关系名称,选择主键表为教师表,外键表为开课表,选择两表的属性为工号。设置情况如图4-9所示。如果两个数据表有引用关系,那么只要在其中的一个表中建立外键约束(FOREIGN KEY),与其有关的另一个数据表中就会出现同名的FOREIGN KEY约束。

打开开课表的属性对话框,单击“关系”选项卡,就可以看到已经建立好的开课表和教师表的外键约束,设置情况如图4-10所示。在“常规-表和列规范”行单击,出现与图4-9一样的选项卡。

图4-9 教师表“关系”选项卡

图4-10 开课表“关系”对话框

③单击“关闭”按钮退回到表设计器。

④同理,选择课程表,建立课程表和开课表的外键约束关系。

至此,开课表的两个外键全部建好。

⑤选择学生表,建立学生表和选课表的外键约束关系。

⑥选择开课表,建立开课表和选课表的外键约束关系。

至此,选课表的两个外键也全部建好。

方法二:使用SQL命令创建。

创建表并包含完整性约束定义,同时定义各个约束名。考虑到后续数据的模拟性,身份证号、移动电话的检查约束不进行定义,实际中需要时可参考学号、工号等进行定义。

①创建学生表的语句如下:

CREATE TABLE 学生表
(
  学号 CHAR(7) NOT NULL,
  身份证号 CHAR(18) NOT NULL,
  姓名 CHAR(8) NOT NULL,
  性别 CHAR(2) DEFAULT '男',
  移动电话 CHAR(11),
  籍贯 VARCHAR(10),
  专业 VARCHAR(20) NOT NULL,
  所在院系 VARCHAR(20) NOT NULL,
  累计学分 INT,
  CONSTRAINT PK_学生表_学号  PRIMARY KEY(学号),
  CONSTRAINT CK_学生表_学号  CHECK(学号 LIKE 'S[0-9][0-9][0-9][0-9][0-9][0-9]')
)

说明:约束名可以不定义;约束可以直接跟在列后。例如:

CREATE TABLE 学生表
(
  学号CHAR(7) NOT NULL PRIMARY KEY(学号) CHECK(学号LIKE 'S)0-9)[0-9][0-9][0-9]
  [0-9][0-9]'),
  身份证号CHAR(18) NOT NULL,
  姓名CHAR(8) NOT NULL,
  性别CHAR(10) DEFAULT '男',
  移动电话CHAR(11),
  籍贯VARCHAR(10),
  专业VARCHAR(20) NOT NULL,
  所在院系VARCHAR(20) NOT NULL,
  累计学分INT
)

②创建课程表的语句如下:

CREATE TABLE 课程表
(
  课号 CHAR(6) NOT NULL,
  课名 VARCHAR(30) NOT NULL,
  学分 INT CHECK(学分>=1 and 学分<=5),
  教材名称 VARCHAR(30),
  编著者 CHAR(8),
  出版社 VARCHAR(20),
  版号 VARCHAR(20),
  定价 money,
  CONSTRAINT PK_课程表_课号  PRIMARY KEY(课号),
  CONSTRAINT CK_课程表_课号  CHECK(课号 LIKE 'C[0-9][0-9][0-9][0-9][0-9]')
)

③创建教师表的语句如下:

CREATE TABLE 教师表
(
  工号 CHAR(6) NOT NULL,
  身份证号 CHAR(18) NOT NULL,
  姓名 CHAR(8) NOT NULL,
  性别 CHAR(2) DEFAULT '男',
  移动电话 CHAR(11),
  籍贯 VARCHAR(10),
  所在院系 VARCHAR(20) NOT NULL,
  职称 CHAR(6),
  负责人 CHAR(6),
  CONSTRAINT PK_教师表_工号  PRIMARY KEY(工号),
  CONSTRAINT CK_教师表_工号 CHECK(工号 LIKE 'T[0-9][0-9][0-9][0-9][0-9]')
)

④创建开课表的语句如下:

CREATE TABLE 开课表
(
  开课号 CHAR(6) NOT NULL,
  课号 CHAR(6) NOT NULL,
  工号 CHAR(6) NOT NULL,
  开课地点 CHAR(6),
  开课学年 CHAR(9),
  开课学期 INT ,
  开课周数 INT DEFAULT 17,
  开课时间 VARCHAR(20),
  限选人数 INT,
  已选人数 INT,
  CONSTRAINT PK_开课表_开课号 PRIMARY KEY(开课号),
  CONSTRAINT FK_开课表_工号 FOREIGN KEY(工号) REFERENCES 教师表(工号)
  ON UPDATE CASCADE ON DELETE CASCADE,
  CONSTRAINT FK_开课表_课号 FOREIGN KEY(课号) REFERENCES 课程表(课号)
  ON UPDATE CASCADE ON DELETE CASCADE,
  CONSTRAINT CK_开课表_开课号 CHECK(开课号 LIKE '[0-9][0-9][0-9][0-9][0-9][0-9]'),
  CONSTRAINT CK_开课表_工号 CHECK(工号 LIKE 'T[0-9][0-9][0-9][0-9] [0-9]'),
  CONSTRAINT CK_开课表_课号 CHECK(课号 LIKE 'C[0-9][0-9][0-9][0-9][0-9]')
)

⑤创建选课表的语句如下:

CREATE TABLE 选课表
(
  学号 CHAR(7) NOT NULL,
  开课号 CHAR(6) NOT NULL,
  成绩 INT CHECK(成绩>=0 and 成绩<=100),
  CONSTRAINT PK_选课表_学号_开课号 PRIMARY KEY(学号,开课号),
  CONSTRAINT FK_选课表_学号 FOREIGN KEY(学号) REFERENCES 学生表(学号)
  ON UPDATE CASCADE ON DELETE CASCADE,
  CONSTRAINT FK_选课表_开课号 FOREIGN KEY(开课号) REFERENCES 开课表(开课号)
  ON UPDATE CASCADE ON DELETE CASCADE,
  CONSTRAINT CK_选课表_学号 CHECK(学号 LIKE 'S[0-9][0-9][0-9][0-9][0-9][0-9]'),
  CONSTRAINT CK_选课表_开课号 CHECK(开课号 LIKE '[0-9][0-9][0-9][0-9][0-9][0-9]')
)

4.4.2 数据表结构的管理

1.修改数据表结构

表结构创建以后,在使用的过程中经常会发现原来创建的表可能存在结构、约束等方面的问题,在这种情况下,需要对原表进行修改。如果用创建一个新表的方法替换原表,将造成表中数据的丢失,而通过修改表则可以在保留表中原有数据的基础上修改表结构,打开、关闭或删除已有约束,或增加新的约束等。

修改表结构有两种方法:一种是利用对象资源管理器,另一种是使用SQL命令。

利用对象资源管理器修改表结构的过程如图4-11所示。单击“修改”命令,将弹出表设计器。

图4-11 修改学生表结构

使用SQL命令修改表,语法格式如下:

ALTER TABLE <表名>
    ADD<新列名><数据类型>[完整性约束]
    |ALTER COLUMN 列名 新类型
    |DROP COLUMN 列名
    |ADD PRIMARY KEY(列名[,...])
    |ADD FOREIGN KEY(列名) REFERENCES 表名(列名)
    |ADD CONSTRAINT 约束名....
    |DROP CONSTRAINT 约束名

图4-12 修改学生表某些属性长度

【例4-2】将教学管理数据库中学生表的“性别”属性的长度改为2。

方法一:使用对象资源管理器,如图4-12所示。

方法二:使用SQL命令,程序如下。

USE 教学管理                          --打开教学管理数据库
GO
ALTER TABLE 学生表
      ALTER COLUMN 性别 CHAR(2)     --修改属性列“性别”

【例4-3】向教学管理数据库中的学生表增加“入学时间”属性,其数据类型为日期型,增加“年龄”属性,其类型为整型,取值为13~70。

方法一:使用对象资源管理器,如图4-13所示。

然后,单击右键,选择“CHECK约束”,出现图4-14所示对话框,建立年龄的约束:单击“添加”按钮,系统自动给定一个约束名,可在“标识-(名称)”处改名为“CK_学生表_年龄”,然后在“常规-表达式”行单击,出现约束表达式文本框,输入“年龄>=13 AND年龄<70”。

方法二:使用SQL命令。

USE 教学管理
GO
ALTER TABLE 学生表
    ADD  入学时间  DATETIME,
        年龄 INT CONSTRAINT CK_学生表_年龄CHECK(年龄>=13 AND年龄<70)

图4-13 向学生表增加新属性列

图4-14 选项卡中输入约束名和约束表达式

注意:新增加的属性列不能定义为“NOT NULL”;新增加的属性可以带有主键约束、参照约束、CHECK约束和默认值。

【例4-4】删除学生表中的属性列年龄。

方法一:使用对象资源管理器。

①展开对象资源管理器的数据库结点,过程如图4-11所示。选择学生表,单击“修改”命令,将弹出表设计器。

②首先删除“年龄”属性的CHECK约束。在设计器窗口单击右键,在快捷菜单上选择“CHECK约束”,选择CK_学生表_年龄约束,如图4-14所示,单击“删除”按钮即可删除。

③选定要删除的“年龄”属性列,如图4-15所示,单击右键,再单击快捷菜单中的“删除”按钮即可删除该属性列。

图4-15 删除数据表的属性列

方法二:使用SQL命令。

USE 教学管理
GO
ALTER TABLE 学生表 DROP CONSTRAINT CK_学生表_年龄    --删除约束
ALTER TABLE 学生表 DROP COLUMN年龄                 --删除去除了约束的属性

或者

ALTER TABLE 学生表
      DROP CONSTRAINT  CK_学生表_年龄,               --删除约束
      COLUMN年龄                                       --删除去除了约束的属性

注意:跟属性列有关的约束和索引删除后,指定的属性才能删除。

2.数据表结构的删除

可以用对象资源管理器或SQL语句删除基本表。

SQL命令的一般格式如下:

DROP  TABLE  <表名>

方法一:使用对象资源管理器。

①选中要删除的数据表,单击右键,在快捷菜单(如图4-10所示)上选择“删除”。

②在“除去对象”对话框中,单击“全部除去”。

方法二:使用SQL命令。

【例4-5】创建一个教室信息表,然后删除它。

USE 教学管理
GO
--创建教室表
CREATE TABLE 教室表
( 编号      INT             IDENTITY,
   名称      VARCHAR(20)   NOT NULL,
   位置      CHAR(6)       NULL,
)
GO
--删除教室表
DROP TABLE  教室表

注意:当删除一个表时,表的定义和表中的所有数据,以及该表的索引、权限设置、约束等均被自动删除,与该表相绑定的规则和默认对象失去与它的绑定关系;但是,使用DROP TABLE语句不能删除系统表和被FOREIGN KEY约束所参照的用户表,必须先删除引用的外键约束或引用的表。