Python并发编程04 /多线程、生产消费者模型、线程进程对比、线程的方法、线程join、守护线程、线程互斥锁

1. 生产消费者模型

  • 定义:编程思想,模型,设计模式,理论等等,都是一种编程的方法,遇到类似的情况,套用即可.

  • 生产者消费者模型三要素:

    生产者: 产生数据的

    消费者: 接收数据做进一步处理的

    容器: 队列,起到缓冲的作用,平衡生产力与消费力,解耦.

  • 代码示例:

    from multiprocessing import Process
    from multiprocessing import Queue
    import time
    import random def producer(q,name):
    for i in range(1,6):
    time.sleep(random.randint(1,2))
    res = f'{i}号包子'
    q.put(res)
    print(f'生产者{name} 生产了{res}') def consumer(q,name):
    while 1:
    try:
    food = q.get(timeout=3)
    time.sleep(random.randint(1, 3))
    print(f'\033[31;0m消费者{name} 吃了{food}\033[0m')
    except Exception:
    return if __name__ == '__main__':
    q = Queue()
    p1 = Process(target=producer,args=(q,'张三'))
    p2 = Process(target=consumer,args=(q,'李四'))
    p1.start()
    p2.start()

2. 线程的理论知识

  • 什么是线程

    标准描述开启一个进程:开启一个进程:进程会在内存中开辟一个进程空间,将主进程的资料数据全部复制一份,线程会执行里面的代码.

    ***进程是资源单位, 线程是执行单位;是操作系统调度的最小单元,是进程中的实际运作单位.

  • 线程vs进程

    1. 开启进程的开销非常大,比开启线程的开销大很多.
    2. 开启线程的速度非常快.要快几十倍到上百倍.
    3. 同一进程内线程与线程之间可以共享数据,进程与进程之间需借助队列等方法实现通信.
  • 线程的应用

    单个进程开启三个线程.并发的执行任务.

    并发:一个cpu 看起来像是同时执行多个任务.

  • 主线程子线程没有地位之分

    一个主线程在干活,当干完活了,得等待其他线程干完活之后,才能结束本进程.

3. 开启线程的两种方式

  • 方式一

    from threading import Thread
    import time def task(name):
    print(f'{name} is running')
    time.sleep(1)
    print(f'{name} is gone') if __name__ == '__main__': t1 = Thread(target=task,args=('张三',)) # args的参数一定要是元组
    t1.start()
    print('===主线程') # 线程是没有主次之分的.
  • 方式二

    from threading import Thread
    import time class MyThread(Thread):
    def __init__(self,name,l1,s1):
    super().__init__()
    self.name = name
    self.l1 = l1
    self.s1 = s1
    def run(self):
    print(f'{self.name} is running')
    time.sleep(1)
    print(f'{self.name} is gone') if __name__ == '__main__':
    t1 = MyThread('张三', [1,2,3], '180')
    t1.start()
    print('=====主线程')

4. 线程、进程对比代码验证

  • 开启速度对比

    # 多进程
    from threading import Thread
    from multiprocessing import Process
    import os def work():
    print('hello') if __name__ == '__main__':
    t=Process(target=work)
    t.start()
    print('主线程/主进程') # 多线程
    from threading import Thread
    import time def task(name):
    print(f'{name} is running')
    time.sleep(1)
    print(f'{name} is gone') if __name__ == '__main__':
    t1 = Thread(target=task,args=('张三',))
    t1.start()
    print('===主线程') # 结论:
    # 多进程是先打印的'主线程/主进程',多线程是先打印的'张三 is running',所以得出线程的开启速度比进程快
  • 对比pid

    # 多进程
    from multiprocessing import Process
    import time
    import os def task(name):
    print(f'子进程: {os.getpid()}')
    print(f'主进程: {os.getppid()}') if __name__ == '__main__':
    p1 = Process(target=task,args=('张三',)) # 创建一个进程对象
    p2 = Process(target=task,args=('李四',)) # 创建一个进程对象
    p1.start()
    p2.start()
    print(f'==主{os.getpid()}') # 多线程
    from threading import Thread
    import os def task():
    print(os.getpid()) if __name__ == '__main__':
    t1 = Thread(target=task)
    t2 = Thread(target=task)
    t1.start()
    t2.start()
    print(f'===主线程{os.getpid()}') # 结论:
    # 线程没有pid,进程才有pid
  • 共享数据的对比,同一个进程内线程共享内部数据

    from threading import Thread
    import os x = 3
    def task():
    global x
    x = 100 if __name__ == '__main__':
    t1 = Thread(target=task)
    t1.start()
    t1.join()
    print(f'===主线程{x}') # 同一进程内的资源数据对于这个进程内的多个线程来说是共享的.

