044.Python线程的数据安全
线程的数据安全
1 数据混乱现象
from threading import Thread,Lock
num = 0
lst = [] def func1():
global num
for i in range(100000):
num -= 1 def func2():
global num
for i in range(100000):
num += 1
for i in range (10):
# 启动线程1
t1 = Thread(target=func1)
t1.start()
# 启动线程2
t2 = Thread(target=func2)
t2.start()
lst.append(t1)
lst.append(t2) for i in lst:
i.join()
print("主线程执行结束")
print(num)
执行
由于共享同一份,会出现数据混乱,如下结果
[root@node10 python]# python3 test.py
主线程执行结束
173624
[root@node10 python]# python3 test.py
主线程执行结束
131875
[root@node10 python]# python3 test.py
主线程执行结束
-4279
[root@node10 python]# python3 test.py
主线程执行结束
-93587
[root@node10 python]# python3 test.py
主线程执行结束
145643
这是因为线程是并发执行,同同一时间会有多个线程拿到同一个数据进行计算,然后再放回去,导致一个数字被多个线程修改,结果不准确
2 引入锁机制
from threading import Thread,Lock
num = 0
lst = [] def func1(lock):
global num
for i in range(100000):
#上锁修改数据
lock.acquire()
num -= 1
#解锁释放
lock.release() def func2(lock):
global num
for i in range(100000):
#使用with,自动上锁和解锁
with lock:
num += 1
lock = Lock()
for i in range (10):
# 启动线程1
t1 = Thread(target=func1,args=(lock,))
t1.start()
# 启动线程2
t2 = Thread(target=func2,args=(lock,))
t2.start()
lst.append(t1)
lst.append(t2) for i in lst:
i.join()
print("主线程执行结束")
print(num)
执行
[root@node10 python]# python3 test.py
主线程执行结束
0
结果正确,但是消耗的时间毕竟长,但是可以换取数据正确
3 信号量
import time,random,os def func(i,sem):
time.sleep(random.uniform(0.1,1))
with sem:
print (i)
print (time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()))
time.sleep(random.uniform(3,6)) sem = Semaphore(5)
for i in range (20):
Thread(target = func,args=(i,sem)).start()
执行
[root@node10 python]# python3 test.py
3
2020-02-23 03:55:24
5
2020-02-23 03:55:24
14
2020-02-23 03:55:24
18
2020-02-23 03:55:24
6
2020-02-23 03:55:24
0
2020-02-23 03:55:27
2
2020-02-23 03:55:29
15
2020-02-23 03:55:29
1
2020-02-23 03:55:29
19
2020-02-23 03:55:30
12
2020-02-23 03:55:32
4
2020-02-23 03:55:32
17
2020-02-23 03:55:34
16
2020-02-23 03:55:34
13
2020-02-23 03:55:35
7
2020-02-23 03:55:36
10
2020-02-23 03:55:36
8
2020-02-23 03:55:39
9
2020-02-23 03:55:39
11
2020-02-23 03:55:40
信号量就相当于同时可以上多把锁
死锁,递归锁,互斥锁
4 死锁现象
模拟一个俱乐部玩枪,两把枪,两盒子弹,四个人同时抢,只有同时抢到子弹和枪的人,才能玩枪
初步逻辑
from threading import Thread,Lock
import time
#首先创建两把锁
gun = Lock()
bullet = Lock()
#定义第一种,先抢到枪,再抢到子弹
def play1(name):
#抢到枪,上锁
gun.acquire()
print("%s拿到枪"%(name))
#抢到子弹上锁
bullet.acquire()
print ("%s抢到子弹"%(name))
print ("开枪玩一会") time.sleep(0.5)
#放下枪,解锁
gun.release()
print("%s放下枪"%(name))
#放下子弹,解锁
bullet.release()
print("%s放下子弹"%(name)) #定义第二种,先抢到子弹,再抢到枪
def play2(name):
#抢到子弹上锁
bullet.acquire()
print ("%s抢到子弹"%(name))
#抢到枪,上锁
gun.acquire()
print("%s拿到枪"%(name))
print ("开枪玩一会")
time.sleep(0.5)
#放下子弹,解锁
bullet.release()
print("%s放下子弹"%(name))
#放下枪,解锁
gun.release()
print("%s放下枪"%(name)) name_lst1 = ["John","Jim"]
name_lst2 = ["Tom","Jerry"]
for name in name_lst1:
Thread(target=play1,args=(name,)).start()
for name in name_lst2:
Thread(target=play2,args=(name,)).start()
这种情况就会有死锁现象
多次执行,结果如下
[root@node10 python]# python3 test.py
John拿到枪
Tom抢到子弹
#阻塞,是因为John和Tom同时上了枪锁和子弹锁,但是都没有解锁,造成死锁
[root@node10 python]# python3 test.py
John拿到枪
John抢到子弹
开枪玩一会
John放下枪
John放下子弹
Jim拿到枪
Tom抢到子弹
#阻塞,这次是John顺利的玩一局,但是到Jim和Tom同时抢到枪和子弹,上锁,但是没有解锁,造成死锁
[root@node10 python]# python3 test.py
John拿到枪
John抢到子弹
开枪玩一会
John放下枪
John放下子弹
Tom抢到子弹
Jim拿到枪
#阻塞
5 递归锁介绍
上几把锁,解几把锁
from threading import Thread,RLock
rlock = RLock()
def func(name):
rlock.acquire()
print(name,1)
rlock.acquire()
print(name,2)
rlock.acquire()
print(name,3)
rlock.release()
rlock.release()
rlock.release() for i in range(10):
t = Thread(target=func,args=("name%s" % (i),) )
t.start() print("程序执行结束")
执行
[root@node10 python]# python3 ceshi.py
name0 1
name0 2
name0 3
name1 1
name1 2
name1 3
name2 1
name2 2
name2 3
name3 1
name3 2
name3 3
程序执行结束
name4 1
name4 2
name4 3
name5 1
name5 2
name5 3
name6 1
name6 2
name6 3
name7 1
name7 2
name7 3
name8 1
name8 2
name8 3
name9 1
name9 2
name9 3
6 利用递归锁,解决死锁现象
临时用于快速解决服务器崩溃死锁的问题,用递归锁应急问题
from threading import Thread,RLock
import time
#首先创建递归锁
gun=bullet = RLock()
#定义第一种,先抢到枪,再抢到子弹
def play1(name):
#抢到枪,上锁
gun.acquire()
print("%s拿到枪"%(name))
#抢到子弹上锁
bullet.acquire()
print ("%s抢到子弹"%(name))
print ("开枪玩一会") time.sleep(0.5)
#放下枪,解锁
gun.release()
print("%s放下枪"%(name))
#放下子弹,解锁
bullet.release()
print("%s放下子弹"%(name)) #定义第二种,先抢到子弹,再抢到枪
def play2(name):
#抢到子弹上锁
bullet.acquire()
print ("%s抢到子弹"%(name))
#抢到枪,上锁
gun.acquire()
print("%s拿到枪"%(name))
print ("开枪玩一会")
time.sleep(0.5)
#放下子弹,解锁
bullet.release()
print("%s放下子弹"%(name))
#放下枪,解锁
gun.release()
print("%s放下枪"%(name)) name_lst1 = ["John","Jim"]
name_lst2 = ["Tom","Jerry"]
for name in name_lst1:
Thread(target=play1,args=(name,)).start()
for name in name_lst2:
Thread(target=play2,args=(name,)).start()
执行
[root@node10 python]# python3 test.py
John拿到枪
John抢到子弹
开枪玩一会
John放下枪
John放下子弹
Jim拿到枪
Jim抢到子弹
开枪玩一会
Jim放下枪
Jim放下子弹
Tom抢到子弹
Tom拿到枪
开枪玩一会
Tom放下子弹
Tom放下枪
Jerry抢到子弹
Jerry拿到枪
开枪玩一会
Jerry放下子弹
Jerry放下枪
执行成功,神奇
7 使用互斥锁解决
从语法上来看,锁是可以互相嵌套的,但是不要使用
上一次锁,就对应解开一把锁,形成互斥锁
拿枪和拿子弹是同时的,上一把锁就够了,不要分开上锁,也不要去写锁的嵌套,容易死锁
from threading import Thread,Lock
import time
#首先创建递归锁
mylock = Lock()
#定义第一种,先抢到枪,再抢到子弹
def play1(name):
#抢到枪和子弹,上锁
mylock.acquire()
print("%s拿到枪"%(name))
print ("%s抢到子弹"%(name)) print ("开枪玩一会")
time.sleep(0.5)
#放下枪,和子弹解锁
print("%s放下枪"%(name))
print("%s放下子弹"%(name))
mylock.release() #定义第二种,先抢到子弹,再抢到枪
def play2(name):
#抢到枪子弹上锁
mylock.acquire()
print ("%s抢到子弹"%(name))
print("%s拿到枪"%(name)) print ("%s开枪玩一会"%(name))
time.sleep(0.5)
#放枪下子弹,解锁
print("%s放下子弹"%(name))
print("%s放下枪"%(name))
mylock.release() name_lst1 = ["John","Jim"]
name_lst2 = ["Tom","Jerry"]
for name in name_lst1:
Thread(target=play1,args=(name,)).start()
for name in name_lst2:
Thread(target=play2,args=(name,)).start()
执行
[root@node10 python]# python3 test.py
John拿到枪
John抢到子弹
开枪玩一会
John放下枪
John放下子弹
Jim拿到枪
Jim抢到子弹
开枪玩一会
Jim放下枪
Jim放下子弹
Tom抢到子弹
Tom拿到枪
Tom开枪玩一会
Tom放下子弹
Tom放下枪
Jerry抢到子弹
Jerry拿到枪
Jerry开枪玩一会
Jerry放下子弹
Jerry放下枪
完成
044.Python线程的数据安全的更多相关文章
- python——线程与多线程进阶
之前我们已经学会如何在代码块中创建新的线程去执行我们要同步执行的多个任务,但是线程的世界远不止如此.接下来,我们要介绍的是整个threading模块.threading基于Java的线程模型设计.锁( ...
- python——线程与多线程基础
我们之前已经初步了解了进程.线程与协程的概念,现在就来看看python的线程.下面说的都是一个进程里的故事了,暂时忘记进程和协程,先来看一个进程中的线程和多线程.这篇博客将要讲一些单线程与多线程的基础 ...
- [python] 线程简介
参考:http://www.cnblogs.com/aylin/p/5601969.html 我是搬运工,特别感谢张岩林老师! python 线程与进程简介 进程与线程的历史 我们都知道计算机是由硬件 ...
- PYTHON线程知识再研习A
前段时间看完LINUX的线程,同步,信息号之类的知识之后,再在理解PYTHON线程感觉又不一样了. 作一些测试吧. thread:模块提供了基本的线程和锁的支持 threading:提供了更高级别,功 ...
- Python 线程(threading) 进程(multiprocessing)
*:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !important; } /* ...
- Python线程:线程的调度-守护线程
Python线程:线程的调度-守护线程 守护线程与普通线程写法上基本么啥区别,调用线程对象的方法setDaemon(true),则可以将其设置为守护线程.在python中建议使用的是thread. ...
- python 线程(一)理论部分
Python线程 进程有很多优点,它提供了多道编程,可以提高计算机CPU的利用率.既然进程这么优秀,为什么还要线程呢?其实,仔细观察就会发现进程还是有很多缺陷的. 主要体现在一下几个方面: 进程只能在 ...
- python线程同步原语--源码阅读
前面两篇文章,写了python线程同步原语的基本应用.下面这篇文章主要是通过阅读源码来了解这几个类的内部原理和是怎么协同一起工作来实现python多线程的. 相关文章链接:python同步原语--线程 ...
- Python学习——Python线程
一.线程创建 #方法一:将要执行的方法作为参数传给Thread的构造方法 import threading import time def show(arg): time.sleep(2) print ...
随机推荐
- SVM(1)模式识别课堂笔记
引言:当两类样本线性可分时,针对我们之前学习的感知机而言,存在多个超平面能将数据分开,这里要讨论什么样的分类面最好的问题.为此,我们形式化的定义了最优分类超平面,他有两点特征:1.能将训练样本没有错误 ...
- OpenGL学习网址2
http://www.opengpu.org/forum.php?mod=viewthread&tid=7525
- A Hybrid Data Association Framework for Robust Online Multi-Object Tracking(2017 IEEE Transactions on Image Processing)
A Hybrid Data Association Framework for Robust Online Multi-Object Tracking 一种用于鲁棒在线多目标跟踪的混合数据关联框架 摘 ...
- 《剑指Offer》第二章(一)题 9 -12
第二章 面试题9:用两个栈实现队列 题目:如面试题,给你两个栈, 实现队列的先进先出,即在队列头删除一个元素以及在队列的尾部添加一个元素 思路:这个题的分析感觉很巧妙,从一个具体的例子入手,找出其中的 ...
- PHP第三方登录——QQ登录
主要内容 简单回顾OAuth协议基本原理 接入QQ登录的前置条件以及开放平台账号申请 引入官方SDK SDK参数配置 SDK核心方法解读 整合QQ登录SDK到Web应用中 SDK优化 调用API的开发 ...
- 【题解】P1559 运动员最佳匹配问题
[题目](https://www.luogu.com.cn/problem/P1559) 题目描述 羽毛球队有男女运动员各n人.给定2 个n×n矩阵P和Q.P[i][j]是男运动员i和女运动员j配对组 ...
- SpringBoot、Spring MVC报错:Unable to create requested service [org.hibernate.engine.jdbc.env.spi.JdbcEnvironment]
出现问题的原因: jdbc配置不正确 解决方案: 1.检查是否已添加数据库驱动jar包 2.检查数据库服务是否启动 3.检查数据库配置文件 主要为:dialect,driver_class,url,u ...
- JS代码格式化时间戳
一.[24小时制]yyyy-MM-dd HH:mm:ss new Date().toJSON() // 2019-12-13T13:12:32.265Z 通过上面的方法,基本就可以将日期格式化,然后稍 ...
- lwip nd没有实现ra,contik有参考
lwip中关于nd的实现,没有路由器的功能,不能发送ra 在contiki中发现有nd发送ra的实现, contiki/core/net/ipv6/uip-ds6.c 在rs的接收处理中,发送soll ...
- k8s系列---存储卷pv/pvc。configMap/secert
因为pod是有生命周期的,pod一重启,里面的数据就没了.所以我们需要数据持久化存储. 在k8s中,存储卷不属于容器,而是属于pod.也就是说同一个pod中的容器可以共享一个存储卷. 存储卷可以是宿主 ...