스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대한 소개
이 기사에서는 스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대해 소개합니다. 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.
이 기사에서는 스레드 간 리소스 공유와 멀티 스레드 프로그래밍에서 일반적으로 사용되는 잠금 메커니즘을 간략하게 소개합니다.
멀티 스레드 프로그래밍에서는 스레드 간 리소스 공유가 자주 사용됩니다. 일반적으로 사용되는 리소스 공유 방법은 다음과 같습니다.
전역 변수(전역)
큐(큐 가져오기 대기열에서)
일반적으로 사용됩니다. 리소스 공유 잠금 메커니즘:
Lock
RLock
Semphore
Condition
(1) 스레드 간 리소스 공유는 다음과 같습니다. 공유, 키워드 글로벌 구현
- 코드 데모:
from threading import Thread, Lock lock = Lock() total = 0 '''如果不使用lock那么,最后得到的数字不一定为0;同时loack不支持连续多次acquire,如果这样做了的后果是死锁!''' def add(): global total global lock for i in range(1000000): lock.acquire() total += 1 lock.release() def sub(): global total global lock for i in range(1000000): lock.acquire() total -= 1 lock.release() thread1 = Thread(target=add) thread2 = Thread(target=sub) # 将Thread1和2设置为守护线程,主线程完成时,子线程也一起结束 # thread1.setDaemon(True) # thread1.setDaemon(True) # 启动线程 thread1.start() thread2.start() # 阻塞,等待线程1和2完成,如果不使用join,那么主线程完成后,子线程也会自动关闭。 thread1.join() thread2.join() total
큐를 사용하여 리소스를 공유하세요. 큐는 스레드로부터 안전합니다.
from threading import Thread, Lock from queue import Queue def add(q): if q.not_full: q.put(1) def sub(q): if q.not_empty: recv = q.get() print(recv) q.task_done() if __name__ =='__main__': # 设置q最多接收3个任务,Queue是线程安全的,所以不需要Lock qu = Queue(3) thread1 = Thread(target=add, args=(qu,)) thread2 = Thread(target=sub, args=(qu,)) thread1.start() thread2.start() # q队列堵塞,等待所有任务都被处理完。 qu.join()
로그인 후 복사(2) 잠금(Lock/RLock/Condition/Semphore)
Lock
Lock은 지속적으로 잠금을 획득할 수 없습니다. 그렇지 않으면 잠금 리소스 경쟁으로 인해 교착 상태가 발생할 수 있습니다.
잠금으로 인해 성능이 저하됩니다.
from threading import Thread, Lock lock = Lock() total = 0 '''如果不使用lock那么,最后得到的数字不一定为0;同时lock不支持连续多次acquire,如果这样做了的后果是死锁!''' def add(): global total global lock for i in range(1000000): lock.acquire() total += 1 lock.release() def sub(): global total global lock for i in range(1000000): lock.acquire() total -= 1 lock.release() thread1 = Thread(target=add) thread2 = Thread(target=sub) # 将Thread1和2设置为守护线程,主线程完成时,子线程也一起结束 # thread1.setDaemon(True) # thread1.setDaemon(True) # 启动线程 thread1.start() thread2.start() # 阻塞,等待线程1和2完成,如果不使用join,那么主线程完成后,子线程也会自动关闭。 thread1.join() thread2.join() total
RLock
RLock은 지속적으로 잠금을 획득할 수 있지만 잠금을 해제하려면 해당 개수의 릴리스가 필요합니다
연속적으로 잠금을 획득할 수 있으므로 함수 내에서 잠긴 함수 호출이 가능합니다.
from threading import Thread, Lock, RLock lock = RLock() total = 0 def add(): global lock global total # RLock实现连续获取锁,但是需要相应数量的release来释放资源 for i in range(1000000): lock.acquire() lock.acquire() total += 1 lock.release() lock.release() def sub(): global lock global total for i in range(1000000): lock.acquire() total -= 1 lock.release() thread1 = Thread(target=add) thread2 = Thread(target=sub) thread1.start() thread2.start() thread1.join() thread2.join() total
조건 조건 변수
조건 조건 변수는 컨텍스트 관리 프로토콜을 따릅니다. with 문을 사용하여 둘러싸는 블록 기간 동안 관련 잠금을 얻습니다.
wait() 메서드는 잠금을 해제한 다음 다른 스레드가 inform() 또는 inform_all()을 호출하여 잠금을 깨울 때까지 차단합니다. 일단 깨어나면 wait()는 잠금을 다시 획득하고 반환합니다. 시간 초과를 지정할 수도 있습니다.
먼저 차단 대기 상태에서 신호를 받기 위해 대기 함수를 시작한 다음, 신호를 보내기 위해 알림 함수를 시작합니다
from threading import Thread, Condition '''聊天 Peaple1 : How are you? Peaple2 : I`m fine, thank you! Peaple1 : What`s your job? Peaple2 : My job is teacher. ''' def Peaple1(condition): with condition: print('Peaple1 : ', 'How are you?') condition.notify() condition.wait() print('Peaple1 : ', 'What`s your job?') condition.notify() condition.wait() def Peaple2(condition): with condition: condition.wait() print('Peaple2 : ', 'I`m fine, thank you!') condition.notify() condition.wait() print('Peaple2 : ', 'My job is teacher.') condition.notify() if __name__ == '__main__': cond = Condition() thread1 = Thread(target=Peaple1, args=(cond,)) thread2 = Thread(target=Peaple2, args=(cond,)) # 此处thread2要比thread1提前启动,因为notify必须要有wait接收;如果先启动thread1,没有wait接收notify信号,那么将会死锁。 thread2.start() thread1.start() # thread1.join() # thread2.join()
Semphore
이 클래스는 세마포 객체를 구현합니다. 세마포어는 release() 호출 수에서 acquire() 호출 수를 뺀 값에 초기 값을 더한 값을 나타내는 원자 카운터를 관리합니다. 필요한 경우 acquire() 메서드는 카운터를 음수로 만들지 않고 반환될 수 있을 때까지 차단됩니다. 지정하지 않으면 기본값은 1입니다.
(3) 다중 프로세스 프로그래밍에 대한 간략한 소개#Semaphore 是用于控制进入数量的锁 #文件, 读、写, 写一般只是用于一个线程写,读可以允许有多个 import threading import time class HtmlSpider(threading.Thread): def __init__(self, url, sem): super().__init__() self.url = url self.sem = sem def run(self): time.sleep(2) print("Download {html} success\n".format(html=self.url)) self.sem.release() class UrlProducer(threading.Thread): def __init__(self, sem): super().__init__() self.sem = sem def run(self): for i in range(20): self.sem.acquire() html_thread = HtmlSpider("https://www.baidu.com/{}".format(i), self.sem) html_thread.start() if __name__ == "__main__": # 控制锁的数量, 每次同时会有3个线程获得锁,然后输出 sem = threading.Semaphore(3) url_producer = UrlProducer(sem) url_producer.start()
로그인 후 복사다중 프로세스 프로그래밍에서는 전역 변수를 프로세스 간에 공유하거나 큐에 넣을 수 없습니다.큐를 사용할 수 없습니다.
다중 프로세스 프로그래밍 통신에는 다음이 필요합니다. 큐, 파이프 사용
프로세스 풀 프로세스 프로그래밍을 사용하는 경우 통신을 위해서는 Manger 인스턴스의 큐를 사용해야 합니다
위 내용은 스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대한 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











