【竺】数据库笔记9——视图的应用2

​视图的含义
        视图是一种虚拟的表。视图从数据库中的一个或多个表导出来的表。视图还可以从已经存在的视图的基础
上定义。数据库中只存放了视图的定义,而并没有存放视图中的数据。这些数据存放在原来的表中。使用
视图查询数据时,数据库系统会从原来的表中取出对应的数据。因此,视图中的数据是依赖于原来的表中
的数据的。一旦表中的数据发生改变,显示在视图中的数据也会发生改变。 


视图的作用
        视图是在原有的表或者视图的基础上重新定义的虚拟表,这可以从原有的表上选取对用户有用的信息。那
些对用户没有用,或者用户没有权限了解的信息,都可以直接屏蔽掉。这样做既使应用简单化,也保证了
系统的安全。视图起着类似于筛选的作用。视图的作用归纳为如下几点:
1.使操作简单化
2.增加数据的安全性
3.提高表的逻辑独立性


创建表t1:
mysql> create table t1(id int(10) primary key not null auto_increment,name varchar(20),sex varchar(6) not null,age int(2),type int(2));
Query OK, 0 rows affected (0.00 sec)


插入t1中的数据:
insert into t1(name,sex,age,type) values('zhangsan','nan',26,1);
insert into t1(name,sex,age,type) values('lisi','nv',26,2);
insert into t1(name,sex,age,type) values('xiaoli','nv',25,1);
insert into t1(name,sex,age,type) values('xiaowang','nan',22,3);


查询数据表t1中的数据:
select * from t1;


查询结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
+----+----------+-----+------+------+
4 rows in set (0.00 sec)


创建表t2:
create table t2(id int(10) primary key not null auto_increment,typeName varchar(20) not null,shouru int(10));


插入表中的数据:
insert into t2(typeName,shouru) values('jishubu',6000);
insert into t2(typeName,shouru) values('yingxiaobu',8000);
insert into t2(typeName,shouru) values('shiyebu',3000);


查询插入的情况:
select * from t2;


查询结果:
+----+------------+--------+
| id | typeName   | shouru |
+----+------------+--------+
|  1 | jishubu    |   6000 |
|  2 | yingxiaobu |   8000 |
|  3 | shiyebu    |   3000 |
+----+------------+--------+
3 rows in set (0.00 sec)


视图的创建格式:
create view 视图名 as 查询语句;


其特点是:
一、视图是一张虚拟的表,表中所有的数据均来自于其他的实体表。
二、实体表中数据的修改其实是会影响到视图表中的数据的。


        假如说我们要查询每一个员工属于哪一个部门,以及他们所对应的工资数额,这时每张表中都只有一部分我们所想要查询的内容,此时我们要想将他们统统查询出来,那么我们就不得不跨表查询。


此时的查询语句如下:
select name,typeName,shouru from t1,t2 where type=t2.id;


查询结果:
+----------+------------+--------+
| name     | typeName   | shouru |
+----------+------------+--------+
| zhangsan | jishubu    |   6000 |
| lisi     | yingxiaobu |   8000 |
| xiaoli   | jishubu    |   6000 |
| xiaowang | shiyebu    |   3000 |
+----------+------------+--------+
4 rows in set (0.02 sec)


当然我们也可以通过创建一张视图表的方式将其做出来。创建视图的方式如下:
create view t1_t2 as select name,typeName,shouru from t1,t2 where type=t2.id;


我们查询视图的时候其实是很简单的,他就和查询数据表的方法其实是一样的,查询语句如下:
select * from t1_t2;


查询结果:
+----------+------------+--------+
| name     | typeName   | shouru |
+----------+------------+--------+
| zhangsan | jishubu    |   6000 |
| lisi     | yingxiaobu |   8000 |
| xiaoli   | jishubu    |   6000 |
| xiaowang | shiyebu    |   3000 |
+----------+------------+--------+
4 rows in set (0.02 sec)


        现在看来以上两种方法其实都差不多,都可以实现对于多张数据表的跨表查询功能,但是假如说我们要查询的数据表非常多,有上百张甚至是更多,而且查询的次数又很多的话,用第一种方法我们发现就会很麻烦,要死人的。反观之,第二种方法就简单多了,只需创建一次视图,使用的时候也很方便,直接拿来就可以了。


        需要注意的是,创建的视图是一张虚拟的表,该表中的数据是依赖于实体表中的数据的,当实体表中数据一作更改,视图表中的数据也就会被相应的更改了。


