MySQL【五】 【事物 】【数据库锁】

 

1.数据库事物

  1. 什么是事务 
    事务是应用程序中一系列严密的操作,所有操作必须成功完成,否则在每个操作中所作的所有更改都会被撤消。也就是事务具有原子性,一个事务中的一系列的操作要么全部成功,要么一个都不做。 
    事务的结束有两种,当事务中的所以步骤全部成功执行时,事务提交。如果其中一个步骤失败,将发生回滚操作,撤消撤消之前到事务开始时的所以操作。

  总结:事物就是一组操作,要么全部完成,要么全部失败!

  2.事物特性 ACID

  事务具有四个特征:原子性( Atomicity )、一致性( Consistency )、隔离性( Isolation )和持续性( Durability )。这四个特性简称为 ACID 特性。

  1 、原子性  (事物内的操作,要么全部成功,要么全部失败)
    事务是数据库的逻辑工作单位,事务中包含的各操作要么都做,要么都不做

  2 、一致性 (事物之前之后,前后数据的一致性)
    事务执行的结果必须是使数据库从一个一致性状态变到另一个一致性状态。因此当数据库只包含成功事务提交的结果时,就说数据库处于一致性状态 如果数据库系统 运行中发生故障,有些事务尚未完成就被迫中断,这些未完成事务对数据库所做的修改有一部分已写入物理数据库,这时数据库就处于一种不正确的状态,或者说是 不一致的状态。

原子性和一致性的的侧重点不同:原子性关注状态,要么全部成功,要么全部失败,不存在部分成功的状态
一致性关注数据的可见性中间状态数据外部不可见,只有最初状态和最终状态数据对外可见
但是因为某某性能的关系,对于一致性做了妥协

  3 、隔离性(多个事物时,相互不能干扰) 
一个事务的执行不能被其它事务干扰。即一个事务内部的操作及使用的数据对其它并发事务是隔离的,并发执行的各个事务之间不能互相干扰。
 

    4 、持续性(一旦事物提交(commit)之后,是不可回滚的) 
也称永久性,指一个事务一旦提交,它对数据库中的数据的改变就应该是永久性的。接下来的其它操作或故障不应该对其执行结果有任何影响。

 3.例子:  现实中的转账操作:

  1. 开启事物  start transaction
  2. 提交事物   commit
  3. 回滚事物   rollback

1.创建表

create table account(
id int(50) not null auto_increment primary key,
a_name VARCHAR(50) not null,
money DOUBLE(5,2) not NULL
);

2.插入数据

insert into account (a_name,money) values('鲁班',159.11),('后羿',333);

3.执行转账

start transaction; -- 开启事物
-- 执行sql语句操作
update account set money = money - 500 where id ='1'; #当提交的两个语句中有一个是有问题的则此次提交,不会执行这一次的内容,
update account set money = money+500 where id = '2'; #但是会将没有问题的一句放入缓存区中,当下一次执行此事务时
#start transaion ---开启事务,(+对应的)会将缓存区中存的内容(正确的哪一句)执行下去,
commit; -- 手动提交事物
rollback; -- 回滚事物 --
查看结果
select * from account;

 #记住一个概念就是当 

 

2.数据库锁

  

1. 锁的基本概念

  当并发事务同时访问一个资源时,有可能导致数据不一致,因此需要一种机制来将数据访问顺序化,以保证数据库数据的一致性。

2. 锁的基本类型

  数据库上的操作可以归纳为两种:读和写。

  多个事务同时读取一个对象的时候,是不会有冲突的。同时读和写,或者同时写才会产生冲突。因此为了提高数据库的并发性能,通常会定义两种锁:共享锁和排它锁。

2.1 共享锁(Shared Lock,也叫S锁)

    共享锁(S)表示对数据进行读操作。因此多个事务可以同时为一个对象加共享锁。(如果试衣间的门还没被锁上,顾客都能够同时进去参观)

2.2 排他锁(Exclusive Lock,也叫X)

       排他锁也叫写锁。

    排他锁表示对数据进行写操作。如果一个事务对对象加了排他锁,其他事务就不能再给它加任何锁了。(某个顾客把试衣间从里面反锁了,其他顾客想要使用这个试衣间,就只有等待锁从里面给打开了)

3.开发中常见的两种锁:

  3.1悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block(阻塞)直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制.

  

