> 백엔드 개발 > 파이썬 튜토리얼 > Python 다중 프로세스에서 병렬 처리를 완료하는 방법 소개

Python 다중 프로세스에서 병렬 처리를 완료하는 방법 소개

巴扎黑
풀어 주다: 2017-08-15 15:08:01
원래의
1596명이 탐색했습니다.

이 글은 주로 Python에서 병렬 처리를 수행하기 위해 여러 프로세스를 사용하는 방법에 대한 요약을 소개합니다. 관심 있는 친구들이 참고할 수 있습니다.

프로세스와 스레드는 컴퓨터 소프트웨어 분야에서 매우 중요한 개념입니다. 스레드는 다르며 밀접하게 관련되어 있습니다.

1. 정의

프로세스는 특정 데이터 수집에 대한 실행 활동과 관련하여 독립적입니다. 스레드는 시스템의 자원 할당 및 스케줄링을 위한 단위로, CPU 스케줄링 및 디스패치의 기본 단위이며, 기본적으로 스레드 자체는 독립적으로 실행될 수 없습니다. 자체 시스템 리소스에는 실행에 필수적인 소수의 리소스(예: 프로그램 카운터, 레지스터 세트 및 스택)만 있지만 프로세스가 소유한 모든 리소스를 동일한 프로세스에 속한 다른 스레드와 공유할 수 있습니다.

2. 관계

스레드는 다른 스레드를 생성하고 취소할 수 있으며, 동일한 프로세스의 여러 스레드가 동시에 실행될 수 있습니다.


프로세스와 관련하여 스레드는 공유할 수 있습니다. 동일한 프로세스의 다른 스레드와 데이터를 공유하지만 자체 스택 공간과 독립적인 실행 순서를 갖습니다.

3. 차이점

프로세스와 스레드의 주요 차이점은 운영 체제 리소스 관리 방법이 다르다는 것입니다. 프로세스는 독립적인 주소 공간을 갖습니다. 프로세스가 충돌한 후에는 보호 모드의 다른 프로세스에 영향을 주지 않으며 스레드는 프로세스의 다른 실행 경로일 뿐입니다. 스레드에는 자체 스택과 지역 변수가 있지만 스레드 사이에 별도의 주소 공간이 없습니다. 따라서 하나의 스레드가 죽으면 전체 프로세스가 죽는 것과 같습니다. 따라서 다중 스레드 프로그램은 다중 스레드 프로그램보다 강력합니다. 프로세스를 전환하면 리소스가 많이 소모되고 효율성이 떨어집니다.

그러나 특정 변수의 동시 실행 및 공유가 필요한 일부 동시 작업의 경우 프로세스가 아닌 스레드만 사용할 수 있습니다.

1) 간단히 말해서 프로그램에는 하나 이상의 프로세스가 있고, 프로세스에는 하나 이상의 스레드가 있습니다.


2) 스레드의 분할 규모는 프로세스의 분할 규모보다 작으므로 멀티 스레드 프로그램은 높은 동시성.


3) 또한 프로세스는 실행 중에 독립된 메모리 단위를 가지며 여러 스레드가 메모리를 공유하므로 프로그램의 실행 효율성이 크게 향상됩니다.


4) 실행 중에는 여전히 스레드와 프로세스 사이에 차이가 있습니다. 각각의 독립 스레드에는 프로그램 실행을 위한 진입점, 순차적 실행 시퀀스 및 프로그램 종료점이 있습니다. 그러나 스레드는 독립적으로 실행될 수 없으며 응용 프로그램 내에 존재해야 하며 응용 프로그램은 다중 스레드 실행 제어를 제공합니다.


5) 논리적인 관점에서 볼 때 멀티스레딩의 의미는 애플리케이션에서 여러 실행 부분을 동시에 실행할 수 있다는 것입니다. 그러나 운영 체제는 프로세스 스케줄링 및 관리, 자원 할당을 구현하기 위해 다중 스레드를 다중 독립 응용 프로그램으로 간주하지 않습니다. 이것이 프로세스와 스레드의 중요한 차이점입니다.


4. 장점과 단점

스레드와 프로세스에는 사용 시 고유한 장점과 단점이 있습니다. 스레드 실행 오버헤드는 작지만 리소스 관리 및 보호에 도움이 되지 않는 반면 프로세스는 정반대입니다. 동시에 스레드는 SMP 시스템에서 실행하기에 적합하며 프로세스는 시스템 간에 마이그레이션될 수 있습니다.