举例来说明吧:
        当我们更改表t1中的数据时,比如说我们将t1表中的zhangsan更改为zhangsan1时,视图t1_t2中的张三也就会相应的更改成了zhangsan1了。


更改t1表中的zhangsan为zhangsan1:
update t1 set name='zhangsan1' where id=1;


修改后的查询语句:
select * from t1;


运行的结果:
+----+-----------+-----+------+------+
| id | name      | sex | age  | type |
+----+-----------+-----+------+------+
|  1 | zhangsan1 | nan |   26 |    1 |
|  2 | lisi      | nv  |   26 |    2 |
|  3 | xiaoli    | nv  |   25 |    1 |
|  4 | xiaowang  | nan |   22 |    3 |
+----+-----------+-----+------+------+
4 rows in set (0.00 sec)


此时的视图查询语句为:
select * from t1_t2;


查询结果:
+-----------+------------+--------+
| name      | typeName   | shouru |
+-----------+------------+--------+
| zhangsan1 | jishubu    |   6000 |
| lisi      | yingxiaobu |   8000 |
| xiaoli    | jishubu    |   6000 |
| xiaowang  | shiyebu    |   3000 |
+-----------+------------+--------+
4 rows in set (0.00 sec)


由此可见,视图中的查询结果真的是被改变成了zhangsan1。


假如说我们不直接更改实体表,而是直接从视图表中更改数据,那查询的结果会是怎样的呢?


视图表中数据的更改,我们可以将之前改过的数据再次尝试着改过来。修改的语句如下;
update t1_t2 set name='zhangsan' where name='zhangsan1';


然后我们再次查询一下视图表中的数据:
select * from t1_t2;


查询结果:
+----------+------------+--------+
| name     | typeName   | shouru |
+----------+------------+--------+
| zhangsan | jishubu    |   6000 |
| lisi     | yingxiaobu |   8000 |
| xiaoli   | jishubu    |   6000 |
| xiaowang | shiyebu    |   3000 |
+----------+------------+--------+
4 rows in set (0.00 sec)


之后我们继续查询一下实体表中的数据,实体表t1中的查询语句如下:
select * from t1;


查询结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
+----+----------+-----+------+------+
4 rows in set (0.00 sec)


        由此可以看出,如果我们不直接操作实体表中的数据,而是间接操作视图表中的数据,那么在视图表中所做的更改也会自动同步到实体表中,这样也可以方便我们对于多张实体表的跨表联合操作。如果对于实体表中插入新的数据,那么视图表中也会做出相应的更改。如下所示:


在实体表t1中插入新的数据:
insert into t1(name,sex,age,type) values('wangwu','nan',27,3);


插入后的查询语句:
select * from t1;


插入后的t1表查询结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
|  5 | wangwu   | nan |   27 |    3 |
+----+----------+-----+------+------+
5 rows in set (0.00 sec)


在视图中的查询语句如下:
select * from t1_t2;


查询结果:
+----------+------------+--------+
| name     | typeName   | shouru |
+----------+------------+--------+
| zhangsan | jishubu    |   6000 |
| lisi     | yingxiaobu |   8000 |
| xiaoli   | jishubu    |   6000 |
| xiaowang | shiyebu    |   3000 |
| wangwu   | shiyebu    |   3000 |
+----------+------------+--------+
5 rows in set (0.00 sec)


由此可以看出,对于实体表中数据的插入,其结果也会自动同步到视图表中的。


假如说我们不直接在实体表中插入数据,而是间接在视图表中插入数据,那么结果又该如何?


在视图表中插入数据:
insert into t1_t2 values('laoliu','shiyebu',5000);


插入的结果:
ERROR 1394 (HY000): Can not insert into join view 'mytest.t1_t2' without fields list


        很不幸,插入失败,也就是说,在多张表组成的视图中是无法间接插入数据的,原因是视图中并没有实体表中的全部字段(即建表的字段),这一点需要格外的注意了。但是这并不妨碍对于单表所建视图的数据的插入(这并不是绝对的,如果实体表中有些字段不允许为空,并且没有默认值时,此时也是无法插入数据的)。


假如说我们想要在视图中删除数据,比如说我们想要在视图中删除wangwu,那又该怎样呢?


在视图中删除wangwu的语句如下:
delete from t1_t2 where name='wangu';


删除结果:
ERROR 1395 (HY000): Can not delete from join view 'mytest.t1_t2'


结果是删除失败。


对于多表视图操作,总结来说那就是只允许对视图进行查询和修改,删除和添加基本上都不允许。


创建单表视图。
create view t1_v as select name,sex,age from t1;


查看是否创建成功:
show tables;


