C++学习路线(数据库部分)六

视图操作

小学的时候,每年都会举办一次抽考活动,意思是从每一个班级里面筛选出几个优秀的同学去参加考试。这时候很多班级筛选出来的这些同学就可以临时组成一个班级,如果我们把每一个班级当做是一张真实的表,这个临时的班级在数据库里可以当做一个视图,也就是说,这个临时的班级其实不是真实存在的,当考试过后,这些学生还是各回各家各找各妈

视图是从一个或多个表中导出来的表,是一种虚拟存在的表。视图就像一个窗口,通过这个窗口可以看到系统专门提供的数据,这样用户可以不看整个数据库表中的数据,而只关心对自己有用的数据。视图可以使用户的操作更方便,而且可以保障数据库系统的安全性。

为什么要使用视图

通过前面章节的知识可以发现,数据库中关于数据的查询有时非常复杂,例如表连接、子查询等,这种查询会让程序员感到非常痛苦,因为它的逻辑太复杂、编写语句比较多,当这种查询需要重复使用时,很难每次都编写正确,从而降低了数据库的实用性。

在具体操作表之前,有时候要求只能操作部分字段,而不是全部字段。例如,在学校里,学生的智商测试结果一般都是保密的,如果因为一时疏忽向查询中多写了关于“智商”的字段,则会让学生的智商显示给所有能够查看该查询结果的人,这时就需要限制使用者操作的字段。

为了提高复杂的SQL语句的复用性和表的操作的安全性,MySQL数据库管理系统提供了视图特性。所谓视图,本质上是一种虚拟表,其内容与真实的表相似,包含一系列带有名称的列和行数据。但是,视图并不在数据库中以存储数据值的形式存在,行和列数据来自定义视图的查询所引用的基本表,并且在具体引用视图时动态生成

视图使程序员只关心感兴趣的某些特定数据和他们所负责的特定任务。这样程序员只能看到视图中所定义的数据,而不是视图所引用表中的数据,从而提高数据库中数据的安全性。

创建视图

虽然视图可以被看成是一种虚拟表,但是其物理上是不存在的,即MySQL并没有专门的位置为视图存储数据。根据视图的概念可以发现其数据来源于查询语句,因此创建视图的基本语法为:

CREATE[OR REPLACE] VIEW viewname[columnlist]   

    AS SELECT statement                               

   其中,CREATE表示创建新的视图;REPLACE表示替换已经创建的视图;viewname为视图的名称;columnlist为属性列;SELECT statement表示SELECT语句

注意! 创建视图需要登陆用户有相应的权限,查看权限方法:

mysql>  use school;   #选择数据库school                                                                                                                    

mysql>  select user, Select_priv, Create_view_priv FROM mysql.user;#查询数据库用户创建和选择视图权限

在单表上创建视图

mysql>  use school;   #选择数据库school

mysql>  alter table student add privacy varchar(64);# 增加私隐列                                                                                      

mysql>  ;#查询数据库用户创建和选择视图权限    

mysql>  CREATE VIEW view_student AS select id, class_id, name from student ;#为学生表创建视图

mysql>  desc view_student;#查看视图

mysql>  select * from view_student;   #根据视图进行查询  

【示例9.1】

在多表上创建视图

    CREATE[OR REPLACE] VIEW viewname[columnlist]   

    AS SELECT statement                               

   其中,CREATE表示创建新的视图;REPLACE表示替换已经创建的视图;viewname为视图的名称;columnlist为属性列;SELECT statement表示SELECT语句;与单表上创建视图不同的是,SELECT子句是涉及到多表的联合查询语句

mysql>  use school;   #选择数据库school

mysql>  alter table student add privacy varchar(64);# 增加私隐列                                                                                      

mysql>  ;#查询数据库用户创建和选择视图权限    

mysql>  CREATE VIEW view_student_class AS select student.id, student.name, class.name,  class.teacher from class inner join student  on class.id = student.class_id;#为学生表创建视图

