什么是Oracle Key-Preserved Table和什么样的视图可以进行DML操作

本文探讨了Oracle数据库中视图的DML操作限制,特别是关于Key-Preserved Table的概念及其对视图更新的影响。通过具体实例说明哪些类型的表可以在视图层面进行修改,并解释了相关错误信息。
在通过DML操作Oracle的视图的时候,有一个很重要的条件是操作的base表必须是Key-Preserved Table.
那么,什么是Key-Preserved Table呢.Oracle给出的定义是:
A table is key preserved if every key of  the table can also be a key of the result of the join.
It is not necessary that the key or keys of a table be selected for it to be key preserved.

It is sufficient that if the key or keys were selected, then they would also be key(s) of the result of the join.

如果某一个表的主键可以作为这个join结果(view通常是几个表的join结果)的主键,那么这个表就是key preserved table.

这个表的主键并非一定要出现在select出来的结果集中(select list里面),但是如果其出现在结果集中,那么它必须可以满足作为这个结果集的主键的要求.

来看一个例子,有这样两个表,dept和employee,以及基于这两个表的testv视图
create table dept(deptid int primary key,deptname varchar2(20))
create table employee(empid int primary key,empname varchar2(20),deptid int)

insert into dept values(1,'dept1');
insert into dept values(2,'dept2');
insert into dept values(3,'dept3');

insert into employee values(1,'emp1',1);
insert into employee values(2,'emp2',1);
insert into employee values(3,'emp3',2);

create view testv 
as select d.deptid deptid,deptname,empid,empname,e.deptid edeptid
from dept d join employee e
on d.deptid=e.deptid
查询这个视图:select * from testv


在testv这个视图中,employee就是一个key preserved table,而dept不是.
那么这个视图可以进行的DML为
delete from testv where empid=1(操作的结果是将employee表中的empid=1的记录delete了,dept表不会有改变)
delete from testv where deptid=1(操作的结果是将employee表中的empid=1和2的记录都delete了,dept表不会有改变)
update testv set empname='empx' where edeptid=1
update testv set empname='empx' where empid=1
update testv set empname='empx' where deptid=1
insert into testv(empid,empname,edeptid) values(4,'emp4',2)
这个视图不可以进行的DML为
update testv set deptname='deptx' where deptid=1
update testv set deptname='deptx' where empid=1
insert into testv(deptid,deptname) values(4,'dept4')
ORA-01779: cannot modify a column which maps to a non key-preserved table
一个View中可以有多个key preserved tables

create table test1(t1id int primary key,t1v varchar2(10))
create table test2(t2id int primary key,t2v varchar2(10))

insert into test1 values(1,'t1');
insert into test1 values(2,'t2');
insert into test1 values(3,'t3');

insert into test2 values(1,'t21');
insert into test2 values(2,'t22');
insert into test2 values(3,'t23');

create view test1v 
as select t1id ,t1v,t2id,t2v
from test1 join test2
on test1.t1id=test2.t2id

在这个视图中,test1,test2都是key preserved table.但是对view的DML操作一次只能作用于一个表,不能同时对多个base 表操作.
select * from test1v


如下面的DML是可以的
update test1v set t1v='t11'
update test1v set t2v='t22'
insert into test1v(t1id,t1v) values(4,'t4')
insert into test1v(t2id,t2v) values(4,'t24')
delete from test1v where t1id=4
delete from test1v where t2id=4(这两条delete语句都是将test1表中的数据delete了??)
但是下面的DML操作是不可以的:
update test1v set  t1v='t11',t2v='t22'
insert into test1v values(5,'t5',5,'t25')
ORA-01776: cannot modify more than one base table through a join view
对于View是否可以进行DML操作还有一些其他限制:
不能有start with和connect by.
不能有group by和having.
不能有集合操作union,intersect和minus.
不能有聚合函数,如AVG,COUNT,MAX等等.
不能有rownum伪列.
当view有check option的时候:
create view test1v 
as select t1id ,t1v,t2id,t2v
from test1 join test2
on test1.t1id=test2.t2id
with check option
insert into test1v(t1id,t1v) values(4,'t4')
update test1v set t1id=4 where t1id=1
ORA-01733: virtual column not allowed here

如何查看那些字段可以插入更新

select * from USER_UPDATABLE_COLUMNS where table_name='TESTV'


这个错误信息通常出现在SQL数据库操作中,特别是在使用视图(View)进行查询时。它的意思是无法从没有保留主键的视图或连接中选择rowid或进行采样。以下是对这个错误的详细解释: 1. **视图的定义**:视图是基于一个或多个表的虚拟表。它本身不存储数据,而是存储查询逻辑。 2. **主键保留(Key Preserved Table)**:在视图的上下文中,主键保留表指的是在视图的查询结果中,主键能够唯一标识每一行的表。如果视图中的某个表的主键在查询结果中仍然唯一标识每一行,那么这个表就是主键保留表。 3. **rowid**:在许多SQL数据库(如Oracle)中,rowid是一个唯一的标识符,用于标识表中的每一行。如果视图没有保留主键,那么数据库无法确定每一行的唯一标识符,从而导致无法选择rowid。 4. **采样(Sampling)**:采样操作通常需要能够唯一标识每一行,以便随机选择或分批处理数据。如果视图没有保留主键,数据库无法保证采样的正确性。 ### 解决方法 1. **确保视图中有主键保留表**:在创建视图时,确保至少有一个表的主键在视图的结果中保留下来。 2. **使用替代方法**:如果无法修改视图,可以考虑在查询中使用其他唯一标识符来代替rowid。 3. **重新设计视图**:如果视图的设计无法满足需求,可能需要重新设计视图,确保其包含主键保留表。 ### 示例 假设我们有两个表:`employees` `departments`,并且我们创建了一个视图 `employee_departments`: ```sql CREATE VIEW employee_departments AS SELECT e.id, e.name, d.department_name FROM employees e JOIN departments d ON e.department_id = d.id; ``` 在这个视图中,`employees` 表的主键 `id` 仍然唯一标识每一行,因此 `employee_departments` 视图有一个主键保留表 `employees`。如果我们在 `employee_departments` 视图中尝试选择 `rowid`,它将不会报错: ```sql SELECT rowid, id, name, department_name FROM employee_departments; ``` 但是,如果我们在视图中没有保留主键,例如: ```sql CREATE VIEW employee_departments_no_key AS SELECT e.name, d.department_name FROM employees e JOIN departments d ON e.department_id = d.id; ``` 在这个视图中,没有保留 `employees` 表的主键 `id`,因此尝试选择 `rowid` 将会报错。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值