pythonl练习笔记——threading创建进程锁Lock()
1 基本概述
采用threading.Lock()对象创建锁lock,即 lock = threading.Lock()
其相关方法主要有
lock.acquire() # lock the lock, possibly blocking until it can be obtained lock.release() # unlock of the lock lock.locked() # test whether the lock is currently locked
Lock(指令锁)是可用的最低级的同步指令;Lock处于锁定状态时,不被特定的线程拥有。
2 示例1
2.1 异常分析
import threading
import time
a = b =0
def value():
while True:
if a != b :
print("a = %d,b =%d"%(a,b))
t=threading.Thread(target=value)
t.start()
while True:
a += 1
b += 1
t.join()
运行结果为一串数字
... a = 3361552,b =3361551 a = 3361552,b =3361551 a = 3361552,b =3361551 a = 3361552,b =3361551 a = 3361552,b =3361551 ...
说明:
(1)按照程序分析来讲,应该没有输出,
(2)打印出好多相同的数字
可能原因
(1)首先明确a b为全局变量,所以当全局变量a b的值发生变化时,进程中的a b值也会随之变化
(2)当主线程中while True 使的a b均+1 时,线程函数中的a b值也会发生变化;但是主线程中的a b值赋值过程有时间间隔, 而线程中没有阻塞函数,所以在a b赋值中间的一个时刻,存在有a != b 这个过程,满足了线程函数的打印条件;所以会有数值打印出来
(3)在运行过程中,CPU高达100%,此时while 赋值运行和线程函数的运行并不是完全一致的,所以出现了数值较大的波动,有的数值仅出现一次或几次,有的数值甚至出现几十次。
(4)会出现当a赋值,当b没有赋值的间隙,子线程执行打印到桌面
2.2 添加sleep时间
import threading
import time
a = b =0
def value():
while True:
if a != b :
print("a = %d,b =%d"%(a,b))
t=threading.Thread(target=value)
t.start()
while True:
time.sleep(1) #添加sleep(1)时间
a += 1
b += 1
t.join()
运行
a = 3,b =3 a = 6,b =6 a = 15,b =15 a = 16,b =16 a = 17,b =17 a = 18,b =18 a = 19,b =19 a = 20,b =20
说明:
(1)打印出的数据之间的间隔不是“等差”关系
(2)运行过程可能是,
(a)a+1 赋值后,b尚未赋值,中间时间内线程判定出 a != b 为 True
(b)在print打印前,b+1赋值已经完成,而sleep(1)后,循环赋值中的a+1被阻塞;
(c)所以此时打印出的 a b值想等。
(3)为了说明 2 的过程,在b += 1 前添加 sleep(1),则打印出a的值比b的值均大1;且a b相同值行数达上百行。
(4)从 2 和 3 过程可以说明线
2.3 颠倒交换
import threading
import time
a = b =0
def value():
while True:
# 不添加global关键词,
# 则会 UnboundLocalError : local variable 'a' referenced before assignment
# 也即在赋值前没有定义局部变量 a b
global a
global b
a += 1
b += 1
t=threading.Thread(target=value)
t.start()
while True:
if a != b:
print("a = %d,b =%d" % (a, b))
t.join()
说明:
该出a b本是全局变量,而当不添加关键字global时则会出现未定义局部变量的报错。——暂时还没有搞懂为什么!
3 示例2
使用进程锁Lock
import threading
a = b =0
# 创建线程锁对象
lock = threading.Lock()
def value():
while True:
lock.acquire()
if a != b :
print("a = %d,b =%d"%(a,b))
lock.release()
t=threading.Thread(target=value)
t.start()
while True:
lock.acquire()
a += 1
b += 1
lock.release()
t.join()
运行,文件一直没有打印出内容
说明
(1)在a b 赋值时加锁,另外一个再加锁,这时候才会阻塞,此时不会打印。
(2)不管是子线程中的while循环 ,还是主线程中的while循环,任意一个去掉 lock.acquire lock.release 后,均不会出现阻塞。均会打印输出内容。
pythonl练习笔记——threading创建进程锁Lock()的更多相关文章
- python从入门到放弃之进程锁lock
# ### lock (互斥锁)"""# 应用在多进程当中# 互斥锁lock : 互斥锁是进程间的get_ticket互相排斥进程之间,谁先抢占到资源,谁就先上锁,等到解 ...
- 13.1、多进程:进程锁Lock、信号量、事件
进程锁: 为什么要有进程锁:假如现在有一台打印机,qq要使用打印机,word文档也要使用打印机,如果没有使用进程锁,可能会导致一些问题,比如QQ的任务打印到一半,Word插进来,于是打印出来的结果是各 ...
- python学习笔记——多进程中的锁Lock
1 进程锁 python编程中,引入了对象互斥锁的概念,来保证共享数据操作的完整性. 每个对象都对应于一个可称为“互斥锁”的标记,这个标记用来保证在任一时刻,只能有一线程访问对象. 在python中我 ...
- 进程锁Lock
例1: 10个人去买票,先去查询余票,有票就去购买.代码如下: # ticket.py {"ticket": 1} # 只有一张票 import json import time ...
- 进程锁 Lock
进程锁其实是锁的输入终端,屏幕,太多的进程,为了防止打印错乱而锁,作用不大 from multiprocessing import Process, Lock def f(l, i): l.acqui ...
- pythonl练习笔记——threading线程中的事件Event
1 事件Event 使用方法:e = threading.Event() Event对象主要用于线程间通信,确切地说是用于主线程控制其他线程的执行. Event事件提供了三个方法:wait等待.cle ...
- python 多进程锁Lock和共享内存
多进程锁 lock = multiprocessing.Lock() 创建一个锁 lock.acquire() 获取锁 lock.release() 释放锁 with lock: 自动获取.释放锁 类 ...
- python线程互斥锁Lock(29)
在前一篇文章 python线程创建和传参 中我们介绍了关于python线程的一些简单函数使用和线程的参数传递,使用多线程可以同时执行多个任务,提高开发效率,但是在实际开发中往往我们会碰到线程同步问题, ...
- python笔记9 线程进程 threading多线程模块 GIL锁 multiprocessing多进程模块 同步锁Lock 队列queue IO模型
线程与进程 进程 进程就是一个程序在一个数据集上的一次动态执行过程.进程一般由程序.数据集.进程控制块三部分组成.我们编写的程序用来描述进程要完成哪些功能以及如何完成:数据集则是程序在执行过程中所需要 ...
随机推荐
- IDEA热部署基于maven的web项目
第一步:首先先创建web项目 按照向导一步一步完成项目创建,我创建的示例项目为hotdeploddemo 第二步:设置项目的web资源 下图中的路径要修改正确,指向src/ma ...
- [leetcode]Binary Tree Zigzag Level Order Traversal @ Python
原题地址:http://oj.leetcode.com/problems/binary-tree-zigzag-level-order-traversal/ 题意: Given a binary tr ...
- WebView 加载网页 加载资源 总结 MD
Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...
- 以AVL树为例理解二叉树的旋转(Rotate)操作
树旋转是在二叉树中的一种子树调整操作, 每一次旋转并不影响对该二叉树进行中序遍历的结果. 树旋转通常应用于需要调整树的局部平衡性的场合. 树旋转包括两个不同的方式, 分别是左旋转和右旋转. 两种旋转呈 ...
- C++ vector用法(转)
在c++中,vector是一个十分有用的容器,下面对这个容器做一下总结. 1 基本操作 (1)头文件#include<vector>. (2)创建vector对象,vector<in ...
- Cocos2d-x教程(31)-TableView的滚动栏
欢迎增加Cocos2d-x 交流群:193411763 转载时请注明原文出处 :http://blog.csdn.net/u012945598/article/details/38587659 在非常 ...
- js正则表达式/replace替换变量方法
转自:http://www.blogjava.net/pingpang/archive/2012/08/12/385342.html 1. javascript 正则对象替换创建和用法:/patter ...
- PHP高级教程-多维数组
PHP 多维数组 一个数组中的值可以是另一个数组,另一个数组的值也可以是一个数组.依照这种方式,我们可以创建二维或者三维数组: 实例 <?php // 二维数组: $cars = array ( ...
- Ubuntu系统环境变量配置文件(转)
原文:http://www.cnblogs.com/eastson/archive/2012/06/15/2550151.html 在Ubuntu中有如下几个文件可以设置环境变量: /etc/prof ...
- cordova 强制竖屏
orentation的默认值是default 可使用的值有:default, landscape (横屏), portait (竖屏) orentation可以将设备锁定方向,不受设备旋转影响. 方案 ...