1、demo
第一个代码是多线程的简单使用,编写了线程如何执行函数和类。
import threading import time class ClassName(threading.Thread): """创建类,通过多线程执行""" def run(self): for i in range(5): print(i) time.sleep(1) def sing(): for i in range(1,11): print("唱歌第 %d 遍" % i) time.sleep(1) def dance(): for i in range(1,16): print("跳舞第 %d 遍" % i) time.sleep(1) def main(): t1 = threading.Thread(target = sing) t2 = threading.Thread(target = dance) t = ClassName() # 启动线程 t1.start() t2.start() t.start() while True: length = len(threading.enumerate()) print("正在运行的线程有 %s" %threading.enumerate()) if length <= 1: break time.sleep(1) if __name__ == '__main__': main()
执行结果可以看到函数 sing、dance和类在同时执行,执行效果太长就不方截图了
2、多线程共享变量
通过定义全局变量,然后再test1函数类部进行更改全局变量,test2打印全局变量。
import threading import time #定义全局变量 g_num = 0 def test1(): """函数test1对全局变量进行更改""" global g_num for i in range(1,10): g_num += 1 print("--- test1 线程 g_num = %d--- " % g_num) def test2(): """函数test2 打印全局变量""" print("--- test2 线程 g_num = %d--- " % g_num) def main(): t1 = threading.Thread(target=test1) t2 = threading.Thread(target=test2) # 启动线程 t1.start() # 增加睡眠是为了保证优先执行函数test1 time.sleep(1) t2.start() print("--- 主线程 g_num = %d--- " % g_num) if __name__ == '__main__': main()
执行结果可以看出,在主线程和创建的两个线程中读取的是一样的值,既可以表明在多线程中变量共享
3、资源竞争
在多线程两个函数中同时更改一个变量时,由于cpu的计算能力,当修改参数的代码块无法一次性执行完成时,就会产生资源竞争
import threading import time # 定义全局变量 g_num = 0 def test1(num): """函数test1对全局变量进行更改""" global g_num for i in range(num): g_num += 1 print("test1 线程 g_num = %d---" % g_num) def test2(num): """函数test2对全局变量进行更改""" global g_num for i in range(num): g_num += 1 print("tes2 线程 g_num = %d---" % g_num) def main(): t1 = threading.Thread(target=test1, args=(1000000, )) t2 = threading.Thread(target=test2, args=(1000000, )) t1.start() t2.start() time.sleep(1) print("主线程 g_num = %d---" % g_num) if __name__ == '__main__': main()
可以先试试传递参数为100时,可以看到g_num = 200 这是因为函数代码可以一次性执行完成,当参数为1000000时代码无法一次性执行完成,g_num!= 2000000
4、互斥锁
互斥锁可以解决资源竞争的问题,原理很简单,通过对代码块上锁,保证该代码执行完成前,其它代码无法进行修改。执行完成后解锁,其它代码就可以执行了。
import threading import time # 创建变量 g_num = 0 # 创建锁默认为开锁状态 mutex = threading.Lock() def test1(num): global g_num for i in range(num): # 上锁 mutex.acquire() g_num += 1 # 解锁 mutex.release() print("--- test1 线程 g_num = %d---" % g_num) def test2(num): global g_num for i in range(num): # 上锁 mutex.acquire() g_num += 1 # 解锁 mutex.release() print("--- test2 线程 g_num = %d---" % g_num) def main(): t1 = threading.Thread(target=test1, args=(1000000, )) t2 = threading.Thread(target=test2, args=(1000000, )) t1.start() t2.start() time.sleep(1) print("--- 主线程 g_num = %d---" % g_num) if __name__ == '__main__': main()
可以看到加了锁之后,代码执行不会出现资源竞争,结果也是正常的。互斥锁,上锁的代码越少越好。
5、死锁
当出现多个锁时,就可能会产生死锁这个情况。当关闭一个锁时,这个锁已经为关闭状态的话,程序就会阻塞。就如同下面这个代码中。函数test1关闭mutexB锁时,函数test2提前将其关闭了,未进行解锁,程序就会一直阻塞。
import threading import time # 创建两个锁A, B mutexA = threading.Lock() mutexB = threading.Lock() def test1(): # 对muctexA上锁 mutexA.acquire() # mutexA上锁后,延时1秒,等待mutexB上锁 print("test1 ---do1---up---") time.sleep(1) # 此时会堵塞,因为mutexB已经上锁 mutexB.acquire() print("test1 ---do1---down---") mutexB.release() # 对mutexA解锁 mutexA.release() def test2(): # 对muctexB上锁 mutexB.acquire() # mutexB上锁后,延时1秒,等待mutexA上锁 print("test2 ---do1---up---") time.sleep(1) # 此时会堵塞,因为mutexB已经上锁 mutexA.acquire() print("test2 ---do1---down---") mutexA.release() # 对mutexA解锁 mutexB.release() def main(): t1 = threading.Thread(target=test1) t2 = threading.Thread(target=test2) t1.start() t2.start() if __name__ == '__main__': main()
代码执行效果可以看到程序会一直阻塞
解决方法
1、在程序编写时,就需要注意避免死锁
2、可以参考银行家算法
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]