MySQL之视图、触发器、事务、索引及其他知识补充

一、视图

视图是将SQL语句的查询结果当做虚拟表实体化保存起来,以后可以反复使用

create view teacher2course as
select * from teacher inner join course on teacher.tid = course.teacher_id;

drop view teacher2course;
-- 视图使用频率不高

二、触发器(trigger)

触发:当满足特征条件时自动执行。

[En]

Trigger: automatically executed when the characteristic conditions are met.

在MySQL只有三种情况下可以触发:

  • 针对表的增
  • 增前
  • 增后
  • 针对表的改
  • 改前
  • 改后
  • 针对表的删
  • 删前
  • 删后

语法结构

create trigger 触发器的名字 before/after insert/update/delete on 表名 for each row
begin
    sql语句
end

建议使用以下方式命名触发器名称:

[En]

Trigger names are recommended to be named in the following ways:

tri_after_insert_t1、tri_before_delete_t1

如何临时修改SQL语句的结束符:

delimiter $$

在MySQL中NEW特指数据对象可以通过点的方式获取字段对应的数据

id    name  pwd  hobby
1     jason  123  read
NEW.name  >>>  jason

案例:

CREATE TABLE cmd (
    id INT PRIMARY KEY auto_increment,
    USER CHAR (32),
    priv CHAR (10),
    cmd CHAR (64),
    sub_time datetime,   -- 提交时间
    success enum ('yes', 'no')   -- 0代表执行失败
);

CREATE TABLE errlog (
    id INT PRIMARY KEY auto_increment,
    err_cmd CHAR (64),
    err_time datetime
);

delimiter $$  -- 将mysql默认的结束符由;换成$$
create trigger tri_after_insert_cmd after insert on cmd for each row
begin
    if NEW.success = 'no' then  -- 新记录都会被MySQL封装成NEW对象
        insert into errlog(err_cmd,err_time) values(NEW.cmd,NEW.sub_time);
    end if;
end $$
delimiter ;  -- 结束之后记得再改回来,不然后面结束符就都是$$了

-- 往表cmd中插入记录,触发触发器,根据IF的条件决定是否插入错误日志
INSERT INTO cmd (
    USER,
    priv,
    cmd,
    sub_time,
    success
)
VALUES
    ('tony','0755','ls -l /etc',NOW(),'yes'),
    ('tony','0755','cat /etc/passwd',NOW(),'no'),
    ('tony','0755','useradd xxx',NOW(),'no'),
    ('tony','0755','ps aux',NOW(),'yes');

-- 查询errlog表记录
select * from errlog;
-- 查看触发器
show triggers;
-- 删除触发器
drop trigger tri_after_insert_cmd;

三、事务(重要)

  • A:原子性 每个事务都是不可分割的最小单位(同一个事务内的多个操作要么同时成功要么同时失败)
  • C:一致性 事务必须是使数据库从一个一致性状态变到另一个一致性状态。一致性与原子性是密切相关的。
  • I:隔离性 事务与事务之间彼此不干扰,即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能互相干扰。
  • D:持久性 持久性也称永久性(permanence),指一个事务一旦提交,它对数据库中数据的改变就应该是永久性的;接下来的其他操作或故障不应该对其有任何影响
start transcation;  -- 开启事务
    诸多SQL操作
rollback;  -- 回滚到操作之前的状态
commit;  -- 确认事务操作,之后不能回滚

示例如下 :

create table user(
id int primary key auto_increment,
name char(32),
balance int
);

insert into user(name,balance)
values
('jason',1000),
('egon',1000),
('tank',1000);

-- 修改数据之前先开启事务操作
start transaction;

-- 修改操作
update user set balance=900 where name='jason';   -- 买支付100元
update user set balance=1010 where name='egon';   -- 中介拿走10元
update user set balance=1090 where name='tank';   -- 卖家拿到90元

-- 回滚到上一个状态
rollback;

-- 开启事务之后,只要没有执行commit操作,数据其实都没有真正刷新到硬盘
commit;
"""开启事务检测操作是否完整,不完整主动回滚到上一个状态,如果完整就应该执行commit操作"""

站在python代码的角度,应该实现的伪代码逻辑:

