问:线程学完了,现在我们开始学习进程了吧?

答:是的。前面说到线程就是我们的手,我们现在可以学习一下我们的“胳膊”了。

  我们有了多线程,为什么还要学习多进程呢?这是因为在Python当中有一把GIL锁的存在,比如某些耗CPU的运算的时候,我们可以运行多进程多个CPU并发的操作进行操作。对于IO操作来说,我们的瓶颈不在于我们的CPU因此我们用多线程操作。进程切换操作不是轻量级的。

  我们首先举例一个数据密集型的操作,来计算斐波那契数列:

from  concurrent.futures import ThreadPoolExecutor,as_completed
from concurrent.futures import ProcessPoolExecutor
import time def fib(n):
if n<=:
return
return fib(n-) + fib(n-) if __name__ == '__main__':
with ThreadPoolExecutor() as executor:
all_task = [executor.submit(fib,(num)) for num in range(,)]
start_time = time.time()
for future in as_completed(all_task):
data = future.result()
print("get result:= {}".format(data))
print("multithread last time is {}".format(time.time()-start_time)) with ProcessPoolExecutor() as executor:
all_task = [executor.submit(fib,(num)) for num in range(,)]
start_time = time.time()
for future in as_completed(all_task):
data = future.result()
print("get result:= {}".format(data))
print("multiprocess last time is {}".format(time.time()-start_time))
#
# multithread last time is 43.156678199768066
# multiprocess last time is 27.62783455848694

  我们明显看到多进程比多线程快。

  我们在以一个IO操作来进行对比:

from  concurrent.futures import ThreadPoolExecutor,as_completed
from concurrent.futures import ProcessPoolExecutor
import time def random_sleep(n):
time.sleep(n)
return n if __name__ == '__main__':
with ThreadPoolExecutor() as executor:
all_task = [executor.submit(random_sleep,(num)) for num in []*]
start_time = time.time()
for future in as_completed(all_task):
data = future.result()
print("get result:= {}".format(data))
print("multithread last time is {}".format(time.time()-start_time)) with ProcessPoolExecutor() as executor:
all_task = [executor.submit(random_sleep,(num)) for num in []*]
start_time = time.time()
for future in as_completed(all_task):
data = future.result()
print("get result:= {}".format(data))
print("multiprocess last time is {}".format(time.time()-start_time))
#
# multithread last time is 20.035860300064087
# multiprocess last time is 20.641016483306885

  

  正式进入我们的进程操作:

import os
import time
# fork只能用于linux下面
pid = os.fork()
print("bobby")
if pid == :
print("子进程{},父进程是{}".format(os.getpid(),os.getppid()))
else:
print("我是父进程:{}".format(pid)) time.sleep()

  这段代码只能在Linux下运行。我们发现的问题是如果主进程结束了,子进程还是会运行的。

  

问:进程如何进行编程?

答:我们懂了线程的编程,进程的编程会变得非常的简单。多余的内容就不再讲解,我们讲解一些不同的包,其实这些包的应用也是跟进程差不多的。

  multiprocessing

import multiprocessing
import time
def get_html(n):
time.sleep(n)
return n if __name__ == '__main__':
progress = multiprocessing.Process(target=get_html,args=(,))
progress.start()
progress.join()

  我们还可以直接获取进程的pid和ppid。

  其他和我们多线程差不都就不详解了。

  使用进程池:

  进程池:Pool和ProcessPoolExecutor。后那个跟线程一样。我们单独说一下Pool这个进程池。

import multiprocessing
import time
from multiprocessing import Pool def get_html(n):
time.sleep(n)
return n if __name__ == '__main__':
progress = multiprocessing.Process(target=get_html,args=(,))
progress.start()
progress.join()
pool = Pool(multiprocessing.cpu_count())
print(multiprocessing.cpu_count())
result = pool.apply_async(get_html,args=(,))
pool.close()

  注意最后要关闭线程池。详细的关于线程池的代码可以参照这里:https://www.cnblogs.com/noah0532/p/10938771.html

  特别要说明的是有两个方法:imap 和 imap_unordered(这个是谁先完成先打印谁)

