Mysql中的事务
1、什么是事务:数据库中的事务是指逻辑上的一组操作,这组操作要么都执行成功,要么都不执行成功。
2、事务的管理:默认情况下Mysql会自动管理事务,一条SQL语句独占一个事务。
也可以使用start transaction、rollback和commit人为方式管理。
在start transaction之后的多条语句就是一个事务,事务commit之前可以rollback。
3、在JDBC中管理事务:
connection.setAutoCommit(false);
connection.rollback();
connection.commit();
多语句时可以创建回滚点:
SavePoint sp = connection.setSavePoint();
connection.rollback(sp);
4、事务的四大特性:
(1)原子性:是指事务是一个不可分割的整体,事务中的操作要么全部发生,要么一个都不发生。
(2)一致性:事务处理前后数据的完整性必须保持一致。
完整性是指一个数据在某个时间点完全满足数据库中约束的要求。
(3)隔离性:是指多个用户访问同一个数据库时,一个用户的事务处理不能被其他用户的事务干扰,多个并发事务之间的数据要相互隔离。
(4)持久性:是指一个事务一旦被提交,它对数据库中的数据改变是永久的。
5、一些概念:
(1)脏读:一个事务读到另一个事务未提交的数据
(2)不可重复读:在一个事务中多次读取某一数据同的结果(一个事务读到另一个事务已经提交的数据)
(3)虚读幻读:是指一个事务内读到其他事务插入的数据,导致前后读取不一致(一个事务读到另一个事务已经提交的数据)
6、隔离级别:
查看当前隔离级别:select @@tx_isolation
设置隔离级别:set [session/global] transaction isolation level read uncommitted
session:只针对当前会话有效,默认为session
global:修改数据库默认隔离级别
Mysql默认隔离级别:repeatable read
(1)read uncommitted -- 数据库会出现脏读、不可重复读、虚读幻读问题
(2)read committed -- 数据库防止出现脏读,不防止不可重复读和虚读幻读问题
(3)repeatable read -- 数据库防止脏读、不可重复读问题,但不能防止虚读幻读问题
(4)serializable -- 数据库运行在单线程模式,所有的问题都会防止
7、锁
(1)排他锁:排他锁和任何锁都不能共存,在任何隔离级别下进行修改都会加排他锁
(2)共享锁:共享锁只能和共享锁共存,在非serializable级别下查询不加任何锁,在serializable级别下查询加共享锁。
8、线程安全问题
多个用户同时对同一数据进行并发修改操作时,就会导致线程安全问题的发生,使用锁机制保证同一时间只有一个线程能访问数据,这样虽然解决了线程安全问题,但由于同一时间只能有一个线程访问该数据,使得数据库的效率非常低。为了找到合适的解决方案,得先对事务的处理有个更深的认识。
(1)如果多个线程同时对同一数据做修改,一定会有线程安全问题,必须解决。
任何隔离级别下做修改操作,都会加排他锁,保证同一时间只能有一个线程访问数据。
(2)如果多个线程同时对同一数据做查询操作,不会产生线程安全问题,无需处理。
(3)如果多个线程都未运行在serializable级别,则查询都不加锁,不会排斥。
(4)如果多个线程都运行在serializable级别,则都加共享锁,不会排斥。
(5)如果多个线程中有serializable,也有其他,则serializable加共享锁,其他不加,不会排斥。
(6)如果一个线程修改数据,其他线程查询时,可能出现脏读、不可重复读、虚读幻读。但不是任何时候都会出现问题,只在某些应用场景才可能造成问题。
(7)在serializable隔离级别下查询加共享锁,共他客户端无论是什么隔离级别只要做修改操作,就加排他锁,只能等待,从而解决虚读幻读问题。
9、更新丢失问题
两个查询对同一个查询结果做修改,后修改会忽略先修改的结果,从而造成先修改的结果丢失。
解决方案一:将数据库隔离级别设置为serializable可避免更新结果丢失,但这样使得数据库效率低下,所以不会这么做。
解决方案二:锁机制
(1)乐观锁:假定每次都不会造成更新丢失,在数据库中增加一版本字段,每次更新都基于上一版本进行。
(2)悲观锁:假定每次都会造成更新丢失,在查询时手动加排他锁(select ... for update)。
如果查询多,修改少应使用乐观锁,减少锁表操作;
如果修改多,查询少应使用悲观锁,减少修改失败。
10、演示不同隔离级别下的并发问题:
假定有如下表,模拟银行转账的操作。
CREATE TABLE `card` (
`id` int(11) NOT NULL auto_increment,
`name` varchar(20) default NULL,
`money` double default NULL,
PRIMARY KEY (`id`)
);
INSERT INTO `card` VALUES(NULL,'john',1000);
INSERT INTO `card` VALUES(NULL,'mary',1000);
set transaction isolation level 设置事务隔离级别
select @@tx_isolation 查询当前事务隔离级别
(1)当把事务的隔离级别设置为read uncommitted时,会引发脏读、不可重复读和虚读
A窗口
set transaction isolation level read uncommitted;
start transaction;
select * from card;
-----发现a帐户是1000元,转到b窗口
B窗口
start transaction;
update card set money=money+100 where name='aaa';
-----不要提交,转到a窗口查询
select * from card
-----发现a多了100元,这时候a读到了b未提交的数据(脏读)
(2)当把事务的隔离级别设置为read committed时,会引发不可重复读和虚读,但避免了脏读
A窗口
set transaction isolation level read committed;
start transaction;
select * from card;
-----发现a帐户是1000元,转到b窗口
B窗口
start transaction;
update card set money=money+100 where name='aaa';
commit;
-----转到a窗口
select * from card;
-----发现a帐户多了100,这时候,a读到了别的事务提交的数据,两次读取a帐户读到的是不同的结果(不可重复读)
(3)当把事务的隔离级别设置为repeatable read(mysql默认级别)时,会引发虚读,但避免了脏读、不可重复读
A窗口
set transaction isolation level repeatable read;
start transaction;
select * from card;
----发现表有4个记录,转到b窗口
B窗口
start transaction;
insert into card(name,money) values('ggg',1000);
commit;
-----转到 a窗口
select * from card;
----可能发现表有5条记如,这时候发生了a读取到另外一个事务插入的数据(虚读)
(4)当把事务的隔离级别设置为Serializable时,会避免所有问题
A窗口
set transaction isolation level Serializable;
start transaction;
select * from card;
-----转到b窗口
B窗口
start transaction;
insert into card(name,money) values('ggg',1000);
-----发现不能插入,只能等待a结束事务才能插入

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