try:
    update user set balance=900 where name='jason'; #买支付100元
    update user set balance=1010 where name='egon'; #中介拿走10元
    update user set balance=1090 where name='tank'; #卖家拿到90元
except 异常:
    rollback;
else:
    commit;

四、存储过程

类似于python中的自定义函数。

delimiter $$
create procedure p1()
begin
    select * from user;
end $$
delimiter ;
-- 调用
call p1();
delimiter $$
create procedure p2(
    in m int,  -- in表示这个参数必须只能是传入不能被返回出去
    in n int,
    out res int  -- out表示这个参数可以被返回出去,还有一个inout表示即可以传入也可以被返回出去
)
begin
    select * from user where id > m and id < n;
    set res=0;  -- 用来标志存储过程是否执行
end $$
delimiter ;
import pymysql

conn = pymysql.connect(
    host='127.0.0.1',
    port=3306,
    user='root',
    passwd='456852',
    db='db6',
    charset='utf8',
    autocommit=True
)
cursor = conn.cursor(cursor=pymysql.cursors.DictCursor)
cursor.callproc('p2',(1,3,10))
@_p1_0=1,@_p1_1=3,@_p1_2=10;
print(cursor.fetchall())

五、函数

相当于Python中的内置函数。

ps:可以通过 help &#x51FD;&#x6570;&#x540D; 查看帮助信息!

  • 移除指定字符
Trim()、LTrim()、RTrim()
  • 大小写转换
Lower()、Upper()
  • 从左右开始获取指定数量的字符
    [En]

    get the specified number of characters starting from the left and right*

Left()、Right()
  • 返回读音相似值
Soundex()"""eg:客户表中有一个顾客登记的用户名为J.Lee但如果这是输入错误真名其实叫J.Lie,可以使用soundex匹配发音类似的where Soundex(name)=Soundex('J.Lie')"""
  • 日期格式
date_format()'''在MySQL中表示时间格式尽量采用2022-11-11形式'''CREATE TABLE blog (    id INT PRIMARY KEY auto_increment,    NAME CHAR (32),    sub_time datetime);INSERT INTO blog (NAME, sub_time)VALUES    ('第1篇','2015-03-01 11:31:21'),    ('第2篇','2015-03-11 16:31:21'),    ('第3篇','2016-07-01 10:21:31'),    ('第4篇','2016-07-22 09:23:21'),    ('第5篇','2016-07-23 10:11:11'),    ('第6篇','2016-07-25 11:21:31'),    ('第7篇','2017-03-01 15:33:21'),    ('第8篇','2017-03-01 17:32:21'),    ('第9篇','2017-03-01 18:31:21');select date_format(sub_time,'%Y-%m'),count(id) from blog group by date_format(sub_time,'%Y-%m');-- 更多日期处理相关函数where Date(sub_time) = '2015-03-01'where Year(sub_time)=2016 AND Month(sub_time)=07;    adddate 增加一个日期    addtime 增加一个时间    datediff计算两个日期差值  ...

六、流程控制

-- if判断if i = 1 THEN       SELECT 1;ELSEIF i = 2 THEN       SELECT 2;ELSE       SELECT 7;END IF;-- while循环SET num = 0 ;WHILE num < 10 DO    SELECT        num ;    SET num = num + 1 ;END WHILE ;

七、索引与慢查询优化(重要)

索引可以简单地理解为帮助您加快数据查询速度的工具,或者可以将索引比作图书目录。

[En]

An index can be simply understood as a tool to help you speed up your data query, or an index can be compared to a book catalog.

索引涉及以下几种数据结构:

[En]

Several data structures are involved in indexing:

  • B树
  • B+树(叶子节点添加了指针)
  • B*树(基于B+树在枝节点也添加了指针)

将某个字段添加成索引就相当于依据该字段建立了一颗B+树,从而加快查询速度。

如果没有将索引添加到字段,则基于该字段查询数据的速度将非常慢(遍历查找)。

[En]

If an index is not added to a field, querying data based on that field will be very slow (traversal lookup).

注意外键 foreign key 不是索引,它仅仅是用来创建表与表之间关系的。

