4.事务、隔离级别、函数

发布时间:2023年12月28日

1 事务

思考:我去银行给朋友汇款,我卡上有1000元,朋友卡上500元,我给朋友转账50元(无手续费),如果,我的钱刚扣,而朋友的钱又没加时,网线断了,怎么办?

1.1 事务的特性
  • 原子性(Atomicity):原子意为最小的粒子,或者说不能再分的事物。数据库事务的不可再分的原则即为原子性。
  • 组成事务的所有查询必须:要么全部执行,要么全部取消(就像上面的银行例子)。
  • 一致性(Consistency):指数据的规则,在事务前/后应保持一致。
  • 隔离性(Isolation):简单点说,某个事务的操作对其他事务不可见的.
  • 持久性(Durability):当事务提交完成后,其影响应该保留下来,不能撤消
2.2 事务的用法步骤
  1. 开启事务(start transaction)
  2. 执行sql操作(普通sql操作)
  3. 提交/回滚(commit/rollback)
    注意:
  • 建表的时候,选择Innodb引擎才支持事务
  • 默认情况下,MySQL是自动提交事务,每次执行一个SQL语句时,如果执行成功,就会向数据库自动提交,而不能回滚。如果某一组操作需要在一个事务中,那么需要使用starttransaction,一旦rollback或commit就结束当次事务,之后的操作又自动提交。
  • 如果需要在当前会话的整个过程中都取消自动提交事务,进行手动提交事务,就需要设置setautocommit=false;或setautocommit=0;那样的话每一句SQL都需要手动commit提交才会真正生效。rollback或commit之前的所有操作都视为一个事务,之后的操作视为另一个事务,还需要手动提交或回滚。
  • 和Oracle一样,DDL语句是不能回滚的,并且部分的DDL语句会造成隐式的提交,因此最好事务中不要涉及DDL语句。

例:

#开启手动处理事务模式
#setautocommit=false;
#开始事务(推荐)
start transaction;
#查看当前表的数据
select * from t_stu_detail;
#删除整张表的数据
delete from t_stu_detail;
#查询该表数据,发现显示删除后的结果
select * from t_stu_detail;
#回滚
rollback
#查看当前表的数据,发现又回来了
select * from t_stu_detail;
#删除整张表的数据
delete from t_stu_detail;
#提交事务
commit;
#查看当前表的数据,发现真删除了
select * from t_stu_detail;
-------------------------------------------------
#插入一条记录
INSERTINTO t_stu_detail VALUES(1,'123456789012345678','1990-01-21','12345678901','a@163.com','北七家');
#保存还原点1
savepoint point1;
#插入一条记录
INSERTINTO t_stu_detail VALUES(2,'123456789012345677','1990-02-21','12345678902','b@163.com','北七家');
#保存还原点2
savepoint point2;
#查看当前效果
select * from t_stu_detail;
#回滚到某个还原点
rollback to point1;
#提交事务
commit;
---------------------------------------------------------
#清空表
truncate t_stu_detail;
#回滚,对于truncate无法回滚
rollback;
-------------------------------------------------------
#修改表结构
alter table t_stu_detail add description varchar(50);
#回滚,对于修改表结构的语句无法回滚
rollback;

2 数据库的隔离级别

对于同时运行的多个事务(多线程并发),当这些事务访问数据库中相同的数据时,如果没有采取必要的隔离机制,就会导致各种并发问题:(问题的本质就是线程安全问题,共享数据的问题)。

  • 脏读:对于两个事务T1,T2,T1读取了已经被T2更新但还没有被提交的字段.之后,若T2回滚,T1读取的内容就是临时且无效的.
  • 不可重复读:对于两个事务T1,T2,T1读取了一个字段,然后T2更新并提交了该字段.之后,T1再次读取同个字段,值就不同了.
  • 幻读:对于两个事务T1,T2,T1从一个表中读取了一个字段,然后T2在该表中插入、删除了一些新的行.之后,如果T1再次读取同一个表,就会多出、少了几行.

数据库事务的隔离性:数据库系统必须具有隔离并发运行各个事务的能力,使它们不会相互影响,避免各种并发问题。一个事务与其他事务隔离的程度称为隔离级别.数据库规定了多种事务隔离级别,不同隔离级别对应不同的干扰程度,隔离级别越高,数据一致性就越好,但并发性越弱。