查询结果:
+------------------+
| Tables_in_mytest |
+------------------+
| foot             |
| student          |
| t1               |
| t1_t2            |
| t1_v             |
| t2               |
+------------------+
6 rows in set (0.00 sec)


查询结果是创建成功!


我们可以查看一下该表的结构:
desc t1_v;


表结构如下:
+-------+-------------+------+-----+---------+-------+
| Field | Type        | Null | Key | Default | Extra |
+-------+-------------+------+-----+---------+-------+
| name  | varchar(20) | YES  |     | NULL    |       |
| sex   | varchar(6)  | NO   |     | NULL    |       |
| age   | int(2)      | YES  |     | NULL    |       |
+-------+-------------+------+-----+---------+-------+
3 rows in set (0.06 sec)


查询单表视图中的内容:
select * from t1_v;


查询结果:
+----------+-----+------+
| name     | sex | age  |
+----------+-----+------+
| zhangsan | nan |   26 |
| lisi     | nv  |   26 |
| xiaoli   | nv  |   25 |
| xiaowang | nan |   22 |
| wangwu   | nan |   27 |
+----------+-----+------+
5 rows in set (0.00 sec)


由此可以看出,对于单表视图来说,查询不成问题。


之后就是尝试修改视图。
update t1_v set name='wangwu2' where name='wangwu';


修改成功!
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0


查询视图语句:
select * from t1_v;


查询结果:
+----------+-----+------+
| name     | sex | age  |
+----------+-----+------+
| zhangsan | nan |   26 |
| lisi     | nv  |   26 |
| xiaoli   | nv  |   25 |
| xiaowang | nan |   22 |
| wangwu2  | nan |   27 |
+----------+-----+------+
5 rows in set (0.00 sec)


查询实体表中的语句:
select * from t1;


查询结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
|  5 | wangwu2  | nan |   27 |    3 |
+----+----------+-----+------+------+
5 rows in set (0.00 sec)


由此我们可以看出,修改语句也同样获得了成功。


接下来我们可以尝试着在视图中插入数据,在视图中插入数据的语句如下:
insert into t1_v values('laoliu','nan',24);


插入结果:
Query OK, 1 row affected (0.00 sec)


成功!


查询视图中的语句:
select * from t1_v;


查询结果:
+----------+-----+------+
| name     | sex | age  |
+----------+-----+------+
| zhangsan | nan |   26 |
| lisi     | nv  |   26 |
| xiaoli   | nv  |   25 |
| xiaowang | nan |   22 |
| wangwu2  | nan |   27 |
| laoliu   | nan |   24 |
+----------+-----+------+
6 rows in set (0.00 sec)


再次查询实体表中的数据:
select * from t1;


查询结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
|  5 | wangwu2  | nan |   27 |    3 |
|  6 | laoliu   | nan |   24 | NULL |
+----+----------+-----+------+------+
6 rows in set (0.00 sec)


删除操作,删除视图中的数据:
delete from t1_v where name='laoliu';


删除成功!
Query OK, 1 row affected (0.00 sec)


查看视图的语句:
select * from t1_v;


查看结果:
+----------+-----+------+
| name     | sex | age  |
+----------+-----+------+
| zhangsan | nan |   26 |
| lisi     | nv  |   26 |
| xiaoli   | nv  |   25 |
| xiaowang | nan |   22 |
| wangwu2  | nan |   27 |
+----------+-----+------+
5 rows in set (0.00 sec)


查看实体表中的语句:
select * from t1;


查看结果:
+----+----------+-----+------+------+
| id | name     | sex | age  | type |
+----+----------+-----+------+------+
|  1 | zhangsan | nan |   26 |    1 |
|  2 | lisi     | nv  |   26 |    2 |
|  3 | xiaoli   | nv  |   25 |    1 |
|  4 | xiaowang | nan |   22 |    3 |
|  5 | wangwu2  | nan |   27 |    3 |
+----+----------+-----+------+------+
5 rows in set (0.00 sec)


同样也是删除成功!


        由此可以看出,此时我们的插入是成功了,只是在实体表中,在type的位置上插入的是一个空字符。当然
,如果说我们的type字段设置的是不为空,那么不好意思,我们就只有失败了。


        总结一下那就是,单表视图和多表视图对于查询和修改都是一样的,但是对于视图中插入的数据却是有一定的限制(实体表中没有被引用的字段必须为null)。单表视图可以删除数据,同时视图和实体表中都同步了删除操作。


查询数据库中的表与视图的方法:
show tables;


