목차
(1) 스레드 간 리소스 공유는 다음과 같습니다. 공유, 키워드 글로벌 구현
Lock
다중 프로세스 프로그래밍에서는 전역 변수를 프로세스 간에 공유하거나 큐에 넣을 수 없습니다.큐를 사용할 수 없습니다.
백엔드 개발 파이썬 튜토리얼 스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대한 소개

스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대한 소개

Oct 26, 2018 pm 05:18 PM
파이썬 멀티스레딩

이 기사에서는 스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대해 소개합니다. 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.

이 기사에서는 스레드 간 리소스 공유와 멀티 스레드 프로그래밍에서 일반적으로 사용되는 잠금 메커니즘을 간략하게 소개합니다.

멀티 스레드 프로그래밍에서는 스레드 간 리소스 공유가 자주 사용됩니다. 일반적으로 사용되는 리소스 공유 방법은 다음과 같습니다.

  • 전역 변수(전역)

  • 큐(큐 가져오기 대기열에서)

일반적으로 사용됩니다. 리소스 공유 잠금 메커니즘:

  • Lock

  • RLock

  • Semphore

  • Condition

(1) 스레드 간 리소스 공유는 다음과 같습니다. 공유, 키워드 글로벌 구현

  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
로그인 후 복사

큐를 사용하여 리소스를 공유하세요. 큐는 스레드로부터 안전합니다.

  1. 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

  1. Lock은 지속적으로 잠금을 획득할 수 없습니다. 그렇지 않으면 잠금 리소스 경쟁으로 인해 교착 상태가 발생할 수 있습니다.

  2. 잠금으로 인해 성능이 저하됩니다.
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

  1. RLock은 지속적으로 잠금을 획득할 수 있지만 잠금을 해제하려면 해당 개수의 릴리스가 필요합니다

  2. 연속적으로 잠금을 획득할 수 있으므로 함수 내에서 잠긴 함수 호출이 가능합니다.
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
로그인 후 복사

조건 조건 변수

  1. 조건 조건 변수는 컨텍스트 관리 프로토콜을 따릅니다. with 문을 사용하여 둘러싸는 블록 기간 동안 관련 잠금을 얻습니다.

  2. 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

  1. 이 클래스는 세마포 객체를 구현합니다. 세마포어는 release() 호출 수에서 acquire() 호출 수를 뺀 값에 초기 값을 더한 값을 나타내는 원자 카운터를 관리합니다. 필요한 경우 acquire() 메서드는 카운터를 음수로 만들지 않고 반환될 수 있을 때까지 차단됩니다. 지정하지 않으면 기본값은 1입니다.

    #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()
    로그인 후 복사
    (3) 다중 프로세스 프로그래밍에 대한 간략한 소개

    다중 프로세스 프로그래밍에서는 전역 변수를 프로세스 간에 공유하거나 큐에 넣을 수 없습니다.큐를 사용할 수 없습니다.

    1. 다중 프로세스 프로그래밍 통신에는 다음이 필요합니다. 큐, 파이프 사용

    2. 프로세스 풀 프로세스 프로그래밍을 사용하는 경우 통신을 위해서는 Manger 인스턴스의 큐를 사용해야 합니다

    위 내용은 스레드 간 리소스 공유와 Python 멀티스레딩에서 일반적으로 사용되는 잠금 메커니즘에 대한 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

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

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

Linux 터미널에서 Python 버전을 볼 때 발생하는 권한 문제를 해결하는 방법은 무엇입니까? Linux 터미널에서 Python 버전을 볼 때 발생하는 권한 문제를 해결하는 방법은 무엇입니까? Apr 01, 2025 pm 05:09 PM

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

10 시간 이내에 프로젝트 및 문제 중심 방법에서 컴퓨터 초보자 프로그래밍 기본 사항을 가르치는 방법? 10 시간 이내에 프로젝트 및 문제 중심 방법에서 컴퓨터 초보자 프로그래밍 기본 사항을 가르치는 방법? Apr 02, 2025 am 07:18 AM

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

한 데이터 프레임의 전체 열을 Python의 다른 구조를 가진 다른 데이터 프레임에 효율적으로 복사하는 방법은 무엇입니까? 한 데이터 프레임의 전체 열을 Python의 다른 구조를 가진 다른 데이터 프레임에 효율적으로 복사하는 방법은 무엇입니까? Apr 01, 2025 pm 11:15 PM

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

중간 독서를 위해 Fiddler를 사용할 때 브라우저에서 감지되는 것을 피하는 방법은 무엇입니까? 중간 독서를 위해 Fiddler를 사용할 때 브라우저에서 감지되는 것을 피하는 방법은 무엇입니까? Apr 02, 2025 am 07:15 AM

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

Uvicorn은 Serving_forever ()없이 HTTP 요청을 어떻게 지속적으로 듣습니까? Uvicorn은 Serving_forever ()없이 HTTP 요청을 어떻게 지속적으로 듣습니까? Apr 01, 2025 pm 10:51 PM

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

문자열을 통해 객체를 동적으로 생성하고 방법을 파이썬으로 호출하는 방법은 무엇입니까? 문자열을 통해 객체를 동적으로 생성하고 방법을 파이썬으로 호출하는 방법은 무엇입니까? Apr 01, 2025 pm 11:18 PM

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

See all articles