이 기사에서는 주로 Python의 다중 프로세스 적용에 대해 설명합니다.

Unix/Linux 운영 체제는 매우 특별한 fork() 시스템 호출을 제공합니다. 일반적인 함수 호출은 한 번 호출하고 한 번 반환하지만, fork()는 한 번 호출하고 두 번 반환합니다. 운영 체제가 자동으로 현재 프로세스(상위 프로세스라고 함)의 복사본을 만들고(자식 프로세스라고 함) 상위 프로세스와 하위 프로세스 내에서 반환됩니다.

하위 프로세스는 항상 0을 반환하고 상위 프로세스는 하위 프로세스의 ID를 반환합니다. 그 이유는 부모 프로세스가 많은 자식 프로세스를 분기할 수 있으므로 부모 프로세스는 각 자식 프로세스의 ID를 기록해야 하고 자식 프로세스는 부모 프로세스의 ID를 가져오기 위해 getpid()만 호출하면 되기 때문입니다.

Python의 os 모듈은 Python 프로그램에서 하위 프로세스를 쉽게 생성할 수 있는 fork를 포함한 일반적인 시스템 호출을 캡슐화합니다.


import os

print('Process (%s) start...' % os.getpid())
# Only works on Unix/Linux/Mac:
pid = os.fork()
if pid == 0:
  print('I am child process (%s) and my parent is %s.' % (os.getpid(), os.getppid()))
else:
  print('I (%s) just created a child process (%s).' % (os.getpid(), pid))
로그인 후 복사

실행 결과는 다음과 같습니다.

Process (876) start...

I ( 876) 방금 자식 프로세스(877)를 만들었습니다.
저는 자식 프로세스(877)이고 부모는 876입니다.

Windows에는 포크 호출이 없으므로 위 코드는 Windows에서 실행될 수 없습니다.


포크 호출을 사용하면 프로세스가 새 작업을 수신할 때 하위 프로세스를 복사하여 새 작업을 처리할 수 있습니다. 일반적인

Apache 서버에는 새 http 요청이 있을 때마다 포트를 수신하는 상위 프로세스가 있습니다. 새로운 http 요청을 처리하기 위해 하위 프로세스를 포크합니다.

multiprocessing

다중 프로세스 서비스 프로그램을 작성하려는 경우 Unix/linux는 의심할 여지 없이 올바른 선택입니다. Windows에는 포크 호출이 없으므로 Windows에서 Python으로 다중 프로세스 프로그램을 작성하는 것이 불가능합니까?

Python은 크로스 플랫폼이므로 자연스럽게 크로스 플랫폼 다중 프로세스 지원을 제공해야 합니다. 다중 처리 모듈은 다중 프로세스 모듈의 크로스 플랫폼 버전입니다.

멀티프로세싱 모듈은 프로세스 개체를 나타내는 Process 클래스를 제공합니다. 다음 예에서는 하위 프로세스를 시작하고 끝날 때까지 기다리는 방법을 보여줍니다.


from multiprocessing import Process
import os

# 子进程要执行的代码
def run_proc(name):
  print('Run child process %s (%s)...' % (name, os.getpid()))

if __name__=='__main__':
  print('Parent process %s.' % os.getpid())
  p = Process(target=run_proc, args=('test',))
  print('Child process will start.')
  p.start()
  p.join()
  print('Child process end.')
로그인 후 복사

创建子进程时,只需要传入一个执行函数和函数的参数,创建一个Process实例,用start()方法启动,这样创建进程比fork()还要简单。

join()方法可以等待子进程结束后再继续往下运行,通常用于进程间的同步。

Pool

如果要启动大量的子进程,可以用进程池的方式批量创建子进程:


from multiprocessing import Pool
import os, time, random

def long_time_task(name):
  print('Run task %s (%s)...' % (name, os.getpid()))
  start = time.time()
  time.sleep(random.random() * 3)
  end = time.time()
  print('Task %s runs %0.2f seconds.' % (name, (end - start)))

if __name__=='__main__':
  print('Parent process %s.' % os.getpid())
  p = Pool(4)
  for i in range(5):
    p.apply_async(long_time_task, args=(i,))
  print('Waiting for all subprocesses done...')
  p.close()
  p.join()
  print('All subprocesses done.')
로그인 후 복사

执行结果如下:

Parent process 669.
Waiting for all subprocesses done...
Run task 0 (671)...
Run task 1 (672)...
Run task 2 (673)...
Run task 3 (674)...
Task 2 runs 0.14 seconds.
Run task 4 (673)...
Task 1 runs 0.27 seconds.
Task 3 runs 0.86 seconds.
Task 0 runs 1.41 seconds.
Task 4 runs 1.91 seconds.
All subprocesses done.

代码解读:

对Pool对象调用join()方法会等待所有子进程执行完毕,调用join()之前必须先调用close(),调用close()之后就不能继续添加新的Process了。

请注意输出的结果,task 0,1,2,3是立刻执行的,而task 4要等待前面某个task完成后才执行,这是因为Pool的默认大小在我的电脑上是4,因此,最多同时执行4个进程。这是Pool有意设计的限制,并不是操作系统的限制。如果改成:


p = Pool(5)
로그인 후 복사

就可以同时跑5个进程。

由于Pool的默认大小是CPU的核数,如果你不幸拥有8核CPU,你要提交至少9个子进程才能看到上面的等待效果。

子进程

很多时候,子进程并不是自身,而是一个外部进程。我们创建了子进程后,还需要控制子进程的输入和输出。

subprocess模块可以让我们非常方便地启动一个子进程,然后控制其输入和输出。

下面的例子演示了如何在Python代码中运行命令nslookup www.python.org,这和命令行直接运行的效果是一样的:


import subprocess

print('$ nslookup www.python.org')
r = subprocess.call(['nslookup', 'www.python.org'])
print('Exit code:', r)
로그인 후 복사

运行结果:

$ nslookup www.python.org
Server: 192.168.19.4
Address: 192.168.19.4#53
Non-authoritative answer:
www.python.org canonical name = python.map.fastly.net.
Name: python.map.fastly.net
Address: 199.27.79.223
Exit code: 0

如果子进程还需要输入,则可以通过communicate()方法输入:


import subprocess