5. 线程的其他方法

  • 代码示例:

    from threading import Thread
    from threading import currentThread
    from threading import enumerate
    from threading import activeCount
    import os
    import time def task():
    print(currentThread()) # 获取当前线程对象
    time.sleep(1)
    print('666') if __name__ == '__main__':
    t1 = Thread(target=task,name='线程1') # name 设置线程名
    t2 = Thread(target=task,name='线程2')
    t1.start()
    t2.start()
    time.sleep(2)
    print(t1.isAlive()) # 判断线程是否活着
    print(t1.getName()) # 获取线程名
    t1.setName('子线程-1') # 设置线程的名称
    print(t1.name) # 获取线程名 *** # threading方法
    print(currentThread()) # 获取当前线程的对象
    print(enumerate()) # 返回一个列表,包含所有的线程对象
    print(activeCount()) # 获取当前线程存活个数 ***
    print(f'===主线程{os.getpid()}')

6. 线程join

  • join: 阻塞 告知主线程要等待我子线程执行完毕之后再执行主线程

  • 代码示例:

    from threading import Thread
    import time def task(name):
    print(f'{name} is running')
    time.sleep(1)
    print(f'{name} is gone') if __name__ == '__main__':
    start_time = time.time()
    t1 = Thread(target=task,args=('张三',))
    t2 = Thread(target=task,args=('李四',))
    t3 = Thread(target=task,args=('王五',)) t1.start()
    t1.join()
    t2.start()
    t2.join()
    t3.start()
    t3.join() print(f'===主线程{time.time() - start_time}') # 线程是没有主次之分的.

7. 守护线程

  • 守护线程:等待非守护子线程以及主线程结束之后,结束.

  • 代码示例:

    示例一:简单使用守护线程

    from threading import Thread
    import time def sayhi(name):
    print('你好!')
    time.sleep(2)
    print('%s say hello' %name) if __name__ == '__main__':
    t = Thread(target=sayhi,args=('张三',))
    # 方式1:
    t.setDaemon(True) # 必须在t.start()之前设置
    # 方式2:
    t.daemon = True
    t.start()
    print('主线程')

    示例二:守护线程在非守护线程结束前先结束

    from threading import Thread
    import time def foo():
    print(123) # 第1步
    time.sleep(1)
    print("end123") # 第4步 def bar():
    print(456) # 第2步
    time.sleep(3)
    print("end456") # 第5步 t1=Thread(target=foo)
    t2=Thread(target=bar) t1.daemon=True
    t1.start()
    t2.start()
    print("---main---") # 第3步 # 结果:
    # 123
    # 456
    # ---main---
    # end123
    # end456

    示例三:守护线程在非守护线程结束之后结束

    from threading import Thread
    import time def foo():
    print(123) # 第1步
    time.sleep(3)
    print("end123") def bar():
    print(456) # 第2步
    time.sleep(1)
    print("end456") # 第4步 t1=Thread(target=foo)
    t2=Thread(target=bar) t1.daemon=True
    t1.start()
    t2.start()
    print("main-------") # 第3步 # 结果:
    # 123
    # 456
    # main-------
    # end456

8. 线程互斥锁

  • 多个任务公抢一个数据,保证数据的安全的目的,要让其串行

  • 代码示例

    from threading import Thread
    from threading import Lock
    import time
    import random x = 100
    def task(lock):
    lock.acquire()
    global x
    temp = x
    time.sleep(0.01)
    temp = temp - 1
    x = temp
    lock.release() if __name__ == '__main__':
    mutex = Lock()
    l1 = []
    for i in range(100):
    t = Thread(target=task,args=(mutex,))
    l1.append(t)
    t.start() for i in l1:
    i.join()
    print(f'主线程{x}')

