MySQL(八)----- 视图

        注意,MySQL从5.0.1版本开始提供视图功能,使用时注意版本;另外,如果从不支持视图的旧版本升级到提供视图的新版本后,要想使用视图还需要升级授权表,使之包含与视图有关的权限。

一、什么是视图

       视图是一种虚拟存在的表,它存储的是查询语句,显示出来的是查询的结果;更直白的说就是当我们需要从表中查询一些信息时需要编写相关SQL语句,将这些SQL语句存储为视图,那么我们调用这些视图的时候就相当于执行了SQL语句,从而可以得到想要的结果。

我先举个例子大家感受一下:

1. 创建表、插入数据并执行查询操作
mysql> create table temp (qty int,price int);
Query OK, 0 rows affected (0.01 sec)

mysql> insert into temp values(3,50),(5,60);
Query OK, 2 rows affected (0.01 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> select * from temp;
+------+-------+
| qty  | price |
+------+-------+
|    3 |    50 |
|    5 |    60 |
+------+-------+
2 rows in set (0.00 sec)

2. 创建视图并执行查询操作
mysql> create view v as select qty,price,qty*price as value from temp;
Query OK, 0 rows affected (0.01 sec)

mysql> select * from v;
+------+-------+-------+
| qty  | price | value |
+------+-------+-------+
|    3 |    50 |   150 |
|    5 |    60 |   300 |
+------+-------+-------+
2 rows in set (0.00 sec)

二、视图有什么作用

       基于上面的例子我们可以大概感受到视图大概是什么,但是具体的作用可能还是不太了解,甚至觉得有没有视图用不用它都无所谓,不要急,介绍完它的作用后你就会觉得它很有必要了。

作用一:简单;视图就像是一个封装了很多功能的函数,我们把一系列复杂的查询语句存储为一个视图,这样在需要频繁使用这些语句时不必反复编写,直接使用视图代替即可。例如:

如果要频繁获取表user的name和表goods的name。就应该使用以下sql语句:
select a.name as username, b.name as goodsname from user as a, goods as b, ug as c where a.id=c.userid and c.goodsid=b.id;
但有了视图就不一样了,创建视图other。示例
create view other as select a.name as username, b.name as goodsname from user as a, goods as b, ug as c where a.id=c.userid and c.goodsid=b.id;
创建好视图后,就可以这样获取user的name和goods的name:
 select * from other;

作用二:对数据库进行重构但仍不会影响程序运行;例如:

假如因为某种需求,需要将user表拆成表usera和表userb,该两张表的结构如下:
        测试表:usera有id,name,age字段
        测试表:userb有id,name,sex字段
这时如果程序端一直使用的sql语句是:select * from user;那就会报错提示该表不存在,这时要么去更改程序的查询语句要么就创建视图。显然,创建视图更简单,成本更低。
以下sql语句创建视图:
create view user as select a.name,a.age,b.sex from usera as a, userb as b where a.name=b.name;
以上假设name都是唯一的。
这时程序端端使用的sql语句:select * from user;就不会报错。这就实现了更改数据库结构,而不用更改脚本程序的功能。

作用三:安全;创建好的视图已经规定好了你能访问到的信息,这样可以对用户信息查询的权限进行分离,另外,虽然对数据表有权限管理,但它并不能限制到具体的行和列,而视图可以。

作用四:数据独立且清晰;视图的结构一旦确立就可以屏蔽原表的结构、数据等的变化对用户造成的影响,想要什么样的数据就创建什么样的视图,非常清晰直观。

        这些弄清楚后,下面就要介绍视图的增删改查具体操作。

三、视图操作

    3.1 创建或修改视图

      创建视图需要有CREATE VIEW权限,并且对于查询涉及到的列要有SELECT权限;如果用CREATE OR REPLACE或ALTER修改则需要视图的DROP权限。创建视图的语法为:

CREATE [ OR REPLACE ] [ ALGORITHM = { UNDEFINED | MERGE | TEMPTABLE } ] VIEW view_name [ (column_list) ] AS select_statement [ WITH [ CASCADED | LOCAL ] CHECK OPTION ]

  • 使用了OR REPLACE可以修改视图,或者直接就CREATE OR REPLACE 替换为ALTER 即变成修改语法;
  • UNDEFINED:未定义指定算法;MERGE:更新视图表数据的同时会更新真实表的数据(默认);TEMPTABLE:只能查询不能更新;
  • CASCADED(默认):必须满足所有针对该视图的所有视图的条件才可以更新;LOCAL:只需满足本视图的条件就可以更新;
  • WITH CHECK OPTION:需要满足相关的检查条件才能进行更新;
这样创建的视图不能修改:
create view v as select qty,price,qty*price as value from temp;

这样创建的可以被修改:
create or replace view v as select qty,price,qty*price as value from temp;

注意:视图创建时FORM关键字后面不能包含子查询,如果必须要的话可以先将子查询的内容创建为一个视图,再对该视图创建视图即可。

       视图是否可更新主要需满足以下四大条件:

      1. 使用 OR REPLACE 创建;

      2. ALGORITHM参数不是使用的TEMPTABLE;

      3. 检查条件关键字 CASCADE 或 LOCAL 需满足相关规定;

      4. 创建视图时查询语句数据的SQL语句不能是以下类型:

  • 包含关键字:聚合函数(SUM,MIN,MAX,COUNT等)、DISTINCT、GROUP BY、HAVING、UNION、UNION ALL;
  • 常量视图;
  • SELECT中包含子查询;
  • JOIN;
  • FROM一个不能跟新的视图;
  • WHERE字句的子查询引用了FORM字句中的表;

举个例子:

--常量视图
create or replace view pi as select 3.1415926 as pi;

--select中包含子查询
create or replace view city_view as select (select city from city where city_id=1);

         对上面的第三点条件再做一点解释:比如我创建一个视图view1,它里面定义了查询条件num < 10;在view1的基础上使用CASCADED创建视图view2,它里面定义了查询条件num > 5;如果此时对view2进行更新设置num = 10,那么他能满足view2的条件不能满足view1的条件,因此更新失败;但view2如果使用的是关键字LOCAL,那么此时更新会成功,因为它只需要满足view2本身的条件即可。

      3.2 删除视图

        用户可以一次删除一个或多个视图,前提是必须要有该视图的DROP权限。相关语法如下:

        DROP VIEW [ IF EXISTS ] view_name [, view_name] .... [ RESTRICT | CASCAD ]

 

      3.3 查看视图

查看当前库下所有视图:

mysql> show full tables where table_type like 'view';
+-----------------+------------+
| Tables_in_test1 | Table_type |
+-----------------+------------+
| v               | VIEW       |
+-----------------+------------+
1 row in set (0.01 sec)

show tables命令不仅会显示所有表的名字,也会显示所有视图的名字:

mysql> show tables;
+-----------------+
| Tables_in_test1 |
+-----------------+
| ai              |
| autoincre_demo  |
| autoincre_demo2 |
| city            |
| country         |
| dept            |
| emp1            |
| myisam_char     |
| payment_2006    |
| payment_2007    |
| payment_all     |
| salary          |
| t               |
| tab_memory      |
| te              |
| temp            |
| v               |
| vc              |
+-----------------+
18 rows in set (0.00 sec)

show table status [from db_name ] [like 'pattern' ]命令不仅可以显示表的信息也可显示视图的信息:

mysql> show table status like 'v' \G;
*************************** 1. row ***************************
           Name: v
         Engine: NULL
        Version: NULL
     Row_format: NULL
           Rows: NULL
 Avg_row_length: NULL
    Data_length: NULL
Max_data_length: NULL
   Index_length: NULL
      Data_free: NULL
 Auto_increment: NULL
    Create_time: NULL
    Update_time: NULL
     Check_time: NULL
      Collation: NULL
       Checksum: NULL
 Create_options: NULL
        Comment: VIEW
1 row in set (0.00 sec)

如果要看某个视图的定义可以使用SHOW CREATE VIEW命令:

mysql> show create view v \G
*************************** 1. row ***************************
                View: v
         Create View: CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `v` AS select `temp`.`qty` AS `qty`,`temp`.`price` AS `price`,(`temp`.`qty` * `temp`.`price`) AS `value` from `temp`
character_set_client: gbk
collation_connection: gbk_chinese_ci
1 row in set (0.00 sec)

也可以通过查看系统表information_schema.views来查看视图相关信息:

mysql> select * from views where table_name  = 'v' \G
*************************** 1. row ***************************
       TABLE_CATALOG: def
        TABLE_SCHEMA: test1
          TABLE_NAME: v
     VIEW_DEFINITION: select `test1`.`temp`.`qty` AS `qty`,`test1`.`temp`.`price` AS `price`,(`test1`.`temp`.`qty` * `test1`.`temp`.`price`) AS `value` from `test1`.`temp`
        CHECK_OPTION: NONE
        IS_UPDATABLE: YES
             DEFINER: root@localhost
       SECURITY_TYPE: DEFINER
CHARACTER_SET_CLIENT: gbk
COLLATION_CONNECTION: gbk_chinese_ci
1 row in set (0.01 sec)

 

 

 

你可能感兴趣的:(MySQL)