一篇文章帶你全面解析不一樣的線程
前言
在將今天的知識點之前,大家是否了解線程,進程和協程了,那我們先來初步了解下吧。
##中央處理器的調度單元,簡單點說就是程式中的末端執行者,相當於小弟的位置。
###有人說python中的線程是個雞肋,這是因為有了GIL,但是又不是一味的雞肋,畢竟在執行io操作時還是挺管用的,只是在執行計算時就顯得不盡人意。下面我們來看看下執行緒的具體使用方法:#######1.導入執行緒模組:
import threading as t
#2.執行緒的用法
tt=t.Thread(group=None,target=None,name=None,args=(),kwargs={},name='',daemon=None) group:线程组,必须是None target:运行的函数 args:传入函数的参数元组 kwargs:传入函数的参数字典 name:线程名 daemon:线程是否随主线程退出而退出(守护线程) Thread方法的返回值还有以下方法: tt.start() : 激活线程, tt.getName() : 获取线程的名称 tt.setName() :设置线程的名称 tt.name : 获取或设置线程的名称 tt.is_alive() :判断线程是否为激活状态 tt.isAlive() :判断线程是否为激活状态 tt.setDaemon() 设置为守护线程(默认:False) tt.isDaemon() :判断是否为守护线程 tt.ident :获取线程的标识符。只有在调用了start()方法之后该属性才有效 tt.join() :逐个执行每个线程,执行完毕后继续往下执行 tt.run() :自动执行线程对象 t的方法也有: t.active_count(): 返回正在运行线程的数量 t.enumerate(): 返回正在运行线程的列表 t.current_thread().getName() 获取当前线程的名字 t.TIMEOUT_MAX 设置t的全局超时时间
#下面我們來看看:
3.建立執行緒
執行緒可以使用Thread方法創建,也可以重寫執行緒類別的run方法實現,執行緒可分為單執行緒和多執行緒。
3.1 使用Thread方法来创建:
3.1.1 单线程
def xc(): for y in range(100): print('运行中'+str(y)) tt=t.Thread(target=xc,args=()) #方法加入到线程 tt.start() #开始线程 tt.join() #等待子线程结束
3.1.2 多线程
def xc(num): print('运行:'+str(num)) c=[] for y in range(100): tt=t.Thread(target=xc,args=(y,)) tt.start() #开始线程 c.append(tt) #创建列表并添加线程 for x in c: x.join() #等待子线程结束
3.2 重写线程的类方法
3.2.1 单线程
class Xc(t.Thread): #继承Thread类 def __init__(self): super(Xc, self).__init__() def run(self): #重写run方法 for y in range(100): print('运行中'+str(y)) x=Xc() x.start() #开始线程 x.join() #等待子线程结束 也可以这么写: Xc().run() 和上面的效果是一样的
3.2.2 多线程
class Xc(t.Thread): #继承Thread类 def __init__(self): super(Xc, self).__init__() def run(self,num): #重写run方法 print('运行:'+str(num)) x=Xc() for y in range(10): x.run(y) #运行
4.线程锁
为什么要加锁,看了这个你就知道了:
多线程在运行时同时访问一个对象会产生抢占资源的情况,所以我们必须得束缚它,所以就要给他加一把锁把他锁住,这就是同步锁。要了解锁,我们得先创建锁,线程中有两种锁:Lock和RLock。
4.1 Lock
使用方法:
# 获取锁 当获取不到锁时,默认进入阻塞状态,设置超时时间,直到获取到锁,后才继续。非阻塞时,timeout禁止设置。如果超时依旧未获取到锁,返回False。 Lock.acquire(blocking=True,timeout=1) #释放锁,已上锁的锁,会被设置为unlocked。如果未上锁调用,会抛出RuntimeError异常。 Lock.release()
互斥锁,同步数据,解决多线程的安全问题:
n=10 lock=t.Lock() def xc(num): lock.acquire() print('运行+:'+str(num+n)) print('运行-:'+str(num-n)) lock.release() c=[] for y in range(10): tt=t.Thread(target=xc,args=(y,)) tt.start() c.append(tt) for x in c: x.join()
这样就显得有条理了,而且输出也是先+后-。Lock在一个线程中多次使用同一资源会造成死锁。
死锁问题:
n=10 lock1=t.Lock() lock2=t.Lock() def xc(num): lock1.acquire() print('运行+:'+str(num+n)) lock2.acquire() print('运行-:'+str(num-n)) lock2.release() lock1.release() c=[] for y in range(10): tt=t.Thread(target=xc,args=(y,)) tt.start() c.append(tt) for x in c: x.join()
4.2 RLock
相比Lock它可以递归,支持在同一线程中多次请求同一资源,并允许在同一线程中被多次锁定,但是acquire和release必须成对出现。
使用递归锁来解决死锁:
n=10 lock1=t.RLock() lock2=t.RLock() def xc(num): lock1.acquire() print('运行+:'+str(num+n)) lock2.acquire() print('运行-:'+str(num-n)) lock2.release() lock1.release() c=[] for y in range(10): tt=t.Thread(target=xc,args=(y,)) tt.start() c.append(tt) for x in c: x.join()
这时候,输出变量就变得仅仅有条了,不在随意抢占资源。关于线程锁,还可以使用with更加方便:
#with上下文管理,锁对象支持上下文管理 with lock: #with表示自动打开自动释放锁 for i in range(10): #锁定期间,其他人不可以干活 print(i) #上面的和下面的是等价的 if lock.acquire(1):#锁住成功继续干活,没有锁住成功就一直等待,1代表独占 for i in range(10): #锁定期间,其他线程不可以干活 print(i) lock.release() #释放锁
4.3 条件锁
等待通过,Condition(lock=None),可以传入lock或者Rlock,默认Rlock,使用方法:
Condition.acquire(*args) 获取锁 Condition.wait(timeout=None) 等待通知,timeout设置超时时间 Condition.notify(num)唤醒至多指定数目个数的等待的线程,没有等待的线程就没有任何操作 Condition.notify_all() 唤醒所有等待的线程 或者notifyAll()
def ww(c): with c: print('init') c.wait(timeout=5) #设置等待超时时间5 print('end') def xx(c): with c: print('nono') c.notifyAll() #唤醒所有线程 print('start') c.notify(1) #唤醒一个线程 print('21') c=t.Condition() #创建条件 t.Thread(target=ww,args=(c,)).start() t.Thread(target=xx,args=(c,)).start()
这样就可以在等待的时候唤醒函数里唤醒其他函数里所存在的其他线程了。
5.信号量
信号量可以分为有界信号量和无解信号量,下面我们来具体看看他们的用法:
5.1 有界信号量
它不允许使用release超出初始值的范围,否则,抛出ValueError异常。
#构造方法。value为初始信号量。value小于0,抛出ValueError异常 b=t.BoundedSemaphore(value=1) #获取信号量时,计数器减1,即_value的值减少1。如果_value的值为0会变成阻塞状态。获取成功返回True BoundedSemaphore.acquire(blocking=True,timeout=None) #释放信号量,计数器加1。即_value的值加1,超过初始化值会抛出异常ValueError。 BoundedSemaphore.release() #信号量,当前信号量 BoundedSemaphore._value
可以看到了多了个release后报错了。
5.2 无界信号量
它不检查release的上限情况,只是单纯的加减计数器。
可以看到虽然多了个release,但是没有问题,而且信号量的数量不受限制。
6.Event
线程间通信,通过线程设置的信号标志(flag)的False 还是True来进行操作,常见方法有:
event.set() flag设置为True event.clear() flag设置为False event.is_set() flag是否为True,如果 event.isSet()==False将阻塞线程; 设置等待flag为True的时长,None为无限等待。等到返回True,未等到超时则返回False event.wait(timeout=None)
下面通过一个例子具体讲述:
import time e=t.Event() def ff(num): while True: if num<5: e.clear() #清空信号标志 print('清空') if num>=5: e.wait(timeout=1) #等待信号标志为真 e.set() print('启动') if e.isSet(): #如果信号标志为真则清除标志 e.clear() print('停止') if num==10: e.wait(timeout=3) e.clear() print('退出') break num+=1 time.sleep(2) ff(1)
设置延迟后可以看到效果相当明显,我们让他干什么事他就干什么事。
7.local
可以为各个线程创建完全属于它们自己的变量(线程局部变量),而且它们的值都在当前调用它的线程当中,以字典的形式存在。下面我们来看下:
l=t.local() #创建一个线程局部变量 def ff(num): l.x=100 #设置l变量的x方法的值为100 for y in range(num): l.x+=3 #改变值 print(str(l.x)) for y in range(10): t.Thread(target=ff,args=(y,)).start() #开始执行线程
那么,可以将变量的x方法设为全局变量吗?我们来看下:
可以看出他报错了,产生错误的原因是因为这个类中没有属性x,我们可以简单的理解为局部变量就只接受局部。
8.Timer
设置定时计划,可以在规定的时间内反复执行某个方法。他的使用方法是:
t.Timer(num,func,*args,**kwargs) #在指定时间内再次重启程序
下面我们来看下:
def f(): print('start') global t #防止造成线程堆积导致最终程序退出 tt= t.Timer(3, f) tt.start() f()
这样就达到了每三秒执行一次f函数的效果。
总结
以上是一篇文章帶你全面解析不一樣的線程的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

