高级SQL
约 2764 个字 118 行代码 预计阅读时间 21 分钟
-
介绍
本章介绍SQL的高级特性,包括: - 连接表达式 - 事务 - 视图 - 索引 - 完整性约束 - SQL数据类型 - 授权
连接表达式(Join Expressions)
-
连接操作基础
- 连接操作接受两个关系并返回一个结果关系
- 这些操作通常作为from子句中的子查询表达式使用
- 连接类型:定义如何处理在另一个关系中没有匹配元组的元组
- 连接条件:定义两个关系中哪些元组匹配,以及结果中包含哪些属性
- 自然连接消除重复属性
-
外连接(Outer Join)类型
外连接保留在另一个关系中没有匹配的元组:
- 左外连接(Left Outer Join):保留左侧关系中所有元组
- 右外连接(Right Outer Join):保留右侧关系中所有元组
- 全外连接(Full Outer Join):保留两个关系中所有元组
未匹配的元组在结果中相应属性用null填充
-
连接语法示例
自然连接(Natural Join)
使用using子句
course FULL OUTER JOIN prereq USING (course_id) -- 避免自然连接潜在危险的例子 SELECT name, title FROM (student NATURAL JOIN takes) JOIN course USING (course_id)
使用on子句
事务(Transactions)
-
事务概念
- 事务是由一系列查询和/或更新语句组成的工作单元
- 事务具有原子性:要么完全执行,要么回滚(就像从未发生过一样)
- 事务在并发环境中彼此隔离
-
事务控制
- 事务隐式开始
- 事务结束方式:
COMMIT WORK
:事务执行的更新在数据库中永久保存ROLLBACK WORK
:撤销事务中所有SQL语句执行的更新- 大多数数据库默认:每个SQL语句自动提交
- 可以为会话关闭自动提交(通过API)
- SQL:1999标准中可以使用:
BEGIN ATOMIC ... END
(大多数数据库不支持)
视图(Views)
-
视图概念
- 在某些情况下,不希望所有用户都看到整个逻辑模型(即数据库中存储的所有实际关系)
- 视图提供了一种机制,可以对某些用户隐藏某些数据
- 视图是一种"虚拟关系",不属于概念模型但对用户可见
-
视图定义
使用
CREATE VIEW
语句定义视图:其中
<查询表达式>
是任何合法的SQL表达式,v
代表视图名称。一旦定义了视图,视图名称可以用来引用该视图生成的虚拟关系。
视图定义不同于通过评估查询表达式创建新关系,而是保存表达式,在使用视图的查询中替换该表达式。
-
视图示例
不含薪资的教师视图
查找所有生物系教师
系部薪资总和视图
CREATE VIEW departments_total_salary(dept_name, total_salary) AS SELECT dept_name, SUM(salary) FROM instructor GROUP BY dept_name
查询薪资总和高于平均值的系
-
视图依赖关系
- 一个视图可以在定义另一个视图的表达式中使用
- 视图关系v1直接依赖于视图关系v2,如果v2用于定义v1的表达式中
- 视图关系v1依赖于视图关系v2,如果v1直接依赖于v2或存在从v1到v2的依赖路径
- 如果视图依赖于自身,则称为递归视图
-
视图扩展
定义以其他视图为基础的视图含义的方法:
- 假设视图v1由表达式e1定义,e1可能包含对视图关系的使用
- 视图扩展重复以下替换步骤:
- 在e1中找到任何视图关系vi
- 将视图关系vi替换为定义vi的表达式
- 重复直到e1中不再存在视图关系
只要视图定义不是递归的,这个循环最终会终止
-
视图更新
对视图的插入、更新和删除操作会转换为对基本关系的操作。
例如:向之前定义的faculty视图添加新元组
这个插入必须表示为向instructor关系插入元组
大多数SQL实现仅允许对简单视图进行更新: - from子句只有一个数据库关系 - select子句只包含关系的属性名称,没有表达式、聚合或distinct说明 - 未在select子句中列出的任何属性都可以设置为null - 查询没有group by或having子句
-
带检查选项的视图
例如,创建历史系教师视图:
如果我们尝试插入生物系教师到历史系教师视图中:
为避免插入非历史系教师,可以在创建视图时使用WITH CHECK OPTION:
-
物化视图
- 物化视图:创建一个物理表,包含定义视图的查询结果中的所有元组
- 如果查询中使用的关系被更新,物化视图结果会过时
- 需要维护视图,在基础关系更新时更新视图
索引(Index)
完整性约束(Integrity Constraints)
-
完整性约束概念
完整性约束防止数据库意外损坏,确保对数据库的授权更改不会导致数据一致性丢失。
例如: - 支票账户必须有大于\(10,000.00的余额 - 银行员工的薪资必须至少为每小时\)4.00 - 客户必须有(非空)电话号码
-
SQL中的约束类型
- NOT NULL
- PRIMARY KEY
- UNIQUE
- CHECK(P),其中P是谓词
-
NOT NULL约束
声明name和budget不能为空:
-
UNIQUE约束
UNIQUE(A1, A2, ..., Am)规定属性A1, A2, ... Am形成候选键。
候选键允许为空(与主键相反)。
-
CHECK约束
CHECK(P)指定谓词P,关系中的每个元组都必须满足这个谓词。
例如,确保semester是fall、winter、spring或summer之一:
-
外键约束
确保在一个关系中出现的某组属性的值也出现在另一个关系中的某组属性中。
例如:如果"Biology"是instructor关系中某个元组的系名,那么在department关系中必须存在"Biology"系的元组。
定义:设A是一组属性。设R和S是包含属性A的两个关系,且A是S的主键。如果R中出现的A的任何值在S中也出现,则A称为R的外键。
外键可以为空。
示例:
CREATE TABLE course ( course_id char(5) PRIMARY KEY, title varchar(20), dept_name varchar(20) REFERENCES department )
或
CREATE TABLE course ( course_id char(5) PRIMARY KEY, title varchar(20), dept_name varchar(20), FOREIGN KEY (dept_name) REFERENCES department ON DELETE CASCADE ON UPDATE CASCADE )
CASCADE的替代操作:SET NULL, SET DEFAULT
-
外键的递归依赖
例如:
CREATE TABLE person ( ID char(10), name char(40), mother char(10), father char(10), PRIMARY KEY ID, FOREIGN KEY father REFERENCES person, FOREIGN KEY mother REFERENCES person )
如何插入元组而不违反约束? - 在插入person之前插入其father和mother - 或者,最初将father和mother设置为null,在插入所有persons后更新(如果father和mother属性声明为NOT NULL则不可行)
-
高级约束声明
使用检查子句中的子查询:
不幸的是:几乎所有数据库都不支持检查子句中的子查询
另一种方式:使用断言
例如: - 对于student关系中的每个元组,tot_cred属性的值必须等于该学生成功完成的课程学分总和 - 教师在同一学期的同一时间段不能在两个不同的教室授课
但也不被任何数据库系统支持
SQL数据类型
-
日期和时间类型
- date:日期,包含(4位)年、月和日
-
例如:
date '2005-7-27'
-
time:一天中的时间,以小时、分钟和秒为单位
-
例如:
time '09:00:30'
或time '09:00:30.75'
-
timestamp:日期加一天中的时间
-
例如:
timestamp '2005-7-27 09:00:30.75'
-
interval:时间段
- 例如:
interval '4 days 3 hours 2 minutes 1 second'
- 从另一个日期/时间/时间戳值中减去日期/时间/时间戳值会得到间隔值
- 间隔值可以添加到日期/时间/时间戳值
-
用户定义类型
SQL中的
CREATE TYPE
构造创建用户定义类型:使用示例:
-
用户定义域
SQL-92中的
CREATE DOMAIN
构造创建用户定义域类型:类型和域类似。域可以有约束,例如NOT NULL:
-
大对象类型
大对象(照片、视频、CAD文件等)作为大对象存储:
-
blob:二进制大对象 - 对象是大量未解释的二进制数据(解释留给数据库系统外部的应用程序)
-
clob:字符大对象 - 对象是大量字符数据
当查询返回大对象时,返回指针而不是大对象本身。
-
授权(Authorization)
-
授权类型
数据库部分的授权形式: - 读(Read) - 允许读取但不允许修改数据 - 插入(Insert) - 允许插入新数据,但不允许修改现有数据 - 更新(Update) - 允许修改但不允许删除数据 - 删除(Delete) - 允许删除数据
修改数据库模式的授权形式: - 索引(Index) - 允许创建和删除索引 - 资源(Resources) - 允许创建新关系 - 变更(Alteration) - 允许在关系中添加或删除属性 - 删除(Drop) - 允许删除关系
-
GRANT语句
GRANT语句用于授予授权:
其中
是: - 用户ID - PUBLIC,允许所有有效用户授予的权限 - 角色(稍后详述) 在视图上授予权限不意味着授予对基础关系的任何权限。
权限的授予者必须已经持有指定项目的权限(或是数据库管理员)。
-
权限类型
- select:允许对关系的读访问,或使用视图查询的能力
-
例如:授予用户U1、U2和U3对instructor关系的select授权:
-
insert:插入元组的能力
-
update:使用SQL update语句更新元组或某些属性的能力
-
delete:删除元组的能力
-
all privileges:作为所有允许权限的简写形式
-
REVOKE语句
REVOKE语句用于撤销授权:
例如:
可以是all以撤销被撤销者可能持有的所有权限 - 如果
包括public,所有用户都会失去权限,除非明确授予他们 - 如果同一权限由不同的授予者两次授予同一用户,则在撤销后用户可能保留权限
- 所有依赖于被撤销权限的权限也会被撤销
-
角色(Roles)
角色是一种区分不同用户可以访问/更新数据库内容的方式。
创建角色:
例如:
一旦创建了角色,我们可以使用以下方式将"用户"分配给角色:
例如:
可以向角色授予权限:
角色可以授予给用户,也可以授予给其他角色:
CREATE ROLE teaching_assistant; GRANT teaching_assistant TO instructor; -- instructor继承teaching_assistant的所有权限
角色链:
-
视图与授权
例如:
CREATE VIEW geo_instructor AS (SELECT * FROM instructor WHERE dept_name = 'Geology'); GRANT SELECT ON geo_instructor TO geo_staff;
当geo_staff成员发出以下命令时:
如果: - geo_staff对instructor没有权限? - 视图创建者对instructor没有某些权限?
-
其他授权特性
-
references权限用于创建外键:
为什么需要这个? -
权限传递:
-