脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Python - python 包之 threading 多线程

python 包之 threading 多线程

2022-11-21 11:12autofelix Python

这篇文章主要介绍了python 包之 threading 多线程,文章通过实例化threading.Thread类创建线程,下文相关资料介绍,需要的朋友可以参考一下

一、创建一个线程

  • 通过实例化threading.Thread类创建线程
?
1
2
3
4
5
6
7
8
9
10
11
12
import threading
 
def func(s):
print(s)
 
if __name__ == '__main__':
# 创建线程
thread = threading.Thread(target=func, args=('hello',))
# 启动线程
thread.start()
# 等待线程结束
thread.join()

二、创建多个线程

?
1
2
3
4
5
6
7
8
9
10
11
12
13
import threading
 
def func(s):
print(s)
 
if __name__ == '__main__':
thread = [
threading.Thread(target=func, args=('1', ))
threading.Thread(target=func, args=('2', ))
]
 
[t.start() for t in thread]
[t.join() for t in thread]

三、线程同步

  • 使用锁实现线程同步
  • threading.Lock是直接通过_thread模块扩展实现的
  • 锁只有“锁定”和“非锁定”两种状态
  • 同一个线程获取锁后,如果在释放锁之前再次获取锁会导致当前线程阻塞,除非有另外的线程来释放锁,如果只有一个线程,并且发生了这种情况,会导致这个线程一直阻塞下去,即形成了死锁。
?
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
import time
import threading
 
# 创建锁
lock = threading.Lock()
# 全局变量
global_resource = [None] * 5
 
def change_resource(para, sleep):
# 请求锁
lock.acquire()
# 这段代码如果不加锁,第一个线程运行结束后global_resource中是乱的,输出为:结果是: ['hello', 'hi', 'hi', 'hello', 'hello']
# 第二个线程运行结束后,global_resource中还是乱的,输出为:结果是: ['hello', 'hi', 'hi', 'hi', 'hi']
global global_resource
for i in range(len(global_resource)):
global_resource[i] = para
time.sleep(sleep)
print("结果是:", global_resource)
 
# 释放锁
lock.release()
 
if __name__ == '__main__':
thread = [
threading.Thread(target=change_resource, args=('hi', 2))
threading.Thread(target=change_resource, args=('hello', 1))
]
 
[t.start() for t in thread]
[t.join() for t in thread]
 
# 结果是: ['hi', 'hi', 'hi', 'hi', 'hi']
# 结果是: ['hello', 'hello', 'hello', 'hello', 'hello']

四、递归锁

  • 上面线程同步使用的是普通锁,也就是只有锁的状态,并不知道是哪个线程加的锁
  • 这样的话使用普通锁时,对于一些可能造成死锁的情况,可以考虑使用递归锁来解决
  • 递归锁和普通锁的差别在于加入了“所属线程”和“递归等级”的概念
  • 释放锁必须有获取锁的线程来进行释放
?
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
import time
import threading
 
# 使用成一个递归锁就可以解决当前这种死锁情况
rlock_hi = rlock_hello = threading.RLock()
 
def test_thread_hi():
# 初始时锁内部的递归等级为1
rlock_hi.acquire()
print('线程test_thread_hi获得了锁rlock_hi')
time.sleep(2)
# 如果再次获取同样一把锁,则不会阻塞,只是内部的递归等级加1
rlock_hello.acquire()
print('线程test_thread_hi获得了锁rlock_hello')
# 释放一次锁,内部递归等级减1
rlock_hello.release()
# 这里再次减,当递归等级为0时,其他线程才可获取到此锁
rlock_hi.release()
 
def test_thread_hello():
rlock_hello.acquire()
print('线程test_thread_hello获得了锁rlock_hello')
time.sleep(2)
rlock_hi.acquire()
print('线程test_thread_hello获得了锁rlock_hi')
rlock_hi.release()
rlock_hello.release()
 
if __name__ == '__main__':
thread = [
threading.Thread(target=test_thread_hi)
threading.Thread(target=test_thread_hello)
]
 
[t.start() for t in thread]
[t.join() for t in thread]

五、信号锁

  • 一个信号量管理一个内部计数器
  • acquire()方法会减少计数器,release()方法则增加计数器
  • 计数器的值永远不会小于零
  • 当调用acquire()时,如果发现该计数器为零,则阻塞线程
  • 直到调用release()方法使计数器增加。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import time
import threading
 
# 创建信号量对象,初始化计数器值为3
semaphore3 = threading.Semaphore(3)
 
 
def thread_semaphore(index):
# 信号量计数器减1
semaphore3.acquire()
time.sleep(2)
print('thread_%s is running...' % index)
# 信号量计数器加1
semaphore3.release()
 
 
if __name__ == '__main__':
# 虽然会有9个线程运行,但是通过信号量控制同时只能有3个线程运行
# 第4个线程启动时,调用acquire发现计数器为0了,所以就会阻塞等待计数器大于0的时候
for index in range(9):
threading.Thread(target=thread_semaphore, args=(index, )).start()

到此这篇关于python 包之 threading 多线程的文章就介绍到这了,更多相关threading 多线程内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.51cto.com/autofelix/5167799

延伸 · 阅读

精彩推荐
  • PythonPython中的单继承与多继承实例分析

    Python中的单继承与多继承实例分析

    这篇文章主要介绍了Python中的单继承与多继承,结合实例详细分析了Python面向对象程序设计中单继承与多继承的概念、原理、实现方法及相关操作注意事项...

    chengqiuming7372021-02-20
  • PythonPyCharm代码回滚,恢复历史版本的解决方法

    PyCharm代码回滚,恢复历史版本的解决方法

    今天小编就为大家分享一篇PyCharm代码回滚,恢复历史版本的解决方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    MrLittleDog13032021-04-10
  • PythonPython 中 list 的各项操作技巧

    Python 中 list 的各项操作技巧

    最近在学习 python 语言。大致学习了 python 的基础语法。觉得 python 在数据处理中的地位和它的 list 操作密不可分,今天把相关基础操作记录到脚本之家平台...

    山羊0.05692020-09-29
  • Python使用Pycharm+PyQt5弹出子窗口的程序代码

    使用Pycharm+PyQt5弹出子窗口的程序代码

    这篇文章主要介绍了使用Pycharm+PyQt5弹出子窗口的解决方法,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要...

    机构师12232022-01-24
  • Pythonpython实战之利用pygame实现贪吃蛇游戏(一)

    python实战之利用pygame实现贪吃蛇游戏(一)

    这篇文章主要介绍了python实战之利用pygame实现贪吃蛇游戏,文中有非常详细的代码示例,对正在学习python的小伙伴们有很好的帮助哟,需要的朋友可以参考下...

    筱羊冰冰8702021-10-28
  • PythonPython补齐字符串长度的实例

    Python补齐字符串长度的实例

    今天小编就为大家分享一篇Python补齐字符串长度的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    橙子伟7042021-04-20
  • Python解决win10 vscode 无法激活python 虚拟环境的问题

    解决win10 vscode 无法激活python 虚拟环境的问题

    这篇文章主要介绍了win10 vscode 无法激活python 虚拟环境的解决办法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友...

    jianwang1611112022-01-21
  • PythonPython编程之字符串模板(Template)用法实例分析

    Python编程之字符串模板(Template)用法实例分析

    这篇文章主要介绍了Python编程之字符串模板(Template)用法,结合具体实例形式分析了Python字符串模板的功能、定义与使用方法,需要的朋友可以参考下...

    坏蛋是我8802020-11-27