PHP主要是過程式編程,但也支持面向對象編程(OOP);Python支持多種範式,包括OOP、函數式和過程式編程。 PHP適合web開發,Python適用於多種應用,如數據分析和機器學習。

PHP適合網頁開發和快速原型開發,Python適用於數據科學和機器學習。 1.PHP用於動態網頁開發,語法簡單,適合快速開發。 2.Python語法簡潔,適用於多領域,庫生態系統強大。

在 Sublime Text 中運行 Python 代碼,需先安裝 Python 插件,再創建 .py 文件並編寫代碼,最後按 Ctrl B 運行代碼,輸出會在控制台中顯示。

PHP起源於1994年,由RasmusLerdorf開發,最初用於跟踪網站訪問者,逐漸演變為服務器端腳本語言,廣泛應用於網頁開發。 Python由GuidovanRossum於1980年代末開發,1991年首次發布,強調代碼可讀性和簡潔性,適用於科學計算、數據分析等領域。

Python更適合初學者,學習曲線平緩,語法簡潔;JavaScript適合前端開發,學習曲線較陡,語法靈活。 1.Python語法直觀,適用於數據科學和後端開發。 2.JavaScript靈活,廣泛用於前端和服務器端編程。

Golang在性能和可擴展性方面優於Python。 1)Golang的編譯型特性和高效並發模型使其在高並發場景下表現出色。 2)Python作為解釋型語言,執行速度較慢,但通過工具如Cython可優化性能。

在 Visual Studio Code(VSCode)中編寫代碼簡單易行,只需安裝 VSCode、創建項目、選擇語言、創建文件、編寫代碼、保存並運行即可。 VSCode 的優點包括跨平台、免費開源、強大功能、擴展豐富,以及輕量快速。

在 Notepad 中運行 Python 代碼需要安裝 Python 可執行文件和 NppExec 插件。安裝 Python 並為其添加 PATH 後,在 NppExec 插件中配置命令為“python”、參數為“{CURRENT_DIRECTORY}{FILE_NAME}”,即可在 Notepad 中通過快捷鍵“F6”運行 Python 代碼。