Go语言是一种高效、简洁且易于学习的编程语言,因其在并发编程和网络编程方面的优势而备受开发者青睐。在实际开发中,数据库操作是不可或缺的一部分,本文将介绍如何使用Go语言实现数据库的增删改查操作。在Go语言中,我们通常使用第三方库来操作数据库,比如常用的sql包、gorm等。这里以sql包为例介绍如何实现数据库的增删改查操作。假设我们使用的是MySQL数据库。

Hibernate多态映射可映射继承类到数据库,提供以下映射类型:joined-subclass:为子类创建单独表,包含父类所有列。table-per-class:为子类创建单独表,仅包含子类特有列。union-subclass:类似joined-subclass,但父类表联合所有子类列。

苹果公司最新发布的iOS18、iPadOS18以及macOSSequoia系统为Photos应用增添了一项重要功能,旨在帮助用户轻松恢复因各种原因丢失或损坏的照片和视频。这项新功能在Photos应用的"工具"部分引入了一个名为"已恢复"的相册,当用户设备中存在未纳入其照片库的图片或视频时,该相册将自动显示。"已恢复"相册的出现为因数据库损坏、相机应用未正确保存至照片库或第三方应用管理照片库时照片和视频丢失提供了解决方案。用户只需简单几步

HTML无法直接读取数据库,但可以通过JavaScript和AJAX实现。其步骤包括建立数据库连接、发送查询、处理响应和更新页面。本文提供了利用JavaScript、AJAX和PHP来从MySQL数据库读取数据的实战示例,展示了如何在HTML页面中动态显示查询结果。该示例使用XMLHttpRequest建立数据库连接,发送查询并处理响应,从而将数据填充到页面元素中,实现了HTML读取数据库的功能。

如何在PHP中使用MySQLi建立数据库连接:包含MySQLi扩展(require_once)创建连接函数(functionconnect_to_db)调用连接函数($conn=connect_to_db())执行查询($result=$conn->query())关闭连接($conn->close())

PHP中处理数据库连接报错,可以使用以下步骤:使用mysqli_connect_errno()获取错误代码。使用mysqli_connect_error()获取错误消息。通过捕获并记录这些错误信息,可以轻松识别并解决数据库连接问题,确保应用程序的顺畅运行。

PHP是一种广泛应用于网站开发的后端编程语言,它具有强大的数据库操作功能,常用于与MySQL等数据库进行交互。然而,由于中文字符编码的复杂性,在处理数据库中文乱码时常常会出现问题。本文将介绍PHP处理数据库中文乱码的技巧与实践,包括常见的乱码原因、解决方法和具体的代码示例。常见的乱码原因数据库字符集设置不正确:数据库在创建时需选择正确的字符集,如utf8或u

MySQL数据库管理系统的基本原理解析MySQL是一种常用的关系型数据库管理系统,它通过结构化查询语言(SQL)来进行数据存储和管理。本文将介绍MySQL数据库管理系统的基本原理,包括数据库的创建、数据表的设计、数据的增删改查等操作,并提供具体的代码示例。一、数据库的创建在MySQL中,首先需要创建一个数据库实例来存储数据。通过以下代码可以创建一个名为"my