mysql>  desc view_student_class;#查看视图

mysql>  select * from view_student_class;   #根据视图进行查询  

【示例9.2】

使用内连接,外连接来进行view视图的创建

查看视图

创建完视图后,像表一样,我们经常需要查看视图信息。在MySQL中,有许多可以实现查看视图的语句,如DESCRIBE、SHOW TABLES、SHOW CREATE VIEW。如果要使用这些语句,首先要确保拥有SHOW VIEW的权限。本节将详细讲解查看视图的方法。

  1. 使用DESCRIBE | DESC语句查看视图基本信息

前面我们已经详细讲解过使用DESCRIBE语句来查看表的基本定义。因为视图也是一张表,只是这张表比较特殊,是一张虚拟的表,所以同样可以使用DESCRIBE语句来查看视图的基本定义。DESCRIBE语句查看视图的语法如下:

    DESCRIBE | DESC viewname;

在上述语句中,参数viewname表示所要查看设计信息的视图名称。

  1. 使用SHOW TABLES语句查看视图基本信息

从MySQL 5.1版本开始,执行SHOW TABLES语句时不仅会显示表的名字,同时也会显示视图的名字。

下面演示通过SHOW TABLES语句查看数据库school中的视图和表的功能,具体SQL语句如下,执行结果如下图所示。

    SHOW TABLES;

  1. 使用 show create view/table 语句查看视图创建信息

    SHOW CREATE TABLEVIEW   viewname;   

更新视图数据

更新视图是指通过视图来插入(INSERT)、更新(UPDATE)和删除(DELETE)表中的数据。因为视图实质是一个虚拟表,其中没有数据,通过视图更新时都是转换到基本表更新。更新视图时,只能更新权限范围内的数据,超出范围就不能更新了。

mysql>  use school;   #选择数据库school

mysql>  alter table student add privacy varchar(64);# 增加私隐列                                                                                      

mysql>  ;#查询数据库用户创建和选择视图权限    

mysql>  CREATE VIEW view_student AS select id, class_id, name from student ;#为学生表创建视图

mysql>  desc view_student;#查看视图

mysql>  select * from view_student;   #根据视图进行查询  

mysql>  update view_student set name='小花花' where name='小花'; #通过视图更新小花为小花花

不能更新的情况:

  1. 视图中包含SUM()、COUNT()、MAX()和MIN()等函数
  2. 图中包含UNION、UNION ALL、DISTINCT、GROUP BY和HAVING等关键字
  3. 视图对应的表存在没有默认值的列,而且该列没有包含在视图里
  4. 包含子查询的视图
  5. 其他特殊情况

修改视图

修改视图是指修改数据库中存在的视图,当基本表的某些字段发生变化的时候,可以通过修改视图来保持与基本表的一致性。ALTER语句来修改视图。

使用ALTER语句修改视图

ALTER VIEW viewname[columnlist]   

AS SELECT statement                               

这个语法中的所有关键字和参数除了alter 外,其他都和创建视图是一样的,因此不再赘述

【示例9.3】对于示例9.2中创建的视图view_student_class,使用一段时间后需要将表示编号的字段id加进去。步骤如下:

mysql>  use school;   #选择数据库school

mysql>  alter table student add privacy varchar(64);# 增加私隐列                                                                                      

mysql>  ;#查询数据库用户创建和选择视图权限    

mysql>  ALTER VIEW view_student_class AS select student.id, student.name, class.name, class.id as class_id, class.teacher from class inner join student  on class.id = student.class_id;#为学生班级表视图增加 class_id 字段

mysql>  desc view_student_class;#查看视图

mysql>  select * from view_student_class;   #根据视图进行查询  

【示例9.3】

删除视图

删除视图是指删除数据库中已存在的视图。删除视图时,只能删除视图的定义,不会删除数据。

