Mysql视图

Posted 路漫漫其修远兮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Mysql视图相关的知识,希望对你有一定的参考价值。

一:视图的概述

视图是一个虚拟表,其内容由查询定义。同真实的表一样,视图包含一系列带有名称的列和行数据。但是,视图并不在数据库中以存储的数据值集形式存在。行和列数据来自由定义视图的查询所引用的表,并且在引用视图时动态生成。

对其中所引用的基础表来说,视图的作用类似于筛选。定义视图的筛选可以来自当前或其它数据库的一个或多个表,或者其它视图。通过视图进行查询没有任何限制,通过它们进行数据修改时的限制也很少。

视图是存储在数据库中的查询的SQL 语句,它主要出于两种原因:安全原因, 视图可以隐藏一些数据,如:社会保险基金表,可以用视图只显示姓名,地址,而不显示社会保险号和工资数等,另一原因是可使复杂的查询易于理解和使用。这个视图就像一个“窗口”,从中只能看到你想看的数据列。这意味着你可以在这个视图上使用SELECT *,而你看到的将是你在视图定义里给出的那些数据列:

二:为什么要用视图

1、 视图能简化用户操作

视图机制使用户可以将注意力集中在所关心地数据上。如果这些数据不是直接来自基本表,则可以通过定义视图,使数据库看起来结构简单、清晰,并且可以简化用户的的数据查询操作。例如,那些定义了若干张表连接的视图,就将表与表之间的连接操作对用户隐藏起来了。换句话说,用户所作的只是对一个虚表的简单查询,而这个虚表是怎样得来的,用户无需了解。

2、 视图使用户能以多种角度看待同一数据

视图机制能使不同的用户以不同的方式看待同一数据,当许多不同种类的用户共享同一个数据库时,这种灵活性是非常必要的。

3、 视图对重构数据库提供了一定程度的逻辑独立性

数据的物理独立性是指用户的应用程序不依赖于数据库的物理结构。数据的逻辑独立性是指当数据库重构造时,如增加新的关系或对原有的关系增加新的字段,用户的应用程序不会受影响。层次数据库和网状数据库一般能较好地支持数据的物理独立性,而对于逻辑独立性则不能完全的支持。

在关系数据库中,数据库的重构造往往是不可避免的。重构数据库最常见的是将一个基本表“垂直”地分成多个基本表。例如:将学生关系Student(Sno,Sname,Ssex,Sage,Sdept),

分为SX(Sno,Sname,Sage)和SY(Sno,Ssex,Sdept)两个关系。这时原表Student为SX表和SY表自然连接的结果。如果建立一个视图Student:

CREATE VIEW Student(Sno,Sname,Ssex,Sage,Sdept)AS SELECT SX.Sno,SX.Sname,SY.Ssex,SX.Sage,SY.Sdept FROM SX,SY 
WHERE SX.Sno=SY.Sno

这样尽管数据库的逻辑结构改变了(变为SX和SY两个表了),但应用程序不必修改,因为新建立的视图定义为用户原来的关系,使用户的外模式保持不变,用户的应用程序通过视图仍然能够查找数据。

当然,视图只能在一定程度上提供数据的逻辑独立,比如由于视图的更新是有条件的,因此应用程序中修改数据的语句可能仍会因为基本表构造的改变而改变。

4、视图能够对机密数据提供安全保护

有了视图机制,就可以在设计数据库应用系统时,对不同的用户定义不同的视图,使机密数据不出现在不应该看到这些数据的用户视图上。这样视图机制就自动提供了对机密数据的安全保护功能。例如,Student表涉及全校15个院系学生数据,可以在其上定义15个视图,每个视图只包含一个院系的学生数据,并只允许每个院系的主任查询和修改本原系学生视图。

5、适当的利用视图可以更清晰地表达查询

例如经常需要执行这样的查询“对每个学生找出他获得最高成绩的课程号”。可以先定义一个视图,求出每个同学获得的最高成绩:

CREATE VIEW VMGRADE
AS
SELECT Sno,MAX(Grade) Mgrade
FROM SC
GROUP BY Sno

