如何在 Django 中保证并发的数据一致性
1. 关于锁
1.1 乐观锁
乐观锁的出发点是,同一条数据很少会因为并发修改而产生冲突,适用于读多写少的场景,用以提高吞吐量。
实现方式,读取一个字段,执行处理逻辑,当需要更新数据时,再次检查该字段是否和第一次读取一致。如果一致,则更新数据,否则拒绝更新,重新读取后再提交。
1.2 悲观锁
悲观锁的出发点是,当一条数据正在被修改时,不允许其他任何关于这条数据的操作。
实现方式,读取一个字段之后,加锁,不允许其他任何读、写操作。执行处理逻辑,更新数据完毕后,释放锁。
1.3 比较
乐观锁的开销远低于悲观锁。
悲观锁可能会导致死锁。当 A 锁定了 a 资源,需要 b 资源。而 b 被 B 锁定,正在等待 a 资源。此时,导致出现死锁。但是,可以通过设置超时来处理这个问题。
悲观锁可以有效降低冲突后,重试的次数。
乐观锁可以提高响应速度。
2. Django 中的事务
Django 默认每条数据库操作都会被立即提交到数据库。
这样会导致一个问题,如果有一系列的数据库操作构成,要么全部执行,要么就全部都不执行,怎么办?
这时,就需要事务。将一系列数据库操作设置为一个事务,提交给数据库执行。
Django 提供 atomic 装饰器以开启事务。 atomic 使用一个参数来指定数据库的名字。如果不设置值,Django 就会使用系统默认的数据库。
2.1 整个 View 函数开启事务
- from django.db import transaction
- @transaction.atomic
- def viewfunc(request):
- # This code executes inside a transaction.
- do_stuff()
2.2 部分函数 do_more_stuff()
开启事务。
- from django.db import transaction
- def viewfunc(request):
- # This code executes in autocommit mode (Django's default).
- do_stuff()
- with transaction.atomic():
- # This code executes inside a transaction.
- do_more_stuff()
2.3 不要在事务中处理异常
- from django.db import transaction
- def viewfunc(request):
- do_stuff()
- try:
- with transaction.commit_on_success():
- do_more_stuff_1() # in transaction
- try:
- do_more_stuff_2() # not in transaction
- except:
- pass
- do_more_stuff_3() # in transaction
- except:
- pass
当退出原子块时,Django 会查看它是否正常退出或者是否有异常来确定是否提交或者回滚。
如果你捕获并处理原子块中的异常,可以能会隐藏 Django 中发生问题的事实。这样可能会造成非预期的行为。
3. 利用 F 函数更新运算
通常更新数据库的操作,需要将对象读取到内存。在内存中进行修改之后,再写回数据库。
在内存中的操作,如果存在同时操作的情况,会导致运算逻辑错误。
F() 函数的作用就是直接生成 SQL 语句,不必将需要更新的对象读取到内存。避免了并发导致的数据不一致问题。
- from django.db.models import F
- reporter = Reporters.objects.get(name='OICQ')
- reporter.stories_filed = F('stories_filed') + 1
- reporter.save()
4. 利用 select_for_update 函数
select_for_update 使用的是悲观锁。
select for update 函数使用数据库查询语句,select ... for update
对数据库进行操作。
这是数据库层面的,解决并发取数据后再修改的问题方法。
- def mark_as_readed(self, notification_id):
- # 让s elect for update 和 update 语句发生在一个完整的事务里面
- with transaction.commit_on_success():
- # 使用select_for_update 来保证并发请求同时只有一个请求在处理,其他的请求
- # 等待锁释放
- notification = Notification.objects.select_for_update().get(pk=notification_id)
- # 没有必要重复标记一个已经读过的通知
- if notication.has_readed:
- return
- notification.has_readed = True
- notification.save()
- # 在这里更新我们的计数器,嗯,我感觉好极了
- self.update_unread_count(-1)
如何在 Django 中保证并发的数据一致性的更多相关文章
- Django中管理并发操作
上一篇我们说了,如何在Django中进行事务操作,数据的原子性操作 涉及了事务操作,我们不得不考虑的另一个问题就是:并发操作 还是那个用户转账的操作 我们使用事务操作解决的操作中途服务器宕机问题 但是 ...
- 【Django】Django如何保证并发操作数据一致性问题
代码示例: 使用 select for update 数据库查询 select ... for update 是数据库层面上专门用来解决并发取数据后再修改的场景的,主流的关系数据库 比如mysql.p ...
- 如何在Django中配置MySQL数据库
直接上图 在项目中直接找到settings 文件 第一步 原始Django自带数据库 第二步将配置改成MySQL的数据 第三步 在__init__文件中告知Django使用MySQL数据 ...
- 如何在django中使用多个数据库
http://blog.chinaunix.net/uid-16728139-id-4364343.html
- Django中的信号
信号 Django 提供一个“信号分发器”,允许解耦的应用在框架的其它地方发生操作时会被通知到. 简单来说,信号允许特定的sender通知一组receiver某些操作已经发生. 这在多处代码和同一事件 ...
- Django 中使用 Celery
起步 在 <分布式任务队列Celery使用说明> 中介绍了在 Python 中使用 Celery 来实验异步任务和定时任务功能.本文介绍如何在 Django 中使用 Celery. 安装 ...
- 「Python-Django」Django中使用数据库的 9 个小技巧
Django 中使用数据库的 9 个小技巧. 1. 过滤器聚合 在 Django 2.0 之前,如果你想得到“用户总数”.“活跃用户总数”等信息时,你不得不使用条件表达式. Django 2.0 中, ...
- 在Django中使用ORM创建图书管理系统
一.ORM(对象关系映射) 很多语言的web框架中都有这个概念 1. 为什么要有ORM? 1. 写程序离不开数据,要使用数据就需要连接数据库,但是不同的数据库在sql语句上(mysql,oracle等 ...
- Django中涉及金融的项目
在Django中,如果一个项目涉及了金融,他的要求是十分严格的. 所以嘞,这里就有一些坑,很多坑,第一次开发的时候很容易出现一系列的错误 在涉及金融计算的地方,不能使用float类型 什么鬼,但事实就 ...
随机推荐
- java hibernate session create
public class RegisterStory { private SysUserCDao sysUserCDao; @Test public void test() { SessionFact ...
- iOS开发调试Reveal使用
推荐通过Xcode中加断点的方式集成Reveal(小缺陷,当你禁用断点时或者不用Xcode而用Appcode开发时,这个方式是不管用). 打开您的iOS工程,选择 View → Navigators ...
- (转)Groupon前传:从10个月的失败作品修改,1个月找到成功 并不挶泥在这个点子上面,它反而往后站一步,看看他们已经做好的这个网站,可以再怎么包装成另一个完完全全不同的网站?所有的人所做的每件失败的事情中, 一定有碰到或含有成功的答案」在里面,只是他们不知道而已。 人不怕失败」,只怕宣布失败」
(转)Groupon前传:从10个月的失败作品修改,1个月找到成功 今天读到 一个非常励志人心的故事 ,就像现在「叶问」有「前传」,最近很火红的团集购网站Groupon 也出现了「Groupon前传」 ...
- 将windows目录共享到linux
1.将windows目录共享 2.安装cifs 3. mount -t cifs -o username=电脑登陆用户名,password=电脑登陆用户密码 //127.0.0.1/abc /var ...
- 实例应用 自定义页面taglib标签
关于继承TagSupport与BodyTagSupport的区别说明 * <code>TagSupport</code>与<code>BodyTagSupport& ...
- Android面试之HashMap的实现原理
1.HashMap与HashTable的区别 HashMap允许key和value为null: HashMap是非同步的,线程不安全,也可以通过Collections.synchronizedMap( ...
- php管理nginx虚拟主机shell脚本
使用php作为shell脚本是一件很方便的事情.理所当然,我们可以使用php脚本来管理 nginx虚拟主机,下面是笔者的 脚本 文件供各位参考 代码如下 复制代码 #!/usr/bin/php -q& ...
- Sublime text —— 自定义Color theme
网上下载,XXX.tmTheme 样式,让后放置于 C:\Users\{用户名}\AppData\Roaming\Sublime Text 2\Packages\Color Scheme - Defa ...
- A标签实现文件下载功能
<a>可直接下载xls,doc,rar,zip,exe,js文件(图片跟txt文件是直接打开的) <a href="wKioJlJolKeCIzkCADd3Wf7OPI42 ...
- 一:php配置注意
display_errors string 该选项设置是否将错误信息作为输出的一部分显示到屏幕,或者对用户隐藏而不显示. 设置 "stderr" 表示发送到 stderr 而不是 ...