print('$ nslookup')
p = subprocess.Popen(['nslookup'], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
output, err = p.communicate(b'set q=mx\npython.org\nexit\n')
print(output.decode('utf-8'))
print('Exit code:', p.returncode)
로그인 후 복사

上面的代码相当于在命令行执行命令nslookup,然后手动输入:

set q=mx
python.org
exit

进程间通信

Process之间肯定是需要通信的,操作系统提供了很多机制来实现进程间的通信。Python的multiprocessing模块包装了底层的机制,提供了Queue、Pipes等多种方式来交换数据。

我们以Queue为例,在父进程中创建两个子进程,一个往Queue里写数据,一个从Queue里读数据:


from multiprocessing import Process, Queue
import os, time, random

# 写数据进程执行的代码:
def write(q):
  print('Process to write: %s' % os.getpid())
  for value in ['A', 'B', 'C']:
    print('Put %s to queue...' % value)
    q.put(value)
    time.sleep(random.random())

# 读数据进程执行的代码:
def read(q):
  print('Process to read: %s' % os.getpid())
  while True:
    value = q.get(True)
    print('Get %s from queue.' % value)

if __name__=='__main__':
  # 父进程创建Queue,并传给各个子进程:
  q = Queue()
  pw = Process(target=write, args=(q,))
  pr = Process(target=read, args=(q,))
  # 启动子进程pw,写入:
  pw.start()
  # 启动子进程pr,读取:
  pr.start()
  # 等待pw结束:
  pw.join()
  # pr进程里是死循环,无法等待其结束,只能强行终止:
  pr.terminate()
로그인 후 복사

运行结果如下:

Process to write: 50563
Put A to queue...
Process to read: 50564
Get A from queue.
Put B to queue...
Get B from queue.
Put C to queue...
Get C from queue.

在Unix/Linux下,multiprocessing模块封装了fork()调用,使我们不需要关注fork()的细节。由于Windows没有fork调用,因此,multiprocessing需要“模拟”出fork的效果,父进程所有Python对象都必须通过pickle序列化再传到子进程去,所有,如果multiprocessing在Windows下调用失败了,要先考虑是不是pickle失败了。

小结

在Unix/Linux下,可以使用fork()调用实现多进程。

要实现跨平台的多进程,可以使用multiprocessing模块。

进程间通信是通过Queue、Pipes等实现的。

多线程

多任务可以由多进程完成,也可以由一个进程内的多线程完成。进程是由若干线程组成的,一个进程至少有一个线程。

由于线程是操作系统直接支持的执行单元,因此,高级语言通常都内置多线程的支持,Python也不例外,并且,Python的线程是真正的Posix Thread,而不是模拟出来的线程。

Python的标准库提供了两个模块:_thread 和 threading,_thread是低级模块,threading是高级模块,对_thread进行了封装。绝大多数情况下,我们只需要使用threading这个高级模块。

启动一个线程就是把一个函数传入并创建Thread实例,然后调用start()开始执行:


import time, threading

# 新线程执行的代码:
def loop():
  print('thread %s is running...' % threading.current_thread().name)
  n = 0
  while n < 5:
    n = n + 1
    print(&#39;thread %s >>> %s&#39; % (threading.current_thread().name, n))
    time.sleep(1)
  print(&#39;thread %s ended.&#39; % threading.current_thread().name)

print(&#39;thread %s is running...&#39; % threading.current_thread().name)
t = threading.Thread(target=loop, name=&#39;LoopThread&#39;)
t.start()
t.join()
print(&#39;thread %s ended.&#39; % threading.current_thread().name)
thread MainThread is running...
thread LoopThread is running...
thread LoopThread >>> 1
thread LoopThread >>> 2
thread LoopThread >>> 3
thread LoopThread >>> 4
thread LoopThread >>> 5
thread LoopThread ended.
thread MainThread ended.
로그인 후 복사

由于任何进程默认就会启动一个线程,我们把该线程称为主线程,主线程又可以启动新的线程,Python的threading模块有个current_thread()函数,它永远返回当前线程的实例。主线程实例的名字叫MainThread,子线程的名字在创建时指定,我们用LoopThread命名子线程。名字仅仅在打印时用来显示,完全没有其他意义,如果不起名字Python就自动给线程命名为Thread-1,Thread-2……

Lock

多线程和多进程最大的不同在于,多进程中,同一个变量,各自有一份拷贝存在于每个进程中,互不影响,而多线程中,所有变量都由所有线程共享,所以,任何一个变量都可以被任何一个线程修改,因此,线程之间共享数据最大的危险在于多个线程同时改一个变量,把内容给改乱了。

来看看多个线程同时操作一个变量怎么把内容给改乱了:


import time, threading
# 假定这是你的银行存款:
balance = 0
def change_it(n):
  # 先存后取,结果应该为0:
  global balance
  balance = balance + n
  balance = balance - n
def run_thread(n):
  for i in range(100000):
    change_it(n)
t1 = threading.Thread(target=run_thread, args=(5,))
t2 = threading.Thread(target=run_thread, args=(8,))
t1.start()
t2.start()
t1.join()
t2.join()
print(balance)
로그인 후 복사

我们定义了一个共享变量balance,初始值为0,并且启动两个线程,先存后取,理论上结果应该为0,但是,由于线程的调度是由操作系统决定的,当t1、t2交替执行时,只要循环次数足够多,balance的结果就不一定是0了。

原因是因为高级语言的一条语句在CPU执行时是若干条语句,即使一个简单的计算:


balance = balance + n
로그인 후 복사

也分两步:

  1. 计算balance + n,存入临时变量中;

  2. 将临时变量的值赋给balance。

也就是可以看成:


x = balance + n
balance = x
로그인 후 복사

数据错误的原因:是因为修改balance需要多条语句,而执行这几条语句时,线程可能中断,从而导致多个线程把同一个对象的内容改乱了。

两个线程同时一存一取,就可能导致余额不对,你肯定不希望你的银行存款莫名其妙地变成了负数,所以,我们必须确保一个线程在修改balance的时候,别的线程一定不能改。

如果我们要确保balance计算正确,就要给change_it()上一把锁,当某个线程开始执行change_it()时,我们说,该线程因为获得了锁,因此其他线程不能同时执行change_it(),只能等待,直到锁被释放后,获得该锁以后才能改。由于锁只有一个,无论多少线程,同一时刻最多只有一个线程持有该锁,所以,不会造成修改的冲突。创建一个锁就是通过threading.Lock()来实现:


balance = 0
lock = threading.Lock()

def run_thread(n):
  for i in range(100000):
    # 先要获取锁:
    lock.acquire()
    try:
      # 放心地改吧:
      change_it(n)
    finally:
      # 改完了一定要释放锁:
      lock.release()
로그인 후 복사

当多个线程同时执行lock.acquire()时,只有一个线程能成功地获取锁,然后继续执行代码,其他线程就继续等待直到获得锁为止。

获得锁的线程用完后一定要释放锁,否则那些苦苦等待锁的线程将永远等待下去,成为死线程。所以我们用try...finally来确保锁一定会被释放。

锁的好处就是确保了某段关键代码只能由一个线程从头到尾完整地执行,坏处当然也很多,首先是阻止了多线程并发执行,包含锁的某段代码实际上只能以单线程模式执行,效率就大大地下降了。其次,由于可以存在多个锁,不同的线程持有不同的锁,并试图获取对方持有的锁时,可能会造成死锁,导致多个线程全部挂起,既不能执行,也无法结束,只能靠操作系统强制终止。

多核CPU

如果你不幸拥有一个多核CPU,你肯定在想,多核应该可以同时执行多个线程。

如果写一个死循环的话,会出现什么情况呢?

打开Mac OS X的Activity Monitor,或者Windows的Task Manager,都可以监控某个进程的CPU使用率。

我们可以监控到一个死循环线程会100%占用一个CPU。如果有两个死循环线程,在多核CPU中,可以监控到会占用200%的CPU,也就是占用两个CPU核心。要想把N核CPU的核心全部跑满,就必须启动N个死循环线程。

试试用Python写个死循环:


import threading, multiprocessing

def loop():
  x = 0
  while True:
    x = x ^ 1

for i in range(multiprocessing.cpu_count()):
  t = threading.Thread(target=loop)
  t.start()
로그인 후 복사

启动与CPU核心数量相同的N个线程,在4核CPU上可以监控到CPU占用率仅有102%,也就是仅使用了一核。

但是用C、C++或Java来改写相同的死循环,直接可以把全部核心跑满,4核就跑到400%,8核就跑到800%,为什么Python不行呢?

因为Python的线程虽然是真正的线程,但解释器执行代码时,有一个GIL锁:Global Interpreter Lock,任何Python线程执行前,必须先获得GIL锁,然后,每执行100条字节码,解释器就自动释放GIL锁,让别的线程有机会执行。这个GIL全局锁实际上把所有线程的执行代码都给上了锁,所以,多线程在Python中只能交替执行,即使100个线程跑在100核CPU上,也只能用到1个核。

GIL是Python解释器设计的历史遗留问题,通常我们用的解释器是官方实现的CPython,要真正利用多核,除非重写一个不带GIL的解释器。

所以,在Python中,可以使用多线程,但不要指望能有效利用多核。如果一定要通过多线程利用多核,那只能通过C扩展来实现,不过这样就失去了Python简单易用的特点。

不过,也不用过于担心,Python虽然不能利用多线程实现多核任务,但可以通过多进程实现多核任务。多个Python进程有各自独立的GIL锁,互不影响。

多线程编程,模型复杂,容易发生冲突,必须用锁加以隔离,同时,又要小心死锁的发生。

Python解释器由于设计时有GIL全局锁,导致了多线程无法利用多核。

ThreadLocal

在多线程环境下,每个线程都有自己的数据。一个线程使用自己的局部变量比使用全局变量好,因为局部变量只有线程自己能看见,不会影响其他线程,而全局变量的修改必须加锁。但是局部变量也有问题,就是在函数调用的时候,传递起来很麻烦:


import threading

# 创建全局ThreadLocal对象:
local_school = threading.local()

def process_student():
  # 获取当前线程关联的student:
  std = local_school.student
  print(&#39;Hello, %s (in %s)&#39; % (std, threading.current_thread().name))

def process_thread(name):
  # 绑定ThreadLocal的student:
  local_school.student = name
  process_student()

t1 = threading.Thread(target= process_thread, args=(&#39;Alice&#39;,), name=&#39;Thread-A&#39;)
t2 = threading.Thread(target= process_thread, args=(&#39;Bob&#39;,), name=&#39;Thread-B&#39;)
t1.start()
t2.start()
t1.join()
t2.join()
로그인 후 복사

全局变量local_school就是一个ThreadLocal对象,每个Thread对它都可以读写student属性,但互不影响。你可以把local_school看成全局变量,但每个属性如local_school.student都是线程的局部变量,可以任意读写而互不干扰,也不用管理锁的问题,ThreadLocal内部会处理。

可以理解为全局变量local_school是一个dict,不但可以用local_school.student,还可以绑定其他变量,如local_school.teacher等等。

ThreadLocal最常用的地方就是为每个线程绑定一个数据库连接,HTTP请求,用户身份信息等,这样一个线程的所有调用到的处理函数都可以非常方便地访问这些资源。

一个ThreadLocal变量虽然是全局变量,但每个线程都只能读写自己线程的独立副本,互不干扰。ThreadLocal解决了参数在一个线程中各个函数之间互相传递的问题。

进程 vs. 线程

我们介绍了多进程和多线程,这是实现多任务最常用的两种方式。现在,我们来讨论一下这两种方式的优缺点。

首先,要实现多任务,通常我们会设计Master-Worker模式,Master负责分配任务,Worker负责执行任务,因此,多任务环境下,通常是一个Master,多个Worker。

如果用多进程实现Master-Worker,主进程就是Master,其他进程就是Worker。

如果用多线程实现Master-Worker,主线程就是Master,其他线程就是Worker。

多进程模式最大的优点就是稳定性高,因为一个子进程崩溃了,不会影响主进程和其他子进程。(当然主进程挂了所有进程就全挂了,但是Master进程只负责分配任务,挂掉的概率低)著名的Apache最早就是采用多进程模式。

多进程模式的缺点是创建进程的代价大,在Unix/Linux系统下,用fork调用还行,在Windows下创建进程开销巨大。另外,操作系统能同时运行的进程数也是有限的,在内存和CPU的限制下,如果有几千个进程同时运行,操作系统连调度都会成问题。

多线程模式通常比多进程快一点,但是也快不到哪去,而且,多线程模式致命的缺点就是任何一个线程挂掉都可能直接造成整个进程崩溃,因为所有线程共享进程的内存。在Windows上,如果一个线程执行的代码出了问题,你经常可以看到这样的提示:“该程序执行了非法操作,即将关闭”,其实往往是某个线程出了问题,但是操作系统会强制结束整个进程。

在Windows下,多线程的效率比多进程要高,所以微软的IIS服务器默认采用多线程模式。由于多线程存在稳定性的问题,IIS的稳定性就不如Apache。为了缓解这个问题,IIS和Apache现在又有多进程+多线程的混合模式,真是把问题越搞越复杂。

线程切换

无论是多进程还是多线程,只要数量一多,效率肯定上不去,为什么呢?

我们打个比方,假设你不幸正在准备中考,每天晚上需要做语文、数学、英语、物理、化学这5科的作业,每项作业耗时1小时。

如果你先花1小时做语文作业,做完了,再花1小时做数学作业,这样,依次全部做完,一共花5小时,这种方式称为单任务模型,或者批处理任务模型。

멀티태스킹 모델로 전환할 계획이라면 전환 속도가 빠르면 중국어 1분, 수학 숙제 1분, 영어로 전환할 수 있습니다. 이 방법은 싱글코어 CPU와 동일합니다. 멀티태스킹도 유치원생의 입장에서 보면 5과목을 동시에 숙제하는 셈입니다.

단, 숙제를 전환하려면 비용이 발생합니다. 예를 들어 중국어에서 수학으로 전환하는 경우 먼저 테이블 위에 있는 중국어 책과 펜을 치운 다음(이를 장면 저장이라고 함) 수학 교과서를 펼쳐야 합니다. 수학 숙제를 시작하기 전에 나침반과 자(이것을 새로운 환경 준비라고 함)를 찾으세요. 프로세스나 스레드를 전환할 때에도 운영 체제는 동일합니다. 먼저 현재 실행 환경(CPU 레지스터 상태, 메모리 페이지 등)을 저장한 다음 새 작업을 위한 실행 환경을 준비해야 합니다(마지막 레지스터 상태 복원, 실행이 시작되기 전에 메모리 페이지 전환 등). 이 전환 프로세스는 빠르지만 시간도 걸립니다. 동시에 수천 개의 작업이 실행되는 경우 운영 체제는 주로 작업을 전환하느라 바빠서 작업을 수행할 시간이 많지 않습니다. 이 상황에서 가장 일반적인 상황은 하드 디스크에서 경고음이 심하게 울리는 것입니다. 창을 클릭하면 응답이 발생하고 시스템은 애니메이션이 정지된 상태가 됩니다.

그래서 멀티태스킹 횟수가 한계에 도달하면 시스템의 모든 리소스가 소모되어 결과적으로 효율성이 급격히 떨어지고 모든 작업이 제대로 완료되지 않습니다.

계산 집약적 vs. IO 집약적

멀티태스킹 사용 여부에 대한 두 번째 고려 사항은 작업 유형입니다. 작업을 컴퓨팅 집약적인 작업과 IO 집약적인 작업으로 나눌 수 있습니다.

컴퓨팅 집약적인 작업은 파이 계산, 비디오의 고화질 디코딩 등과 같이 많은 양의 계산이 필요하고 CPU 리소스를 소비하는 것이 특징이며 모두 CPU의 컴퓨팅 성능에 의존합니다. 이러한 컴퓨팅 집약적인 작업은 멀티 태스킹으로도 완료할 수 있지만 작업이 많을수록 작업 전환에 더 많은 시간이 소요되고 작업 실행 시 CPU의 효율성이 낮아집니다. CPU 사용, 컴퓨팅 집약적인 작업 동시 작업 수는 CPU 코어 수와 동일해야 합니다.

컴퓨팅 집약적인 작업은 주로 CPU 리소스를 소비하므로 코드 실행 효율성이 중요합니다. Python과 같은 스크립팅 언어는 매우 비효율적으로 실행되며 계산 집약적인 작업에는 전혀 적합하지 않습니다. 계산 집약적인 작업의 경우 C 언어로 작성하는 것이 좋습니다.

두 번째 유형의 작업은 IO 집약적입니다. 네트워크 및 디스크 IO와 관련된 작업은 모두 IO 집약적 작업입니다. 이 유형의 작업의 특징은 CPU 소비가 매우 적고 대부분의 작업 시간이 대기 중이라는 것입니다. (IO 속도가 CPU 및 메모리 속도보다 훨씬 낮기 때문입니다.) IO 집약적인 작업의 경우 작업이 많을수록 CPU 효율성은 높아지지만 한계가 있습니다. 가장 일반적인 작업은 웹 애플리케이션과 같은 IO 집약적인 작업입니다.

IO 집약적인 작업을 실행하는 동안 99%의 시간이 IO에 소요되고 CPU에 소요되는 시간은 매우 적습니다. 따라서 Python과 같이 매우 느린 스크립트 언어를 대체하려면 매우 빠른 C 언어를 사용하세요. 운영 효율성을 전혀 향상시킬 수 없습니다. IO 집약적인 작업의 경우 가장 적합한 언어는 개발 효율성이 가장 높은(코드 양이 가장 적은) 언어가 첫 번째 선택이고 C 언어는 최악입니다.

비동기 IO

CPU와 IO의 엄청난 속도 차이를 고려하면 작업은 실행 중에 IO 작업을 기다리는 데 대부분의 시간을 소비하므로 단일 프로세스 단일 스레드 모델은 다른 작업을 실행할 수 없게 만듭니다. 따라서 다중 작업의 동시 실행을 지원하려면 다중 프로세스 모델 또는 다중 스레드 모델이 필요합니다.

최신 운영 체제는 IO 작업을 크게 개선했습니다. 가장 큰 특징은 비동기 IO를 지원한다는 것입니다. 운영 체제에서 제공하는 비동기 IO 지원을 최대한 활용하면 단일 프로세스 단일 스레드 모델을 사용하여 여러 작업을 수행할 수 있습니다. 이 새로운 모델을 Nginx를 지원하는 웹 서버입니다. 비동기식 IO. 단일 코어 CPU에서 실행되며 단일 프로세스 모델을 사용하면 멀티태스킹을 효율적으로 지원할 수 있습니다. 멀티 코어 CPU에서는 멀티 코어 CPU를 최대한 활용하면서 여러 프로세스(CPU 코어 수와 동일)를 실행할 수 있습니다. 시스템의 전체 프로세스 수는 매우 제한되어 있으므로 운영 체제 스케줄링은 매우 효율적입니다. 멀티태스킹을 구현하기 위해 비동기식 IO 프로그래밍 모델을 사용하는 것이 주요 추세입니다.

Python 언어에 해당하는 단일 프로세스 비동기 프로그래밍 모델을 코루틴이라고 합니다. 이벤트 기반의 효율적인 다중 작업 프로그램을 작성할 수 있습니다. 코루틴을 작성하는 방법은 나중에 논의하겠습니다.

분산 프로세스

스레드와 프로세스 중에서는 프로세스가 더 안정적이고 프로세스가 여러 머신에 배포될 수 있는 반면 스레드는 최대 동일한 머신의 여러 CPU에만 배포될 수 있으므로 선호되어야 합니다.

Python의 다중 처리 모듈은 여러 프로세스를 지원할 뿐만 아니라 관리자 하위 모듈도 여러 프로세스를 여러 컴퓨터에 배포하는 것을 지원합니다. 서비스 프로세스는 네트워크 통신에 의존하여 작업을 여러 다른 프로세스에 배포하는 스케줄러 역할을 할 수 있습니다. 관리자 모듈은 잘 캡슐화되어 있으므로 네트워크 통신의 세부 사항을 몰라도 분산 다중 프로세스 프로그램을 쉽게 작성할 수 있습니다.

예: 동일한 시스템에서 실행되는 대기열을 통해 통신하는 다중 프로세스 프로그램이 이미 있는 경우 작업을 처리하는 프로세스의 작업량이 많아 작업을 보내는 프로세스와 머신의 두 머신에 작업을 처리하는 프로세스입니다. 분산 프로세스를 사용하여 구현하는 방법은 무엇입니까?

原有的Queue可以继续使用,但是,通过managers模块把Queue通过网络暴露出去,就可以让其他机器的进程访问Queue了。

我们先看服务进程,服务进程负责启动Queue,把Queue注册到网络上,然后往Queue里面写入任务:


import random, time, queue
from multiprocessing.managers import BaseManager

# 发送任务的队列:
task_queue = queue.Queue()
# 接收结果的队列:
result_queue = queue.Queue()

# 从BaseManager继承的QueueManager:
class QueueManager(BaseManager):
  pass

# 把两个Queue都注册到网络上, callable参数关联了Queue对象:
QueueManager.register(&#39;get_task_queue&#39;, callable=lambda: task_queue)
QueueManager.register(&#39;get_result_queue&#39;, callable=lambda: result_queue)
# 绑定端口5000, 设置验证码&#39;abc&#39;:
manager = QueueManager(address=(&#39;&#39;, 5000), authkey=b&#39;abc&#39;)
# 启动Queue:
manager.start()
# 获得通过网络访问的Queue对象:
task = manager.get_task_queue()
result = manager.get_result_queue()
# 放几个任务进去:
for i in range(10):
  n = random.randint(0, 10000)
  print(&#39;Put task %d...&#39; % n)
  task.put(n)
# 从result队列读取结果:
print(&#39;Try get results...&#39;)
for i in range(10):
  r = result.get(timeout=10)
  print(&#39;Result: %s&#39; % r)
# 关闭:
manager.shutdown()
print(&#39;master exit.&#39;)
로그인 후 복사

当我们在一台机器上写多进程程序时,创建的Queue可以直接拿来用,但是,在分布式多进程环境下,添加任务到Queue不可以直接对原始的task_queue进行操作,那样就绕过了QueueManager的封装,必须通过manager.get_task_queue()获得的Queue接口添加。

然后,在另一台机器上启动任务进程(本机上启动也可以):


import time, sys, queue
from multiprocessing.managers import BaseManager

# 创建类似的QueueManager:
class QueueManager(BaseManager):
  pass

# 由于这个QueueManager只从网络上获取Queue,所以注册时只提供名字:
QueueManager.register(&#39;get_task_queue&#39;)
QueueManager.register(&#39;get_result_queue&#39;)

# 连接到服务器,也就是运行task_master.py的机器:
server_addr = &#39;127.0.0.1&#39;
print(&#39;Connect to server %s...&#39; % server_addr)
# 端口和验证码注意保持与task_master.py设置的完全一致:
m = QueueManager(address=(server_addr, 5000), authkey=b&#39;abc&#39;)
# 从网络连接:
m.connect()
# 获取Queue的对象:
task = m.get_task_queue()
result = m.get_result_queue()
# 从task队列取任务,并把结果写入result队列:
for i in range(10):
  try:
    n = task.get(timeout=1)
    print(&#39;run task %d * %d...&#39; % (n, n))
    r = &#39;%d * %d = %d&#39; % (n, n, n*n)
    time.sleep(1)
    result.put(r)
  except Queue.Empty:
    print(&#39;task queue is empty.&#39;)
# 处理结束:
print(&#39;worker exit.&#39;)
로그인 후 복사

任务进程要通过网络连接到服务进程,所以要指定服务进程的IP。http://www.jb51.net/article/65112.htm

小结

Python的分布式进程接口简单,封装良好,适合需要把繁重任务分布到多台机器的环境下。

注意Queue的作用是用来传递任务和接收结果,每个任务的描述数据量要尽量小。比如发送一个处理日志文件的任务,就不要发送几百兆的日志文件本身,而是发送日志文件存放的完整路径,由Worker进程再去共享的磁盘上读取文件。

위 내용은 Python 다중 프로세스에서 병렬 처리를 완료하는 방법 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