Linux 터미널에서 Python 버전을 보려고 할 때 Linux 터미널에서 Python 버전을 볼 때 권한 문제에 대한 솔루션 ... Python을 입력하십시오 ...

10 시간 이내에 컴퓨터 초보자 프로그래밍 기본 사항을 가르치는 방법은 무엇입니까? 컴퓨터 초보자에게 프로그래밍 지식을 가르치는 데 10 시간 밖에 걸리지 않는다면 무엇을 가르치기로 선택 하시겠습니까?

Python의 Pandas 라이브러리를 사용할 때는 구조가 다른 두 데이터 프레임 사이에서 전체 열을 복사하는 방법이 일반적인 문제입니다. 두 개의 dats가 있다고 가정 해

Fiddlerevery Where를 사용할 때 Man-in-the-Middle Reading에 Fiddlereverywhere를 사용할 때 감지되는 방법 ...

Uvicorn은 HTTP 요청을 어떻게 지속적으로 듣습니까? Uvicorn은 ASGI를 기반으로 한 가벼운 웹 서버입니다. 핵심 기능 중 하나는 HTTP 요청을 듣고 진행하는 것입니다 ...

파이썬에서 문자열을 통해 객체를 동적으로 생성하고 메소드를 호출하는 방법은 무엇입니까? 특히 구성 또는 실행 해야하는 경우 일반적인 프로그래밍 요구 사항입니다.

Linux 터미널에서 Python 사용 ...