for result in  pool.imap(get_html,[,,]):

  

  进程间的通信:

  进程间的通信和线程间的通信有一样的也有不一样的地方,比如锁就不能使用了。

  举一个简单的例子:用队列进行通信

from multiprocessing import Process,Queue
# from queue import Queue # 这个queue就不能用了
import time def producer(queue):
queue.put("a")
time.sleep() def consumer(queue):
time.sleep()
data = queue.get()
print(data) if __name__ == '__main__':
queue = Queue()
my_producer = Process(target=producer,args=(queue,))
my_consumer = Process(target=consumer, args=(queue,))
my_producer.start()
my_consumer.start()
my_producer.join()
my_consumer.join()

  在多进程的编程中不能用之前的queue了,带用multiprocessing里面的queue,这一带你要注意

  我们再举一个共享变量的例子:

from multiprocessing import Process
import time def producer(a):
a +=
time.sleep() def consumer(a):
time.sleep()
print(a) if __name__ == '__main__':
a =
my_producer = Process(target=producer,args=(a,))
my_consumer = Process(target=consumer, args=(a,))
my_producer.start()
my_consumer.start()
my_producer.join()
my_consumer.join()

  我们发现我们的全局变量不能用了,正如我们前面说的,我们再进程中每一块的变量是单独的,不能共享的。

  另外multiprocessing中的queue也不能用在进程池当中。如果我们想在进程当中应用就带用Manager当中的Queue

from multiprocessing import Process,Queue,Manager,Pool
import time def producer(queue):
queue.put("a")
time.sleep() def consumer(queue):
time.sleep()
data = queue.get()
print(data) if __name__ == '__main__':
queue = Manager().Queue()
pool = Pool() pool.apply_async(producer,args=(queue,))
pool.apply_async(consumer, args=(queue,)) pool.close()
pool.join()

  另外,我们还可以通过我们的pipe管道来进行通讯,但是Pipe只能使用两个进程间的通信,如果是两个交换pipe的性能比queue高

from multiprocessing import Process,Queue,Manager,Pool,Pipe
import time def producer(pipe):
pipe.send("bobby") def consumer(pipe):
print(pipe.recv()) if __name__ == '__main__':
# pipe只能用于两个进程间的通讯
receive_pipe,send_pipe = Pipe()
my_producer = Process(target=producer,args=(send_pipe,))
my_consumer = Process(target=consumer, args=(receive_pipe,)) my_producer.start()
my_consumer.start()
my_producer.join()
my_consumer.join()

  重点:进程间的共享内存操作:Manager().dict(),array()....常用的数据类型都有。

from multiprocessing import Process,Queue,Manager,Pool,Pipe

def add_data(p_dict,key,value):
p_dict[key] = value if __name__ == '__main__':
progress_dict = Manager().dict() first_progess = Process(target=add_data,args=(progress_dict,"bobby1",))
second_progess = Process(target=add_data, args=(progress_dict, "bobby1", )) first_progess.start()
second_progess.start()
first_progess.join()
second_progess.join() print(progress_dict)
# {'bobby1': }