在MySQL中,可使用DROP VIEW语句来删除视图,但是用户必须拥有DROP权限。删除视图的语法如下:

    DROP VIEW viewname [,viewnamen];

在上述语句中,参数viewname表示所要删除视图的名称,可同时指定删除多个视图。

mysql>  use school;   #选择数据库school      

mysql>  ;#查询数据库用户创建和选择视图权限    

mysql>  CREATE VIEW view_student_class AS select student.id, student.name, class.name, class.id as class_id, class.teacher from class inner join student  on class.id = student.class_id;#为学生表创建视图

mysql>  drop view view_student_class;#删除视图

 

触发器

触发器的概念精讲

在实际开发中往往会碰到这样的情况:

当我们对一个表进行数据操作时,需要同步对其它的表执行相应的操作,正常情况下,如果我们使用sql语句进行更新,将需要执行多条操作语句

比如,在某些棋牌游戏中,当玩家充值金币后,玩家表数据库中金币增加的同时,玩家所属的代理会得到相应的提成并计入代理的收益中,即代理数据库表提成记录字段也要同步更新

而以上的场景,我们可以轻松使用触发器来实现!

触发器(TRIGGER)是由事件来触发某个操作。这些事件包括INSERT语句、UPDATE语句和DELETE语句。当数据库系统执行这些事件时,就会激活触发器执行相应的操作。MySQL从5.0.2版本开始支持触发器。

通过本章的学习,我们将了解触发器的含义和作用、如何创建触发器、查看触发器和删除触发器的方法。同时,可以了解各种事件的触发器的执行情况。
 

创建触发器 

在MySQL中创建触发器通过SQL语句CREATE TRIGGER来实现,其语法形式如下:

       CREATE trigger trigger_name BEFORE|AFTER trigger_EVENT     

       ON TABLE_NAME FOR EACH ROW trigger_STMT               

在上述语句中,参数trigger_name表示要创建的触发器名;

参数BEFORE和AFTER指定了触发器执行的时间,前者在触发器事件之前执行触发器语句,后者在触发器事件之后执行触发器语句;

参数trigger_EVENT表示触发事件,即触发器执行条件,包含DELETEINSERTUPDATE语句;参数TABLE_NAME表示触发事件的操作表名;参数FOR EACH ROW表示任何一条记录上的操作满足触发事件都会触发该触发器

参数trigger_STMT表示激活触发器后被执行的语句。执行语句中如果要引用更新记录中的字段,对于INSERT语句,只有NEW是合法的,表示当前已插入的记录;对于DELETE语句,只有OLD才合法,表示当前删除的记录;而UPDATE语句可以和NEW(更新后)以及OLD(更新前)同时使用。

注意:不能创建具有相同名字的触发器。另外,对于具有相同触发程序动作时间和事件的给定表,不能有两个触发器。因此,对于有经验的用户,在创建触发器之前,需要查看MySQL中是否已经存在该标识符的触发器和触发器的相关事件。

【示例10-1】执行SQL语句CREATE TRIGGER,在数据库school中存在两个表对象:学员表student和班级表 class,创建触发器实现向学员表中插入记录时,就会在插入之后更新班级表中的人数,当我们删除某条学员的记录时,就会在删除后更新班级表中的人数,具体步骤如下:

mysql>  use school;   #选择数据库school                                           

mysql>  CREATE TABLE class (                                                        

  `id` int NOT NULL AUTO_INCREMENT,                                              

  `name` varchar(128) DEFAULT NULL,                                               

  `teacher` varchar(64) DEFAULT NULL,  

  `count`  int DEFAULT 0,                                           

  UNIQUE KEY `id` (`id`)                                                              

);  #创建班级表 class                                                                 

mysql> insert into class values(101, '萌新一班', 'Martin', 0),(102, '萌新二班', 'Rock', 0),(103, '萌新三班', 'Janny', 0);  #创建成绩表 grade                                                 