Oracle支持的2种事务隔离级别:READCOMMITED,SERIALIZABLE.Oracle默认的事务隔离级别为:READCOMMITED
Mysql支持4中事务隔离级别.Mysql默认的事务隔离级别为:REPEATABLE-READ每启动一个mysql程序,就会获得一个单独的数据库连接.每个数据库连接都有一个变量@@tx_isolation,表示当前的事务隔离级别.

  • 查看当前的隔离级别:SELECT@@tx_isolation;
  • 查看全局的隔离级别:select@@global.tx_isolation;
  • 设置当前mySQL连接的隔离级别:settx_isolation=‘repeatable-read’;
  • 设置数据库系统的全局的隔离级别:setglobaltx_isolation=‘read-committed’;

2.1 隔离级别描述

未授权读取(Read Uncommitted)

也称为读未提交(Read Uncommitted):允许事务读取其他事务未提交的数据,会引发脏读取、不可重复读和虚读,但避免了更新丢失。如果一个事务已经开始写数据,则另外一个事务则不允许同时进行写操作,但允许其他事务读此行数据。该隔离级别可以通过“排他写锁”实现。

-- A窗口
set transaction isolation level  read uncommitted;--设置A用户的数据库隔离级别为Read uncommitted(读未提交)
start transaction;--开启事务
select * from account;--查询A账户中现有的钱,转到B窗口进行操作
select * from account--发现a多了100元,这时候A读到了B未提交的数据(脏读)
-- B窗口
start transaction;--开启事务
update account set money=money+100 where name='A';--不要提交,转到A窗口查询
授权读取(Read Committed)

也称为读提交(Read Committed):会引发不可重复读取和虚读,但避免脏读取。这可以通过“瞬间共享读锁”和“排他写锁”实现。读取数据的事务允许其他事务继续访问该行数据,但是未提交的写事务将会禁止其他事务访问该行。

-- A窗口
set transaction isolation level  read committed;
start transaction;
select * from account;--发现a帐户是1000元,转到b窗口
select * from account;--发现a帐户多了100,这时候,a读到了别的事务提交的数据,两次读取a帐户读到的是不同的结果(不可重复读)
-- B窗口
start transaction;
update account set money=money+100 where name='aaa';
commit;--转到a窗口
可重复读取(Repeatable Read)(mysql默认级别)

可重复读取(Repeatable Read):确保事务可以多次从一个字段中读取相同的值,好比在事务开启时对现有的数据进行了拍照,其他事务对数据的修改,不管事务是否提交,我这里读取的是拍照下来的数据,可以避免脏读和不可重复读,但幻读的问题仍然存在。注意:INNODB使用了MVCC(MultiversionConcurrencyControl),即多版本并发控制技术防止幻读。真正的像拍照一样,其他事务新插入或删除的记录也看不出来。

-- A窗口
set transaction isolation level repeatable read;
start transaction;
select * from account;--发现表有4个记录,转到b窗口
select * from account;--可能发现表有5条记录,这时候发生了a读取到另外一个事务插入的数据(虚读)
-- B窗口
start transaction;
insert into account(name,money) values('ggg',1000);
commit;--转到a窗口

序列化(Serializable)

提供严格的事务隔离。它要求事务序列化执行,事务只能一个接着一个地执行,不能并发执行,性能十分低下。仅仅通过“行级锁”是无法实现事务序列化的,必须通过其他机制保证新插入的数据不会被刚执行查询操作的事务访问到。

-- A窗口
set transaction isolation level Serializable;
start transaction;
select * from account;--转到b窗口
-- B窗口
start transaction;
insert into account(name,money) values('ggg',1000);--发现不能插入,只能等待a结束事务才能插入

3. 函数

MySQL 数据库提供了很多函数包括:

  • 数学函数;
  • 字符串函数;
  • 日期和时间函数;
  • 条件判断函数;流程控制函数;
  • 系统信息函数;
  • 加密函数;
  • 格式化函数;
数学函数

在这里插入图片描述

字符串函数

在这里插入图片描述

日期时间函数

在这里插入图片描述

  • (1)DATE_ADD(datetime,INTERVAL expr type)
    在这里插入图片描述

  • (2)DATE_FORMAT(datetime ,fmt)和 STR_TO_DATE(str, fmt)
    在这里插入图片描述

流程函数

在这里插入图片描述

文章来源:https://blog.csdn.net/muLanlh/article/details/135248884
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。