查询结果:
+------------------+
| Tables_in_mytest |
+------------------+
| foot             |
| student          |
| t1               |
| t1_t2            |
| t1_v             |
| t2               |
+------------------+
6 rows in set (0.00 sec)


        我这个数据库中的表单和视图比较少,区分起来很简单,但是假如说数据库中的表太多,再加上自己所创建的视图,到时候连自己也很难分清楚哪个是表,哪个是视图,为了区别一下他们两个,我们需要一种不同的查询方式


查询视图的方式如下:
show table status where comment='view';


查询结果:
+-------+--------+---------+------------+------+----------------+-------------+-
----------------+--------------+-----------+----------------+-------------+-----
--------+------------+-----------+----------+----------------+---------+
| Name  | Engine | Version | Row_format | Rows | Avg_row_length | Data_length |
Max_data_length | Index_length | Data_free | Auto_increment | Create_time | Upda
te_time | Check_time | Collation | Checksum | Create_options | Comment |
+-------+--------+---------+------------+------+----------------+-------------+-
----------------+--------------+-----------+----------------+-------------+-----
--------+------------+-----------+----------+----------------+---------+
| t1_t2 | NULL   |    NULL | NULL       | NULL |           NULL |        NULL |
           NULL |         NULL |      NULL |           NULL | NULL        | NULL
        | NULL       | NULL      |     NULL | NULL           | VIEW    |
| t1_v  | NULL   |    NULL | NULL       | NULL |           NULL |        NULL |
           NULL |         NULL |      NULL |           NULL | NULL        | NULL
        | NULL       | NULL      |     NULL | NULL           | VIEW    |
+-------+--------+---------+------------+------+----------------+-------------+-
----------------+--------------+-----------+----------------+-------------+-----
--------+------------+-----------+----------+----------------+---------+
2 rows in set (0.00 sec)


由于表和视图在外观上看来有很大的相似形,为了比较容易区分它们,我们可以查看一下表的创建属性。


查询语句如下:
show create table t1_v;


查询结果:
+------+------------------------------------------------------------------------
--------------------------------------------------------------------------------
------------------+----------------------+----------------------+
| View | Create View


                  | character_set_client | collation_connection |
+------+------------------------------------------------------------------------
--------------------------------------------------------------------------------
------------------+----------------------+----------------------+
| t1_v | CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFI
NER VIEW `t1_v` AS select `t1`.`name` AS `name`,`t1`.`sex` AS `sex`,`t1`.`age` A
S `age` from `t1` | utf8                 | utf8_general_ci      |
+------+------------------------------------------------------------------------
--------------------------------------------------------------------------------
------------------+----------------------+----------------------+
1 row in set (0.00 sec)


从第一行的view中我们可以看出t1_v其实是一张视图。


我们可以换一下t1,其查询语句如下:
show create table t1;


查询结果如下:
+-------+-----------------------------------------------------------------------
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
----------------------------+
| Table | Create Table



                            |
+-------+-----------------------------------------------------------------------
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
----------------------------+
| t1    | CREATE TABLE `t1` (
  `id` int(10) NOT NULL AUTO_INCREMENT,
  `name` varchar(20) DEFAULT NULL,
  `sex` varchar(6) NOT NULL,
  `age` int(2) DEFAULT NULL,
  `type` int(2) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8 |
+-------+-----------------------------------------------------------------------
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
----------------------------+
1 row in set (0.00 sec)


这个则在表头表明它是一张table表单。


查看表头的目的是区分视图和实体表,因为他们在修改与删除操作上都是不同的。


视图的修改方式:
alter view 视图名 as 查询语句


视图的修改语句:
alter view t1_v as select id,name,sex,age from t1;


查询修改后的t1_v的语句如下:
select * from t1_v;


修改后的结果:
+----+----------+-----+------+
| id | name     | sex | age  |
+----+----------+-----+------+
|  1 | zhangsan | nan |   26 |
|  2 | lisi     | nv  |   26 |
|  3 | xiaoli   | nv  |   25 |
|  4 | xiaowang | nan |   22 |
|  5 | wangwu2  | nan |   27 |
+----+----------+-----+------+
5 rows in set (0.02 sec)


修改成功!


视图的删除方式:
drop view 视图名


视图的删除语句:
drop view t1_v;


查询删除后的语句:
show tables;


删除后的结果:
+------------------+
| Tables_in_mytest |
+------------------+
| foot             |
| student          |
| t1               |
| t1_t2            |
| t2               |
+------------------+
5 rows in set (0.00 sec)


由此可以看出,视图删除成功!

转载:(10条消息) mysql视图的应用总结_思考、总结、专注-CSDN博客

(0)

相关推荐