Python说文解字_Python之多任务_03的更多相关文章

  1. Python说文解字_Python之多任务_01

    Python 之 多任务: Python之多任务是现在多任务编程运用Python语言为载体的一种体现.其中涵盖:进程.线程.并发等方面的内容,以及包括近些年在大数据运算.人工智能领域运用强大的GPU运 ...

  2. Python说文解字_Python之多任务_05

    问:在Py3.5之前yield表现非常好,在Py3.5之后为了将予以变得更加明确,就引入了async和await关键词用于定义原生的协议. 答:async和await原生协程: async def d ...

  3. Python说文解字_Python之多任务_02

    第三部分:Semaphore控制进入数量的锁 有时候可能需要运行多个工作线程同时访问一个资源,但要限制总数.例如,连接池支持同时连接,但是数目可能是固定的,或者一个网络应用可能支持固定数据的并发下载. ...

  4. Python说文解字_Python之多任务_04

    问:并发.并行.同步.异步.阻塞.非阻塞 答: 并发.并行: 并发是指一个时间段内(不是指的时间点),有几个程序在同一个CPU上运行,但是任意时刻只有一个程序在CPU上运行.对人类的时钟来说1秒钟能干 ...

  5. Python说文解字_详解元类

    1.深入理解一切接对象: 1.1 什么是类和对象? 首先明白元类之前要明白什么叫做类.类是面向对象object oriented programming的重要概念.在面向对象中类和对象是最基本的两个概 ...

  6. Python说文解字_杂谈05

    1. isinstance和type: is和==符号,is指的是内存地址,是不是一个对象,ID知否相同 集成链 class A: pass class B(A): pass b = B() prin ...

  7. Python说文解字_杂谈09

    1. 元类编程代码分析: import numbers class Field: pass class IntField(Field): # 数据描述符: # 初始化 def __init__(sel ...

  8. Python说文解字_杂谈08

    1. Python变量到底是什么? Python和Java中的变量本质不一样,python的变量实质是一个指针 int str,便利贴 a = 1 # 1. a贴在1上面 # 2. 它的过程是先生成对 ...

  9. Python说文解字_杂谈07

    1. 深入dict from collections.abc import Mapping,MutableMapping # dict 属于mapping类型 a = {} print(isinsta ...

随机推荐

  1. Java的equals方法的使用技巧

    Java的equals方法的使用技巧 1.业务场景: 在某个社交软件中,要求每个用户的用户名(name)必须独一无二,那么在每次增加新用户的时候,都要对该用户的注册名进行判断,如果当前用户名已经被占用 ...

  2. Day7 - A - Visible Lattice Points POJ - 3090

    A lattice point (x, y) in the first quadrant (x and y are integers greater than or equal to 0), othe ...

  3. 038、Java中逻辑运算之非运算“!”

    01.代码如下: package TIANPAN; /** * 此处为文档注释 * * @author 田攀 微信382477247 */ public class TestDemo { public ...

  4. 030、Java中的求模计算

    01.代码如下: package TIANPAN; /** * 此处为文档注释 * * @author 田攀 微信382477247 */ public class TestDemo { public ...

  5. WC2020 联训 #19 矩阵

    好不容易自己切一道题 链接 Description 在一个 \(n×(n+1)\) 的棋盘上放棋子, \(n\) 行中每行都恰好有两枚棋子,并且 \(n+1\) 列中每列都至多有两枚棋子,设 \(n= ...

  6. 【剑指Offer】面试题27. 二叉树的镜像

    题目 请完成一个函数,输入一个二叉树,该函数输出它的镜像. 例如输入:      4    /   \   2     7  / \   / \ 1   3 6   9 镜像输出:      4   ...

  7. js实现连续输入之后发送请求

    输入框是我们经常会用到的功能,想要实现输入就请求的功能 但是在实际开发中,为了减少服务器压力,会在输入之后停留1s没有输入之后再进行搜索 研究之后用原生js及表单写了一个简单的demo,如果有好的de ...

  8. 用fiddler监控移动端的通讯

    用fiddler监控移动端的通讯  1 依次打开Fiddler->Tools->Fiddler Options在[Connection]面板里将Allow remote computers ...

  9. 快速进阶Vue3.0

    在2019.10.5日发布了Vue3.0预览版源码,但是预计最早需要等到 2020 年第一季度才有可能发布 3.0 正式版. 可以直接看 github源码. 新版Vue 3.0计划并已实现的主要架构改 ...

  10. Django 模板渲染

    模板语言 {{ 变量 }} {% 逻辑 %} {{ 变量 }} {{ 变量 }}中的点号 用于取出字典/列表等类型数据的值 {{ list.2 }} 获取列表list中索引为2的值 {{ dict.n ...