<del>当获取线程锁时,此线程独占CPU资源
锁主要用来解决资源竞争问题。
### Lock锁原语对象
下面的例子,数字顺序变乱是其次;但如果,多个线程同时操作全局变量num会造成最终的结果小于10000。
同时生成10000个线程,线程生成后,运行瞬间结束,所以生成的数字顺序是正常的, 显示结果也是正常的。
当主线程还在生成下一个线程时,上一个子线程就已经结束了,所以不会有资源竞争。
```python
#!/usr/bin/env python
# coding=utf-8
from threading import Thread, Lock
from time import sleep
num = 0
def add():
global num
num += 1
print num
lock = Lock()
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
```
在线程中延迟了1秒,当主线程还在生成下一个线程时,上一个子线程还没结束了,生成线程的速度远快于线程结束的速度,造成资源竞争,数字顺序变乱,结果不正常。
```python
from threading import Thread, Lock
from time import sleep
num = 0
def add():
global num
sleep(1)
num += 1
print num
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
```
---
如果想解决上面的问题,可以在对全局变量进行造作时加锁。
实际上,这种做法只能避免资源竞争,但是按顺序输出无法保证
```python
#!/usr/bin/env python
# coding=utf-8
from threading import Thread, Lock
from time import sleep
num = 0
def add():
global num
sleep(1)
# 加锁后,只有一个线程能获得锁,所以操作变量后立即释放锁
lock.acquire()
num += 1
lock.release()
print num
# 生成锁
lock = Lock()
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
```
---
### 递归锁RLock
使单一线程可以获得已持有的锁。
产生死锁
说明:
```python
#!/usr/bin/env python
# coding=utf-8
from threading import Thread, Lock
from time import sleep
num = 0
num2 = 0
def add():
global num, num2
sleep(1)
# 获得一把锁
lock.acquire()
num += 1
# 等待自己的上一把锁释放,可是自己一直没有释放。
lock.acquire()
num2 += 1
lock.release()
lock.acquire()
print (num,num2)
# 生成锁
lock = Lock()
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
```
使用RLcok解决死锁问题, RLock使单一线程能够再次获取自己的锁
```python
#!/usr/bin/env python
# coding=utf-8
from threading import Thread, RLock
from time import sleep
num = 0
num2 = 0
def add():
global num, num2
sleep(1)
# 获得一把锁
lock.acquire()
num += 1
# 获得一把锁
lock.acquire()
num2 += 1
# 释放两次
lock.release()
lock.release()
print num,num2
# 生成锁
lock = RLock()
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
```
### 使用上下文管理器
使用上下文管理器,可以简化代码。进入该套件之前,上下文管理器会调用acquire()方法,并在完成后调用release()方法。
~~~
#!/usr/bin/env python
# coding=utf-8
#python2
from threading import Thread, RLock
from time import sleep
num = 0
num2 = 0
def add():
global num, num2
sleep(1)
# 获得一把锁
with lock:
num += 1
# 获得一把锁
with lock:
num2 += 1
print num,num2
# 生成锁
lock = RLock()
for i in range(10000):
add_thread = Thread(target=add, args=())
add_thread.start()
~~~
- 前言
- 环境搭建
- pypi
- 打包
- Python 2 和 Python 3 的版本之间差别
- 项目
- 第一部分
- 第1章 基础
- Python安装
- python代码文件类型
- python对象
- 核心数据类型
- 核心数据类型--整型和浮点型
- 核心数据类型--字符串
- str.format
- 核心数据类型--列表
- 核心数据类型--元组
- 核心数据类型--字典
- 核心数据类型--集合
- 核心数据类型--文件对象
- 调用bash
- 标准输入输出
- str-repr
- 字符编码
- 迭代器和生成器
- 第2章 语句和语法
- 赋值语句
- if语句
- while语句
- for语句
- assert
- 第3章 函数
- 函数作用域
- 工厂函数
- 内置函数
- 递归
- 嵌套作用域和lambda
- 参数传递
- 函数式编程
- property可写与可读
- 第5章 模块
- 模块导入
- 模块命名空间
- 相对导入和绝对导入
- 模块重载
- 在模块中隐藏数据
- 过渡性重载
- 第6章 类
- 面向对象还是面向过程?
- 构造函数 析构函数
- call
- 运算符重载
- str()
- 待定
- 即时生成属性
- 多态
- 线程和进程
- thread模块
- threading模块
- threading线程锁
- 糖果机
- multiprocessing
- 阻塞非阻塞同步异步
- 单线程和多线程对比
- 生产者消费者模型
- 第二部分
- 获取系统资源信息
- 获取进程所占的物理内存
- dmidecode获取系统信息
- 网络编程
- 网络基础
- python中的套接字
- socket模块
- 第三部分 高级功能
- 闭包入门
- 闭包的应用
- 装饰器入门
- 装饰器应用
- 第四部分 项目实战
- graphite
- 模块
- collections
- datetime
- Enum
- faker
- fabric
- fileinput
- fire
- fnmatch
- getpass
- glob
- hashlib
- heapq
- json模块
- log
- os
- Paramiko
- parser
- platform
- pyyaml
- Queue
- random
- re
- 特殊符号和字符
- re模块
- shelves
- subprocess
- time
- urllib_urllib2_requests
- urllib urllib2
- requests
- 标准模块ConfigParser
- 扩展模块Mysqldb
- 扩展模块dns
- 扩展模块request
- uuid
- cacheout 缓存库
- delorean 时间
- 附录
- 内置函数
- python实现各种排序算法
- 常见报错
- pymongo
- pyrocksdb
- 常用
- ERROR