-- 创建唯一索引需要提前排查是否有重复数据
select count(字段) from 表名;
select count(distinct(字段)) from 表名;

-- 查看当前表内部索引值
show index from 表名;

-- 创建主键索引
alter table t1 add primary key pri_id(id);

-- 创建唯一索引
alter table t1 add unique key uni_age(age);

-- 创建普通索引
alter table t1 add index idx_name(name);

-- 前缀索引(属于普通索引)
"""
避免对大列建索引,如果有就使用前缀索引
eg:博客内容 百度搜索内容等
"""
alter table t1 add index idx_name(name(4));

-- 联合索引(属于普通索引,遵循最左匹配原则)
alter table t1 add index idx_all(name,age,sex);

-- 删除索引
alter table t1 drop index 索引名(idx_name、idx_all...);
  • 全表扫描:不走索引,遍历表查找数据,效率极低,生产环境下尽量不要书写类似SQL;
  • 索引扫描:走索引,加快数据查询,建议书写该类型SQL;

explain 就是帮助我们查看SQL语句属于那种扫描。

常见的索引扫描类型:

从上到下,性能从最差到最好,生产环境下认为至少要达到 range级别。

不要遵循索引(熟悉四项或更多项):

[En]

Do not follow the index (familiar with four or more items):

  • 没有查询条件,或查询条件未编制索引
    [En]

    there are no query conditions, or the query conditions are not indexed*

  • 查询结果为原表中的大部分数据(超过25%)
    [En]

    the query result is most of the data in the original table (more than 25%)*

  • 指数本身无效,统计数据不实
    [En]

    the index itself is invalid and the statistics are not true*

  • 查询条件使用函数对索引列进行操作,包括(+、-、等)
    [En]

    query conditions use functions to operate on or on index columns, including (+, -, , etc.)*
  • 隐式转换导致索引失败; eg:字段是字符串类型,查询使用整形
  • <> &#xFF0C;not in不走索引 单独的>,
  • like "%_" 百分号在最前面不走;
  • 单独引用联邦索引中不在第一位置的索引列(不遵循最左侧匹配原则)
    [En]

    refer separately to index columns that are not in the first position in the federated index (do not follow the leftmost matching principle)*

索引的创建会加快数据的查询速度,但是一定程度会拖慢数据的插入和删除速度。

八、隔离级别

在SQL标准中定义了四种隔离级别,每一种级别都规定了一个事务中所做的修改。

InnoDB支持所有隔离级别: set transaction isolation level

  • read uncommitted(未提交读) 事务中的修改即使没有提交,对其他事务也都是可见的,事务可以读取未提交的数据,这一现象也称之为 "&#x810F;&#x8BFB;"
  • read committed(提交读)—大多数数据库系统默认的隔离级别 一个事务从开始直到提交之前所作的任何修改对其他事务都是不可见的,这种级别也叫做 "&#x4E0D;&#x53EF;&#x91CD;&#x590D;&#x8BFB;"
  • repeatable read(可重复读)—MySQL默认隔离级别 能够解决 "&#x810F;&#x8BFB;"问题,但是无法解决 "&#x5E7B;&#x8BFB;" 所谓 "&#x5E7B;&#x8BFB;"指的是当某个事务在读取某个范围内的记录时另外一个事务又在该范围内插入了新的记录,当之前的事务再次读取该范围的记录会产生幻行,InnoDB和XtraDB通过多版本并发控制(MVCC)及间隙锁策略解决了该问题。
  • serializable(可串行读) 强制事务串行执行,很少使用该级别