注:要使用悲观锁,我们必须关闭mysql数据库的自动提交属性,因为MySQL默认使用autocommit模式,也就是说,当你执行一个更新操作后,MySQL会立刻将结果进行提交。关闭自动提交命令为:set autocommit=0;

设置完autocommit后,我们就可以执行我们的正常业务了。具体如下:

//0.开始事务
start transaction; //1.查询出数据信息
select * from ren where p_id='p001' for update; //2.修改当前人员信息
update ren set p_name='张1丰' where p_id='p001'; //3. 提交事务
commit;

在另外的查询页面执行:

select * from ren where p_id='p001' for update;

会发现当前查询会进入到等待状态,不会显示出数据,当上面的sql执行完毕提交事物后,当前sql才会显示结果.

上面的第一步我们执行了一次查询操作:select * from ren where p_id='p001'  for update;

与普通查询不一样的是,我们使用了select…for update的方式,这样就通过数据库实现了悲观锁。此时在ren表中,id为p001的 那条数据就被我们锁定了,其它的事务必须等本注:需要注意的是,在事务中,只有SELECT ... FOR UPDATE 或LOCK IN SHARE MODE(共享锁) 同一笔数据时会等待其它事务结束后才执行,一般SELECT ... 则不受此影响。拿上面的实例来说,当我执行select * from ren where p_id='p001' for update;后。我在另外的事务中如果再次执行select * from ren where p_id='p001' for update;则第二个事务会一直等待第一个事务的提交,此时第二个查询处于阻塞的状态,但是如果我是在第二个事务中执行select * from ren where p_id='p001';则能正常查询出数据,不会受第一个事务的影响。

补充:MySQL select…for updateRow LockTable Lock

上面我们提到,使用select…for update会把数据给锁住,不过我们需要注意一些锁的级别,MySQL InnoDB默认Row-Level Lock,所以只有「明确」地指定主键,MySQL 才会执行Row lock (只锁住被选取的数据) ,否则MySQL 将会执行Table Lock (将整个数据表单给锁住表锁)。

- 引擎
- innodb
- 事务
- 行锁/表锁
- 表锁:
select * from tb for update;
- 行锁:
select id,name from tb where id=2 for update ;
- myisam
- 全文索引
- 快
- 表锁
- select * from tb for update;

注意: 悲观锁的确保了数据的安全性,在数据被操作的时候锁定数据不被访问,但是这样会带来很大的性能问题。因此悲观锁在实际开发中使用是相对比较少的。

3.2 乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库如果提供类似于write_condition机制的其实都是提供的乐观锁。

使用乐观锁的两种方式:

1.使用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现 方式。何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加一。当我们提交更新的时候,判断数据库表对应记录 的当前版本信息与第一次取出来的version值进行比对,如果数据库表当前版本号与第一次取出来的version值相等,则予以更新,否则认为是过期数 据。

2.乐观锁定的第二种实现方式和第一种差不多,同样是在需要乐观锁控制的table中增加一个字段,名称无所谓,字段类型使用时间戳 (datatime), 和上面的version类似,也是在更新提交的时候检查当前数据库中数据的时间戳和自己更新前取到的时间戳进行对比,如果一致则OK,否则就是版本冲突。

代码实例:

 update account set name='ygz' , version = version+1 where id = '1' and version = 1;

两种锁各有优缺点:不可认为一种好于另一种,像乐观锁适用于写入比较少的情况下,即冲突真的很少发生的时候,这样可以省去了锁的开销,加大了系统的整个吞吐量。但如果经常产生冲突,上层应用会不断的进行retry(重试),这样反倒是降低了性能,所以这种情况下用悲观锁就比较合适.

4.数据类型介绍

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
       bit[(M)]
           二进制位(101001),m表示二进制位的长度(1-64),默认m=1
 
       tinyint[(m)] [unsigned] [zerofill]
 
           小整数,数据类型用于保存一些范围的整数数值范围:
           有符号:
               -128 ~ 127.
           无符号:
           ~ 255
 
       int[(m)][unsigned][zerofill]
 
           整数,数据类型用于保存一些范围的整数数值范围:
               有符号:
                   -2147483648 ~ 2147483647
               无符号:
            ~ 4294967295
 
           特别的:整数类型中的m仅用于显示,对存储范围无限制。例如: int(5),当插入数据2时,select 时数据显示为: 00002
 
       bigint[(m)][unsigned][zerofill]
           大整数,数据类型用于保存一些范围的整数数值范围:
               有符号:
                   -9223372036854775808 ~ 9223372036854775807
               无符号:
~  18446744073709551615
 
       decimal[(m[,d])] [unsigned] [zerofill]
           准确的小数值,m是数字总个数(负号不算),d是小数点后个数。 m最大值为65,d最大值为30。
 
           特别的:对于精确数值计算时需要用此类型
                  decaimal能够存储精确值的原因在于其内部按照字符串存储。
 
       FLOAT[(M,D)] [UNSIGNED] [ZEROFILL]
           单精度浮点数(非准确小数值),m是数字总个数,d是小数点后个数。
               无符号:
                   -3.402823466E+38 to -1.175494351E-38,
                   1.175494351E-38 to 3.402823466E+38
               有符号:
                   1.175494351E-38 to 3.402823466E+38
 
           **** 数值越大,越不准确 ****
 
       DOUBLE[(M,D)] [UNSIGNED] [ZEROFILL]
           双精度浮点数(非准确小数值),m是数字总个数,d是小数点后个数。
 
               无符号:
                   -1.7976931348623157E+308 to -2.2250738585072014E-308
                   2.2250738585072014E-308 to 1.7976931348623157E+308
               有符号:
                   2.2250738585072014E-308 to 1.7976931348623157E+308
           **** 数值越大,越不准确 ****
 
 
       char (m)
           char数据类型用于表示固定长度的字符串,可以包含最多达255个字符。其中m代表字符串的长度。
           PS: 即使数据小于m长度,也会占用m长度
       varchar(m)
           varchar数据类型用于变长的字符串,可以包含最多达255个字符。其中m代表该数据类型所允许保存的字符串的最大长度,只要长度小于该最大值的字符串都可以被保存在该数据类型中。
 
           注:虽然varchar使用起来较为灵活,但是从整个系统的性能角度来说,char数据类型的处理速度更快,有时甚至可以超出varchar处理速度的50%。因此,用户在设计数据库时应当综合考虑各方面的因素,以求达到最佳的平衡
 
       text
           text数据类型用于保存变长的大字符串,可以组多到65535 (2**16 − 1)个字符。
 
       mediumtext
           A TEXT column with a maximum length of 16,777,215 (2**24 − 1) characters.
 
       longtext
           A TEXT column with a maximum length of 4,294,967,295 or 4GB (2**32 − 1) characters.
 
 
       enum
           枚举类型,
           An ENUM column can have a maximum of 65,535 distinct elements. (The practical limit is less than 3000.)
           示例:
               CREATE TABLE shirts (
                   name VARCHAR(40),
                   size ENUM('x-small''small''medium''large''x-large')
               );
               INSERT INTO shirts (name, size) VALUES ('dress shirt','large'), ('t-shirt','medium'),('polo shirt','small');
 
       set
           集合类型
           A SET column can have a maximum of 64 distinct members.
           示例:
               CREATE TABLE myset (col SET('a''b''c''d'));
               INSERT INTO myset (col) VALUES ('a,d'), ('d,a'), ('a,d,a'), ('a,d,d'), ('d,a,d');
 
       DATE
           YYYY-MM-DD(1000-01-01/9999-12-31)
 
       TIME
           HH:MM:SS('-838:59:59'/'838:59:59'
 
       YEAR
           YYYY(1901/2155)
 
       DATETIME
 
           YYYY-MM-DD HH:MM:SS(1000-01-01 00:00:00/9999-12-31 23:59:59    Y)
 
       TIMESTAMP
 
           YYYYMMDD HHMMSS(1970-01-01 00:00:00/2037 年某时)                                      

  

 
分类: mysql

day41 python【事物 】【数据库锁】的更多相关文章

  1. day42 事物,数据库锁

    事物是把一些sql语句作为一个原子性操作,就是说我会写好几条sql语句,然后我想把这好几条的sql语句作为一个整体,然后让这个整体一起去运行,不可以拆分开,就像我们用面粉做一个馒头一样,我需要把这些面 ...

  2. python【事物 】【数据库锁】

    1.数据库事物 1. 什么是事务  事务是应用程序中一系列严密的操作,所有操作必须成功完成,否则在每个操作中所作的所有更改都会被撤消.也就是事务具有原子性,一个事务中的一系列的操作要么全部成功,要么一 ...

  3. MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁

    浏览目录: 1.视图 2.触发器 3.存储过程 4.函数 5.事物 6.数据库锁 7.数据库备份 1.视图 视图:是一个虚拟表,其内容由查询定义.同真实的表一样,视图包含一系列带有名称的列和行数据 视 ...

  4. MySQL 事物和数据库锁

    1.数据库事物 1. 什么是事务  事务是应用程序中一系列严密的操作,所有操作必须成功完成,否则在每个操作中所作的所有更改都会被撤消.也就是事务具有原子性,一个事务中的一系列的操作要么全部成功,要么一 ...

  5. mysql第五篇 : MySQL 之 视图、触发器、存储过程、函数、事物与数据库锁

    第五篇 : MySQL 之 视图.触发器.存储过程.函数.事物与数据库锁 一.视图 视图是一个虚拟表(非真实存在的),其本质是‘根据SQL语句获取动态的数据集,并为其命名‘ ,用户使用时只需使用“名称 ...

  6. python操作数据库

    一,安装mysql 如果是windows 用户,mysql 的安装非常简单,直接下载安装文件,双击安装文件一步一步进行操作即可. Linux 下的安装可能会更加简单,除了下载安装包进行安装外,一般的l ...

  7. ABAP锁、数据库锁

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  8. Sqlite事物与锁

    1事务 事务定义了一组SQL命令的边界,这组命令或者作为一个整体被全部执行,或者都不执行.事务的典型实例是转帐. 2事务的范围 事务由3个命令控制:BEGIN.COMMIT和ROLLBACK.BEGI ...

  9. 一例胜千言,详谈SQL Sever数据库锁

    1 前言 数据库大并发操作要考虑死锁和锁的性能问题.看到网上大多语焉不详(尤其更新锁),所以这里做个简明解释,为下面描述方便,这里用T1代表一个数据库执行请求,T2代表另一个请求,也可以理解为T1为一 ...

随机推荐

  1. 原生js实现选项卡

    html代码: <div class="tab"> <ul> <li class="selected">图片</li& ...

  2. 『cs231n』RNN之理解LSTM网络

    概述 LSTM是RNN的增强版,1.RNN能完成的工作LSTM也都能胜任且有更好的效果:2.LSTM解决了RNN梯度消失或爆炸的问题,进而可以具有比RNN更为长时的记忆能力.LSTM网络比较复杂,而恰 ...

  3. Python基础--Python简介和入门

    ☞写在前面 在说Python之前,我想先说一下自己为什么要学Python,我本人之前也了解过Python,但没有深入学习.之前接触的语言都是Java,也写过一些Java自动化用例,对Java语言只能说 ...

  4. thinkphp关于时间加减几天

    1.当前时间,往后退5天: date('Y-m-d H:i:s',strtotime('-1 days')); 2.有固定时间,往后面退一天或者七天,或者30天: 比如时间:$time = 2014- ...

  5. Oracle12cr1新特性之容器数据库(CDB)和可插拔数据库(PDB) 的启动和关闭

    Oracle12c中引入的多宿主选项(multitenant option)允许一个容器数据库容纳多个独立的可插拔数据库(PDB).本文将说明如何启动和关闭容器数据库(CDB)和可插拔数据库(PDB) ...

  6. chrome plugins

    ehpomnigmfglbkmnboidmmhhmicfdmom_1_1_0知行-时间管理 必开 Adkill and Media Download Cnblogs Wz(博客园网摘) Kami - ...

  7. html <frame>标签使用

    标签定义 frameset 中的一个特定的窗口(框架) frameset中的每个框架都可以设置不同的属性,比如border,scrolling,noresize等 frame的常用属性 width: ...

  8. 自动化测试框架Taffy

    Taffy Taffy是基于nosetests的自动化测试框架. Taffy主要用来测试后台服务(包括且不限于Http, Dubbo/hessian, Webservice, Socket等类型接口) ...

  9. delphi image控件上画矩形的问题

    画的时候发现鼠标点的位置和矩形的位置有偏移 后来查看发现是由于image控件时stretch属性设置为true,这样在画的时候是缩放了的 也就是画到image显示的图片上去了 所以如果要画矩形,要按照 ...

  10. Linux免密码登陆

    A电脑实现不用密码登陆到B电脑! vim /etc/ssh/sshd_config   编辑文件 01.在A电脑上生成ssh密钥 ssh-keygen -t rsa02.把A电脑生成的id_rsa.p ...