然后用如下的查询语句完成查询:

SELECT SC.Sno,Cno FROM SC,VMGRADE WHERE SC.Sno = VMGRADE.Sno AND SC.Grade = VMGRADE.Mgrade;

三:准备数据

在创建视图前应先看看是否有权限:

SELECT SELECT_priv,create_view_priv from mysql.user WHERE user=‘root‘;

Y表示有创建的权限

技术分享

1:员工表

CREATE TABLE t_employee(
        ID INT  PRIMARY KEY  AUTO_INCREMENT,
        NAME CHAR(30) NOT NULL,
        SEX  CHAR(2) NOT NULL,
        AGE INT NOT NULL,
        DEPARTMENT CHAR(10) NOT NULL,
        SALARY  INT NOT NULL,
        HOME CHAR(30),
        MARRY CHAR(2) NOT NULL DEFAULT  ‘否‘,       
        HOBBY CHAR(30)
 );

插入数据:

INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY) 
VALUES(NULL,‘小红‘,‘女‘,20,‘人事部‘,‘4000‘,‘广东‘,‘否‘,‘网球‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY) 
VALUES(NULL,‘明日‘,‘女‘,21,‘人事部‘,‘9000‘,‘北京‘,‘否‘,‘网球‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY) 
VALUES(NULL,‘天天‘,‘男‘,22,‘研发部‘,‘8000‘,‘上海‘,‘否‘,‘音乐‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY)
 VALUES(NULL,‘大大‘,‘女‘,23,‘研发部‘,‘9000‘,‘重庆‘,‘否‘,‘无‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY)
 VALUES(NULL,‘王下‘,‘女‘,24,‘研发部‘,‘9000‘,‘四川‘,‘是‘,‘足球‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY)
 VALUES(NULL,‘无名‘,‘男‘,25,‘销售部‘,‘6000‘,‘福建‘,‘否‘,‘游戏‘);
INSERT INTO t_employee(ID, NAME, SEX, AGE,DEPARTMENT, SALARY, HOME, MARRY, HOBBY) 
VALUES(NULL,‘不知道‘,‘女‘,26,‘销售部‘,‘5000‘,‘山西‘,‘否‘,‘篮球‘);

查询看结果

技术分享

然后再定义一张员工信息表:

create TABLE t_employee_detail(
ID INT PRIMARY KEY,
POS CHAR(10) NOT NULL,
EXPERENCE CHAR(10) NOT NULL,
CONSTRAINT `FK_ID` FOREIGN KEY(ID) REFERENCES t_employee(ID)
);

插入数据:

INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(1,‘人事管理‘,‘工作二年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(2,‘人事招聘‘,‘工作二年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(3,‘初级工程师‘,‘工作一年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(4,‘中级工程师‘,‘工作二年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(5,‘高级工程师‘,‘工作三年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(6,‘销售代表‘,‘工作二年‘);
INSERT INTO t_employee_detail(ID,POS,EXPERENCE) VALUES(7,‘销售员‘,‘工作一年‘);

查询看结果

技术分享

四:开始创建视图

①创建视图

在员工表上创建视图

CREATE VIEW V_VIEW1(ID, NAME, SEX, AGE,DEPARTMENT) AS SELECT ID, NAME, SEX, AGE,DEPARTMENT FROM t_employee;
select * from V_VIEW1;  //查看视图数据

技术分享

②多表上创建视图

CREATE VIEW V_VIEW2(ID, NAME, SEX, AGE,DEPARTMENT,POS,EXPERENCE) AS SELECT a.ID, a.NAME, a.SEX, a.AGE,
a.DEPARTMENT,b.POS,b.EXPERENCE FROM t_employee a,t_employee_detail b WHERE a.ID=b.ID;
select * from V_VIEW2;    //查看视图数据

技术分享

③查看视图

(1)DESCRIBE 命令

技术分享

(2)SHOW TABLE STATUS

SHOW TABLE STATUS LIKE ‘V_VIEW2‘;

在这里就不显示了,自己拿数据测试一下

(3)SHOW CREATE view命令

SHOW CREATE VIEW V_VIEW2;

在这里就不显示了,自己拿数据测试一下

④修改视图

(1)CREATE OR REPLACE命令
CREATE OR REPLACE VIEW V_VIEW1(ID, NAME, SEX) AS SELECT ID, NAME, SEX  FROM t_employee;
SELECT * FROM V_VIEW1;

技术分享

(2) ALTER 命令

ALTER VIEW  V_VIEW1(ID, NAME) AS SELECT ID, NAME  FROM t_employee;
SELECT * FROM V_VIEW1;

技术分享

⑤更新视图

在MySQL中,更新视图是指通过视图来插入(INSERT)、更新(UPDATE)和删除(DELETE)表中的数据。因为视图是一个虚拟表,其中没有数据,所以通过视图更新时,都是转换到基本表来更新。

更新视图时,只能更新权限范围内的数据。超出了范围,就不能更新。

SELECT * FROM V_VIEW2;  //查看更新前

技术分享

UPDATE V_VIEW2 SET POS=‘高级工程师‘ WHERE NAME=‘天天‘;

更新后

技术分享

对应 的真实表上的数据也发生改变 了

SELECT * FROM t_employee_detail WHERE t_employee_detail.ID=3;

技术分享

不可更新的视图:

某些视图是可更新的。也就是说,可以在诸如UPDATE、DELETE或INSERT等语句中使用它们,以更新基表的内容。对于可更新的视图,在视图中的行和基表中的行之间必须具有一对一的关系。还有一些特定的其他结构,这类结构会使得视图不可更新。更具体地讲,如果视图包含下述结构中的任何一种,那么它就是不可更新的:

· 聚合函数(SUM(), MIN(), MAX(), COUNT()等)。

· DISTINCT

· GROUP BY

· HAVING

· UNION或UNION ALL

· 位于选择列表中的子查询

· Join

· FROM子句中的不可更新视图

· WHERE子句中的子查询,引用FROM子句中的表。

· 仅引用文字值(在该情况下,没有要更新的基本表)。

· ALGORITHM = TEMPTABLE(使用临时表总会使视图成为不可更新的)。

注意

视图中虽然可以更新数据,但是有很多的限制。一般情况下,最好将视图作为查询数据的虚拟表,而不要通过视图更新数据。

因为,使用视图更新数据时,如果没有全面考虑在视图中更新数据的限制,就可能会造成数据更新失败。

①创建不可更新的视图:(使用临时表的算法)

create algorithm=temptable view v_view4 as select * from t_employee;
show create view v_view4\G;

View: v_view4

Create View: CREATE ALGORITHM=TEMPTABLE DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `v_view4` AS select `t_employee`.`ID` AS `ID`,`t_employee`.`NAME` AS `NAME`,`t_employee`.`SEX` AS `SEX`,`t_employee`.`AGE` AS `AGE`,`t_employee`.`DEPARTMENT` AS `DEPARTMENT`,`t_employee`.`SALARY` AS `SALARY`,`t_employee`.`HOME` AS `HOME`,`t_employee`.`MARRY` AS `MARRY`,`t_employee`.`HOBBY` AS `HOBBY` from `t_employee`
character_set_client: gbk
collation_connection: gbk_chinese_ci

尝试更新视图,报错,(一定程度上保证了基表数据的安全性)

update v_view4 set NAME = ‘xxx‘ where ID=1;

ERROR 1288 (HY000): The target table v_view4 of the UPDATE is not updatable

更改视图的算法:(原来是temptable,改为merge,从而视图变成可以更新了) 

 

alter algorithm=merge view v_view4 as select * from t_employee;
update v_view4 set NAME = ‘xxx‘ where ID=1;  //视图和表的数据都更新了

Query OK, 1 row affected (0.00 sec)

Rows matched: 1  Changed: 1  Warnings: 0

以上是关于Mysql视图的主要内容,如果未能解决你的问题,请参考以下文章

从mysql的片段中加载ListView

回收站视图未显示在片段中

如何从片段内的列表视图打开链接网址?

如何使列表视图出现在片段中?

如何从活动中更改片段中视图的可见性

片段中的网格视图