• < Python全景系列-5 > 解锁Python并发编程:多线程和多进程的神秘面纱揭晓


    欢迎来到我们的系列博客《Python全景系列》!在这个系列中,我们将带领你从Python的基础知识开始,一步步深入到高级话题,帮助你掌握这门强大而灵活的编程语法。无论你是编程新手,还是有一定基础的开发者,这个系列都将提供你需要的知识和技能。

     

    这是本系列的第五篇,我们将深入探讨Python中的并发编程,特别关注多线程和多进程的应用。我们将先从基本概念开始,然后通过详细举例探讨每一种机制,最后分享一些实战经验以及一种优雅的编程技巧。

     

    第一部分:多线程介绍

    线程是操作系统中最小的执行单元。在单个程序或进程内,可以并发运行多个线程,共享进程的资源,如内存和文件描述符。

    1.1 Python中的多线程

    Python支持多线程编程,并提供了`threading`模块作为支持。这个模块提供了`Thread`类,我们可以通过创建其实例并向其传递函数来创建新线程。当然,你也可以通过继承`Thread`类并重写`run()`方法来创建自定义线程。下面是一个多线程编程的例子:

    复制代码
    import threading
    
    def print_numbers():
        for i in range(10):
            print(i)
    
    def print_letters():
        for letter in 'abcdefghij':
            print(letter)
    
    # 创建线程
    t1 = threading.Thread(target=print_numbers)
    t2 = threading.Thread(target=print_letters)
    
    # 启动线程
    t1.start()
    t2.start()
    
    # 等待线程结束
    t1.join()
    t2.join()
    复制代码

     

    在上面的例子中,我们定义了两个函数:一个打印数字,另一个打印字母。然后我们创建了两个线程,每个线程的目标是执行这些函数。`start()`方法用于启动线程,而`join()`方法用于等待线程完成。

     

    1.2 多线程的实际应用

    尽管Python的多线程因为全局解释器锁(GIL)的存在,并不能实现真正的并行,但是它们在I/O密集型任务中仍然很有用。GIL是CPython解释器的一个互斥锁,保证在任何时刻只有一个线程在执行。这意味着在CPU密集型任务中,多线程可能不是最佳选择,因为它们无法充分利用多核CPU。

    然而,在I/O密集型任务中,多线程能够提高程序性能。例如,如果一个程序需要从多个源下载文件,那么使用多线程可以使得当一个线程等待网络响应时,其他线程可以继续下载其他文件。这样,程序可以在同一时间从多个源下载文件,大大提高了效率。

     

    第二部分:多进程介绍

    进程是操作系统中独立的执行实体,每个进程都有自己的内存空间、文件描述符等资源。与线程不同,进程之间的资源

    并不共享,每个进程都有自己独立的资源。

     

    2.1 Python中的多进程

     

    Python通过`multiprocessing`模块提供了多进程支持。类似于多线程,我们可以通过创建`Process`类的实例并向其传递函数来创建新进程。我们也可以通过继承`Process`类并重写`run()`方法来创建自定义进程。

    以下是一个简单的多进程编程的例子:

    复制代码
    import multiprocessing
    
    def print_numbers():
        for i in range(10):
            print(i)
    
    def print_letters():
        for letter in 'abcdefghij':
            print(letter)
    
    # 创建进程
    p1 = multiprocessing.Process(target=print_numbers)
    p2 = multiprocessing.Process(target=print_letters)
    
    # 启动进程
    p1.start()
    p2.start()
    
    # 等待进程结束
    p1.join()
    p2.join()
    复制代码

    这个例子和前面的多线程例子类似,不同的是这里我们创建的是两个进程,而不是线程。

     

    2.2 多进程的实际应用

    多进程可以实现真正的并行,使得Python程序可以利用多核CPU。因此,对于CPU密集型任务,多进程通常比多线程更有优势。另一方面,多进程的开销比多线程大,而且进程间的通信和同步也比线程间的更为复杂。因此,对于I/O密集型任务,或者需要频繁通信的任务,多线程可能会是更好的选择。

    第三部分:优化并发编程的技巧

    在Python中,`concurrent.futures`模块为多线程和多进程编程提供了高级接口,可以让我们更加简洁地编写代码。

    这个模块提供了`ThreadPoolExecutor`和`ProcessPoolExecutor`两个类,它们分别用于创建线程池和进程池。这两个类都实现了相同的接口,你可以使用`submit()`方法提交任务,然后使用`as_completed()`函数等待任务完成。

    下面是一个使用`concurrent.futures`模块的示例:

     
    复制代码
    import concurrent.futures
    
    def print_numbers():
        for i in range(10):
            print(i)
    
    def print_letters():
        for letter in 'abcdefghij':
            print(letter)
    
    # 使用线程池
    with concurrent.futures.ThreadPoolExecutor() as executor:
        future1 = executor.submit(print_numbers)
        future2 = executor.submit(print_letters)
        for future in concurrent.futures.as_completed([future1, future2]):
            pass
    
    # 使用进程池
    with concurrent.futures.ProcessPoolExecutor() as executor:
        future1 = executor.submit(print_numbers)
        future2 = executor.submit(print_letters)
        for future in concurrent.futures.as_completed([future1, future2]):
            pass
    复制代码

    在上面的例子中,我们创建了线程池和进程池,然后向它们提交任务。可以看到,使用`concurrent.futures`模块,我们的代码更加简洁,易读性和可维护性也有所提高。

    总结

    Python的多线程和多进程都是非常强大的工具,可以帮助我们编写出更高效的程序。然而,它们也各有优缺点,需要我们根据具体的任务和需求来选择。同时,Python还提供了`concurrent.futures`模块,可以使我们的并发编程变得更加简单和高效。

    我们希望本文能帮助你更好地理解和使用Python的多线程和多进程。如果你有任何疑问或者建议,欢迎在评论区留言。

    【第一时间获得Python全视角更新信息,请关注本人微信公众号: Python全视角

     

  • 相关阅读:
    spring boot项目上传头像
    动态规划——123. 买卖股票的最佳时机 III
    最难的IB课程为什么含金量最高?
    计算机毕业设计SSM电影购票和评分系统【附源码数据库】
    linux下使用selenium调用谷歌浏览器的一些问题
    华为云鲲鹏架构docker部署2048小游戏
    股票买卖问题I、II、III、IV、V、VI
    MapReduce项目案例1
    (六)Selenium自动化测试实战—unittest框架
    $nextTick属性使用与介绍
  • 原文地址:https://www.cnblogs.com/xfuture/p/17418197.html