mysql>  CREATE TABLE `student` (                                                  

  `id` int NOT NULL AUTO_INCREMENT UNIQUE,                                                            

  `name` varchar(64) DEFAULT NULL,                                                

  `class_id` int DEFAULT NULL,                                                      

  `sex` enum('F','M') DEFAULT NULL                                                  

);

mysql> create trigger tri_insert_student after insert on student for each row update class set count=count+1 where class.id = NEW.class_id;   #创建触发器,新增学员班级人数增1

                                                                               

mysql> insert into student values(1,'小花',101,'M'),(2,'小红',102, 'F'),(3,'小军',102,'F'),(4,'小白',101,'F');  #插入多条记录   

mysql> select count from class  ;  #查询class 表人数  

mysql> create trigger tri_delete_student after delete on student for each row update class set count=count-1 where id = OLD.class_id; #创建触发器,删除学员班级人数减1

触发器包含多条执行语句

       CREATE   trigger trigger_name BEFORE|AFTER trigger_EVENT     

       ON TABLE_NAME FOR EACH ROW                                 

           BEGIN                                                        

            trigger_STMT                                                 

           END                                                           

在上述语句中,比“只有一条执行语句的触发器”语法多出来两个关键字BEGIN和END,在这两个关键字之间是所要执行的多个执行语句的内容,执行语句之间用分号隔开。

在MySQL中,一般情况下用“;”符号作为语句的结束符号,可是在创建触发器时,需要用到“;”符号作为执行语句的结束符号。为了解决该问题,可以使用关键字DELIMITER语句。例如,“DELIMITER $$”可以将结束符号设置成“$$”

mysql>  use school;   #选择数据库school         

mysql>  create table grade(id int UNIQUE AUTO_INCREMENT,  math tinyint unsigned, chinese tinyint unsigned, english tinyint unsigned);       #创建成绩表 grade   

mysql> insert into grade values(1, 80, 87, 91),(2, 72, 64, 89),(3, 54, 69, 87),(4, 78, 79, 89);  #插入多条记录                                      

mysql> DELIMITER $$                                                                                

mysql> create trigger tri_delete_student after delete on student for each row

        BEGIN                                    

         Delete from grade where id = OLD.id;  #删除成绩表中的记录                                                        

         update class set count=count-1 where id = OLD.class_id; #更新班级表中的记录   

         END;                                    

         $$                                       

         DELIMITER ;                             

查看触发器

SHOW TRIGGERS语句查看触发器

那么如何查看MySQL软件中已经存在的触发器呢?在MySQL软件中查看已经存在的触发器,通过SQL语句SHOW TRIGGERS来实现,其语法形式如下,执行上面的SQL语句,执行结果如图9-10所示。

  SHOW TRIGGERS ;  

通过图9-10的执行结果可以发现,执行完“SHOW TRIGGERS”语句后会显示一个列表,在该列表中会显示出所有触发器的信息。其中,参数Trigger表示触发器的名称;参数Event表示触发器的激发事件;参数Table表示触发器对象触发事件所操作的表;参数Statement表示触发器激活时所执行的语句;参数Timing表示触发器所执行的时间。

查看系统表triggers实现查看触发器

在MySQL中,在系统数据库information_schema中存在一个存储所有触发器信息的系统表triggers,因此查询该表格的记录也可以实现查看触发器功能。系统表triggers的表结构

mysql>             #选择数据库information_schema                  

mysql>  select * from triggers;                                                        

mysql>  select * from triggers where trigger_name=tri_delete_student; #查询系统表triggers中的触发器                                                                   

触发器的删除

在MySQL软件中,可以通过DROP TRIGGER语句通过工具删除触发器。

在MySQL中,删除触发器可以通过SQL语句DROP TRIGGER来实现,其语法形式如下:

    DROP TRIGGER trigger_name; 

在上述语句中,参数trigger_name表示所要删除的触发器名称。

猜你喜欢

转载自blog.csdn.net/weixin_45169583/article/details/143362778