python学习笔记-tip53(多进程--multiprocessing)

操作系统背景知识

Unix/Linux系统

Unix/Linux提供了一个fork()函数系统调用,这个fork()函数比较特殊?
为什么特殊呢?
因为普通的函数调用一次,那么函数就返回一次,而这个fork()函数调用一次呢,它会返回两次!
为什么会返回两次呢?
因为操作系统自动把当前进程(成为父进程),复制一份(这一份称为子进程),然后函数分别在父进程和子进程内返回。

而子进程,永远返回0,而父进程返回子进程的id。

为什么这么设计呢?答案是:这样做的话一个父进程就可以 fork 出很多子进程,所以父进程要记下每个子进程的 id ,而子进程只需要调用 getppid() 就可以拿到父进程的 id。

python 的 os 模块封装了常见的系统调用,其中就包括 fork,可以在 python 中轻松创建子进程:

示例代码:

    import os
    #os.getpid()获取当前进程id
    print('Process (%s) start...' %os.getpid())
    #下方代码只能在Unix/Linux/Mac系统使用,用来创建子进程(Mac是Unix系统中的一种)
    pid=os.fork()
    if pid==0:
        #因为当当前的pid为0时,说明当前的进程属于子进程,所以需要通过getppid()获得父进程id
       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))

我们来看下实际效果(我的机器为Mac,所以此代码适用)


根据输出结果,我们应该能够理解 fork() 函数的作用了,fork()函数虽然只调用了一次,但是他反回了两次,所以有两个输出结果。

有了fork调用,一个进程在接到新任务时就可以复制出一个子进程来处理新任务,常见的Apache服务器就是由父进程监听端口,每当有新的http请求时,就fork出子进程来处理新的http请求。


ok,你可能会想,上面的这么多话都是说给使用 mac 或者使用 linux 系统的人听的,如果现在在用 window 那该怎么办呢?

放心,python 为 windows 电脑提供了 multiprocessing模块

下面很重要哦

而这个multiprocessing模块就是跨平台版本的多进程模块

multiprocessing模块

multiprocessing 模块提供了一个 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.')

执行结果如下:

Parent process 928.
Process will start.
Run child process test (929)...
Process end.

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

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

Pool

如果要大量创建子进程,可以使用进程池 Pool 批量创建子进程

  from multiprocessing import Pool
  import time,os,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 senconds.'%(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('Waitting for all subProcesses done...')
      p.close()
      p.join()
      print('All subProcesses done.')      

我们来看下真实案例:


我们来分析下:

  • 对 Pool 对象调用 join() 方法,会等待所有子进程执行完毕;
  • 调用 join() 之前必须先调用 close(),调用 close() 之后不能继续添加新的 Process 了;
  • 请注意输出的结果:task0,1,2,3是立即执行的,而task4要等待前面某个task执行完毕之后才会执行。这是因为Pool的默认大小定义成了4,因此,最多同时执行4各进程,这是Pool有意设计的限制,并不是操作系统的限制。

如果 改成

p=Pool(5)

就可以同时跑5个进程。

由于Pool的默认大小是 cpu 的核数,如果你拥有8核cpu,要提交至少9个子进程,才能看到上边了类似的等待结果。

子进程-subprocess

很多时候,子进程并不是自身,而是一个外部进程。我们创建了子进程后,还需要控制子进程的输入和输出。
subprocess 模块可以让我们非常方便的启动一个子进程,然后控制其输入和输出

下面例子演示了如何在python代码中运行命令:

      nslookup www.python.org,这和命令行直接运行的效果是一样的

我们先来看下,直接在命令行执行 nslookup www.python.org的输出

ok,我们下面来看下代码:

  import subprocess
  print('& nslookup www.python.org')
  r=subprocess.call(['nslookup','www.python.org'])
  print('Exit code:',r)

我们来看下运行结果:


如果子进程还需要输入,则可以通过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()

我们来看下实际案例


注意

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

总结

  • 在Unix/Linux下,可以使用 fork() 调用实现多进程
  • 要实现跨平台的多进程,可以使用mutiprocessing 模块
  • 进程间通信是通过 Queue、Pipes 等实现的
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容