Original: https://www.cnblogs.com/JZjuechen/p/15929882.html
Author: JZEason
Title: MySQL之视图、触发器、事务、索引及其他知识补充

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/507802/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • DM8数据库备份还原的原理及应用

    ▷▶介质管理层(MML) DM8根据自身实现自定义了一套API接口SBT来负责备份数据在存储介质上的实际读写操作。 整个存储介质上的读写操作,由MML调用SBT完成。 目前 DM8…

    数据库 2023年6月11日
    0110
  • 浅谈事务隔离级别、MVCC及相关特性

    文采不是很好,应该有一些表达不清楚的地方,请改正。 [En] Literary talent is not very good, there should be some uncl…

    数据库 2023年5月24日
    0113
  • 23种设计模式之访问者模式(Visitor Pattern)

    文章目录 概述 访问者模式的优缺点 访问者模式的使用场景 访问者模式的结构和实现 * 模式结构 模式实现 总结 概述 访问者模式把数据结构和作用于结构上的操作解耦合,使得操作集合可…

    数据库 2023年6月6日
    0118
  • 我的创作纪念日

    机缘 2018 年 08 月 07 日是我的创作一周年纪念日。大三刚过完,我还是一名 IT 小白,也并没有考研的想法,当时应该是在实习,偶尔一次上网搜索代码问题的时候看到了 CSD…

    数据库 2023年6月6日
    0132
  • Linux–>软件包管理

    rpm用于互联网下载包的打包和安装工具,它包含在某些Linux分发版本中。 它生成具有.RPM扩展名的文件。 RPM是RedHat Package Manager(RedHat软件…

    数据库 2023年6月14日
    0127
  • Java的日志框架之Logback

    前言 在Java的日志系统里面,有两个概念,一个叫做日志框架,如我们熟悉的Logback, Log4j, Log4j2, JDK自带的java.util.logging等;一个叫做…

    数据库 2023年6月11日
    0103
  • JDK自带线程池学习

    JDK自带线程池 线程池的状态 线程有如下状态 RUNNING状态:Accept new tasks and process queued tasks SHUTDOWN状态:Don…

    数据库 2023年6月11日
    0138
  • 达梦产品技术支持培训-day6-DM性能诊断与优化

    (本文只作为个人随笔用途,非官方文档,请勿作他用,谢谢) 1、DM8查询优化基本思路 1.1 操作系统性能诊断 linux常用系统监控命令 top 主要关注DMserver 的CP…

    数据库 2023年6月11日
    0100
  • MySQL主从同步

    1. 主从同步的定义 主从同步使得数据可以从一个数据库服务器复制到其他服务器上,在复制数据时,一个服务器充当主服务器(master),其余的服务器充当从服务器(slave)。因为复…

    数据库 2023年6月14日
    0104
  • MySQL启动报:[ERROR] The server quit without updating PID file

    修改配置后 MySQL启动不了,报错: 看见这个不要惊慌,先把刚才修改的配置注释掉,看是不是配置有误!大部分是手误造成。 如果不行,再尝试一下方法: 解决方法 : 给予权限,执行 …

    数据库 2023年6月14日
    0140
  • Java8日期时间(LocalDate、LocalTime、LocalDateTime)

    在看题的时候发现了Java8中新加入的日期时间类,觉得这个小哥写的不错,自己也跟着练习下。原文地址:https://blog.csdn.net/yy139926/article/d…

    数据库 2023年6月14日
    0104
  • python中set()函数的用法

    set() 函数创建一个无序不重复元素集,可进行关系测试,删除重复数据,还可以计算交集、差集、并集等。 set([iterable]) iterable — 可迭代对象…

    数据库 2023年6月11日
    0106
  • Javaweb08-Ajax项目-分页条件查询 + 增删改

    1、登录 欢迎登录课工场KH96动漫管理系统 用户名: 用户密码: 欢迎登录课工场KH96动漫管理系统 用户名: 用户密码: $(function(){ //使用jQuery的Aj…

    数据库 2023年6月16日
    0137
  • Vue插件

    Vue插件 插件作用 插件通常用来为 Vue 添加全局功能 例如:1、添加全局资源:指令/过滤器/过渡等。如 vue-touch2、通过全局混入来添加一些组件选项。3、添加 Vue…

    数据库 2023年6月11日
    099
  • Django后台美化

    Django后台美化 1.Xadmin 1.1 安装 通过如下命令安装xadmin的最新版: pip install https://github.com/sshwsfc/xadm…

    数据库 2023年6月14日
    0107
  • mqtt长连接报错32000

    背景 项目需要使用mqtt协议建立长连接,我是客户端,需要连上服务端同学的提供的地址;客户端使用的是paho提供的客户端sdk,如下: org.eclipse.paho org.e…

    数据库 2023年6月11日
    0161
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球