Python并发编程04 /多线程、生产消费者模型、线程进程对比、线程的方法、线程join、守护线程、线程互斥锁的更多相关文章

  1. Python之queue模块以及生产消费者模型

    队列 队列类似于一条管道,元素先进先出,进put(arg),取get() 有一点需要注意的是:队列都是在内存中操作,进程退出,队列清空,另外,队列也是一个阻塞的形态. 队列分类 队列有很多中,但都依赖 ...

  2. Python - Asyncio模块实现的生产消费者模型

    [原创]转载请注明作者Johnthegreat和本文链接 在设计模式中,生产消费者模型占有非常重要的地位,这个模型在现实世界中也有很多有意思的对应场景,比如做包子的人和吃包子的人,当两者速度不匹配时, ...

  3. python并发编程之多线程基础知识点

    1.线程理论知识 概念:指的是一条流水线的工作过程的总称,是一个抽象的概念,是CPU基本执行单位. 进程和线程之间的区别: 1. 进程仅仅是一个资源单位,其中包含程序运行所需的资源,而线程就相当于车间 ...

  4. 29 python 并发编程之多线程理论

    一 什么是线程 在传统操作系统中,每个进程有一个地址空间,而且默认就有一个控制线程 线程顾名思义,就是一条流水线工作的过程,一条流水线必须属于一个车间,一个车间的工作过程是一个进程 车间负责把资源整合 ...

  5. 三 python并发编程之多线程-理论

    一 什么是线程 在传统操作系统中,每个进程有一个地址空间,而且默认就有一个控制线程 线程顾名思义,就是一条流水线工作的过程,一条流水线必须属于一个车间,一个车间的工作过程是一个进程 车间负责把资源整合 ...

  6. python并发编程之多线程理论部分

    阅读目录 一 什么是线程 二 线程的创建开销小 三 线程与进程的区别 四 为何要用多线程 五 多线程的应用举例 六 经典的线程模型(了解) 七 POSIX线程(了解) 八 在用户空间实现的线程(了解) ...

  7. 35、python并发编程之多线程(理论篇)

    一 什么是线程 二 线程的创建开销小 三 线程与进程的区别 四 为何要用多线程 五 多线程的应用举例 六 经典的线程模型(了解) 七 POSIX线程(了解) 八 在用户空间实现的线程(了解) 九 在内 ...

  8. 【Java并发编程】:生产者—消费者模型

    生产者消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一存储空间,生产者向空间里生产数据,而消费者取走数据. 这里实现如下情况的生产--消费模型: 生产者不断交替地生产两组数据“姓 ...

  9. python 并发编程之多线程

    一.线程理论 1.什么是线程 ​ 多线程(即多个控制线程)的概念是,在一个进程中存在多个线程,多个线程共享该进程的地址空间,相当于一个车间内有多条流水线,都共用一个车间的资源. 所以,进程只是用来把资 ...

随机推荐

  1. [CF453D]Little Pony and Elements of Harmony

    题目   点这里看题目. 分析   设\(count(x)\)为\(x\)的二进制中\(1\)的个数.因此\(f(u,v)=count(u\oplus v)\)   看一看每次转移,我们发现最不友好的 ...

  2. 大厂面试过程复盘(微信/阿里/头条均拿offer,附答案篇)

    背景 本人前端,3年经验,由于个人的原因,决定跳槽,于是大概3月开始找工作,总历时大概2个月,面试了微信/阿里/头条,三家都拿到了offer,来分享一下面经. 问题比较多,而且很多面试题都是跟个人项目 ...

  3. springboot的坑

    No tests found for given includes:xxxx 找不到findOne()方法 可以用它的findById(id).orElse(null); 存在就会直接返回值,如果不存 ...

  4. Spark文档阅读之一:Spark Overview

    Document: https://spark.apache.org/docs/latest/index.html 版本:2.4.5   1. spark的几种执行方式 1)交互式shell:bin/ ...

  5. Spring Boot 在启动时进行配置文件加解密

    Spring Boot Application 事件和监听器 寻找到application.yml的读取的操作. 从spring.factories 中查看到 # Application Listen ...

  6. Android学习笔记菜单资源文件

    创建菜单资源 menu_one.xml <?xml version="1.0" encoding="utf-8"?> <menu xmlns: ...

  7. Windows 程序设计(4) MFC 03 -系列学习

    本文整体目录和绝大部门内容来自 [鸡啄米网站]的MFC系列文章,欢迎支持原创 (一)VS2010/MFC编程入门之前言 VC++全称是Visual C++,是由微软提供的C++开发工具,它与C++的根 ...

  8. MDX

    简介 把md文件里的图片转成base64,方便发给别人和上传博客园等博客平台 初衷 用Typora写markdown的感觉很爽,但是每当我写好一篇文章,想要发给小伙伴们炫耀炫耀,或者上传博客园,CSD ...

  9. psp表格

    陈康杰psp表格 PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟) Planning 计划 10 10 Estimate 估计这个任务 ...

  10. sqlserver导致服务器异常卡死

    1.业务反应,服务器三天两头就要重启一次,要不然直接hang掉,登上服务器,异常的慢,大概进去需要十分钟的时间,查看一下电脑配置,8核8G的物理机. 2.查看一下任务管理器中的资源使用情况,发现cpu ...