原创文章,禁止转载。
更新视图时也会修改对应的基本表,但有时更新视图会产生报错。
不可更新视图的原因
1.视图中包含的字段不是基础表中的直接字段
如果视图 S_G 中的字段 Gavg 是由聚合函数(如 AVG())计算而来,那么它不直接对应于基础表中的某一列。在这种情况下,数据库无法确定如何将对 Gavg 的更新操作传递到基础表。
示例:
CREATE VIEW S_G AS
SELECT Sno, AVG(Grade) AS Gavg
FROM SC
GROUP BY Sno;
- Gavg 是从 SC 表中的 Grade 列计算得出的。
- AVG() 是聚合函数,结果无法对应到基础表 SC 的单个字段。
- 更新 Gavg 无法被转换为对 SC 表的更新操作,因此视图不可更新。
2.视图中包含的字段来自多表
如果视图是基于多表连接创建的,例如:
CREATE VIEW S_G AS
SELECT SC.Sno, SC.Grade, Students.Sname
FROM SC
JOIN Students ON SC.Sno = Students.Sno
GROUP BY SC.Sno, Students.Sname;
- 此视图包含了多个表的字段(SC 和 Students)。
- 数据库无法唯一确定更新操作应该应用到哪个基础表,因此视图不可更新。
3.视图中包含计算字段或虚拟列
如果视图中的字段是计算得来的,而不是直接存储在基础表中,例如:
CREATE VIEW S_G AS
SELECT Sno, Grade * 0.1 AS WeightedGrade
FROM SC;
- WeightedGrade 是基于 Grade 计算得出的字段。
- 更新 WeightedGrade 无法映射到基础表,因为它没有对应的存储字段。
4.视图定义中使用了DISTINCT、GROUP BY、HAVING 等
-
这些操作会聚合或过滤数据,使得更新视图的单条记录时无法准确对应到基础表。
-
数据库无法确定如何将更新操作转换为基础表的更新。
可更新视图的条件
1.视图是简单视图,仅基于单个基础表。
2.视图的所有字段都直接对应于基础表的字段。
3.视图定义中不包含以下元素:
聚合函数(如 SUM()、AVG() 等)
计算字段
DISTINCT
GROUP BY、HAVING
多表连接
实例演示
1. 创建表 SC
这是一个记录学生编号、课程编号及其分数的表。
CREATE TABLE SC (
Sno VARCHAR(10) NOT NULL, -- 学生编号
Cno VARCHAR(10) NOT NULL, -- 课程编号
Grade INT -- 分数
);

2. 插入测试数据
插入一些示例数据供测试。
INSERT INTO SC (Sno, Cno, Grade) VALUES
('200215121', 'C101', 85),
('200215121', 'C102', 90),
('200215121', 'C103', 88),
('200215122', 'C101', 78),
('200215122', 'C102', 82),
('200215122', 'C103', 79);

3. 创建基于 AVG 的视图 S_G
该视图计算每位学生的平均分数。
CREATE VIEW S_G AS
SELECT Sno, AVG(Grade) AS Gavg
FROM SC
GROUP BY Sno;
- 视图
S_G包含两列:Sno:学生编号。Gavg:每位学生的平均分数。

4. 尝试更新视图
尝试通过视图更新平均分 Gavg:
UPDATE S_G
SET Gavg = 90
WHERE Sno = '200215121';
报错:

这是由于 Gavg 是通过聚合函数 AVG(Grade) 计算得来的字段,并不直接存在于基础表 SC 中。数据库无法将对 Gavg 的更新转换为对基础表的更新。
426

被折叠的 条评论
为什么被折叠?



