Python多线程 threading

Python多线程 threading

在很多程序设计问题中,都要求程序能够停下手头的工作,改为处理其他一些问题,再返回回来继续执行。可以通过多种途径达到这个目的。最开始的时候,那些掌握机器低级语言的程序员编写一些“中断服务例程”,主进程的暂停是通过硬件级的中断实现的。尽管这是一种有用的方法,但编出的程序很难移植,由此造成了另一类的代价高昂问题。中断对那些实时性很强的任务来说是很有必要的。但对于其他许多问题,只要求将问题划分进入独立运行的程序片断中,使整个程序能更迅速地响应用户的请求 。

最开始,线程只是用于分配单个处理器的处理时间的一种工具。但假如操作系统本身支持多个处理器,那么每个线程都可分配给一个不同的处理器,真正进入“并行运算”状态。从程序设计语言的角度看,多线程操作最有价值的特性之一就是程序员不必关心到底使用了多少个处理器。程序在逻辑意义上被分割为数个线程;假如机器本身安装了多个处理器,那么程序会运行得更快,毋需作出任何特殊的调校。根据前面的论述,大家可能感觉线程处理非常简单。但必须注意一个问题:共享资源!如果有多个线程同时运行,而且它们试图访问相同的资源,就会遇到一个问题。举个例子来说,两个线程不能将信息同时发送给一台打印机。为解决这个问题,对那些可共享的资源来说(比如打印机),它们在使用期间必须进入锁定状态。所以一个线程可将资源锁定,在完成了它的任务后,再解开(释放)这个锁,使其他线程可以接着使用同样的资源 。

多线程是为了同步完成多项任务,不是为了提高运行效率,而是为了提高资源使用效率来提高系统的效率。线程是在同一时间需要完成多项任务的时候实现的 。

threading

在Python中使用threading模块来进行多线程编程。threading中,线程只有两个状态,就绪和运行。

在单核cpu中,不存在真正的并行。多个线程看似是同时进行,其实是cpu使用很快的速度对多个线程逐个运行,每次运行一定的时间或者指令。

threading.Thread

主要参数用法
target目标函数
args函数参数
kwargs关键字传参
daemon是否是darmon线程
name线程名

Python中的线程比较简单,线程只有运行和结束状态。也无法主动终止或者挂起。一旦启动,除非发生异常或者主线程退出,线程都不会突然结束。

使用一个函数和参数来初始化threading.Thread类,并调用类方法start,就开启一个新线程了。

import threading


def add(x,y):
    print(x + y)



t = threading.Thread(target=add, args=(4,5))
t.start()

调用start方法后,会创建一个新线程,并在新线程中调用类方法run。所以也可以通过继承threading.Thread类并重载run方法来实现多线程。

class MyThread(threading.Thread):

    def __init__(self, x, y):
        self.x = x
        self.y = y

    def run(self) -> None:
        print(self.x + self.y)

daemon

  • Python中的daemon线程有别于Linux系统中的daemon进程

  • deamon属性是bool类型,只接受True和False

  • 线程对象创建时,daemon的值默认会与创建对象的线程相同,主线程的默认值为False

主线程退出时所有线程都会退出。当主线程执行结束后,如果所有子线程中没有non-daemon线程(即daemon属性为False),主线程就会退出。如果有任意一个non-daemon的子线程,主线程就会处于stop状态等待直至所有non-daemon线程结束。

名称用法
daemon属性需要在start()调用前修改否则引发RunTimeError
isDaemon()是否为daemon线程
setDaemon同样需要在start()前设置

join

threading.Thread类有join方法

方法参数
jointimeout,阻塞到线程结束或者超过该时间

调用后当前线程会阻塞直至调用的线程对象对应的线程结束或者timeout为止。

在主线程末使用join也能达到non-daemon的效果

t = threading.Thread(target=add, args=(4, 5), daemon=True)
t.start()
#t.join()
while(t.is_alive()):
    pass

使用while也能达到类似的效果

local

如果在多个线程中使用同一个全局变量,多个线程的修改就会相互影响,并且之间的影响是不可预测的。所以在threading模块中提供了一个local类隐性地为每个线程创建单独的变量。

用法:直接无参实例化一个threading.local对象,并在不同线程中给对象添加属性。

实现:

def __init__(self):
    # The key used in the Thread objects' attribute dicts.
    # We keep it a string for speed but make it unlikely to clash with
    # a "real" attribute.
    self.key = '_threading_local._localimpl.' + str(id(self))
    # { id(Thread) -> (ref(Thread), thread-local dict) }
    self.dicts = {}

local中使用线程对象的id来作为key构建dict,value为线程对象和线程字典组成的二元组,value中的dict存了在该线程给对象添加的所有属性。实际上在访问该对象属性时,访问的是各自的字典中的值,之间互不干扰。

注:很多时候使用局部变量能更方便的解决问题。

Timer类

threading.Timer类再start后不会马上执行,会在延迟时间后再执行,并且在执行前可以通过cancel方法来取消。

用法:与Thread相同,第一个位置参数为延迟时间。

线程安全

由于操作系统不存在真正的并行,每个线程都是执行一段时间就中止执行其他线程。当这些之间存在相互干扰的话,就会出现问题。例如,一个计算过程执行了一半,另外一个线程把其中的一个值给改了,这就会导致意料之外的结果,这样的操作是线程不安全的。

import threading

def foo():
    for i in range(100):
        print(i)

for i in range(5):
    t = threading.Thread(target=foo)
    t.start()
    
# result
#85
#63
#878668

在上述结果中,会出现多个结果在同一行,同时空多行的打印情况。可见print是不安全的,线程的中断影响到了打印效果。

解决线程安全问题的方法有多种,可以使用一个标志变量来设锁,当有一个线程进入打印时,上锁让其他线程等待。但是设锁是有性能代价的,需要具体需求来定。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Python多线程是通过threading模块来实现的。它允许程序在同一时间内执行多个线程,从而提高程序的效率。使用多线程可以让程序在执行耗时操作时不会阻塞主线程,从而提高程序的响应速度。在Python中,使用threading模块创建线程非常简单,只需要继承Thread类并实现run方法即可。同时,Python还提供了一些线程同步的机制,如锁、信号量、条件变量等,可以保证多个线程之间的数据同步和互斥访问。 ### 回答2: Python 是一种非常常用的编程语言,它的一个主要特点是它可以方便快捷地处理并行任务。Python 中处理并行任务的机制有很多,其中最常用的机制是多线程 threading多线程是指在一个程序中同时运行多个线程,这些线程可以共享一些系统资源,如 CPU 和内存。这样做的好处是可以使程序运行更加高效,快速地响应用户的操作。多线程因其高效、灵活、易于实现等特点,在多种应用场景中得到了广泛应用。 Python 中的 threading 模块是用于实现基本多线程功能的标准库之一。使用 threading 模块可以轻松创建和管理多个线程。可以使用 threading.Thread() 函数实例化一个线程对象和它相应的线程函数,然后通过调用 start() 方法启动该线程。在 start() 方法被调用之后,线程就开始了运行,并可以执行与其他线程并行运行的一些任务。当线程运行完所需任务之后,可以使用 join() 方法阻塞主线程,以等待该线程的结束。 使用 threading 模块,还可以使用锁(lock)机制来控制多个线程对于共享资源的访问。锁机制允许线程互斥地访问共享资源,以避免线程之间的竞争和冲突。如果有多个线程都希望使用同一个共享资源,可以使用 threading.Lock() 函数创建一个锁对象,然后使用 acquire() 方法获取锁,使用 release() 方法释放锁。 需要注意的是,虽然多线程可以提高程序运行的效率,但不当地使用多线程也可能会导致程序出现并发问题,如死锁等。因此,在使用多线程时,应谨慎地考虑各个线程对于共享资源的访问顺序和方式,以确保程序的安全和正确性。 总之,Python 中的多线程 threading 机制是非常强大和灵活的。通过它我们可以轻松地实现并行处理的功能,提高程序的效率和性能。但在使用多线程时,需要注意确保程序的安全和正确性,避免出现并发问题。 ### 回答3: Python多线程模块threadingPython中实现多线程的解决方案之一。threading模块提供了一个Thread类作为多线程的基础,可以通过派生Thread类和重写它的run()方法来定义线程的行为。 Python多线程实现为“线程全局解释器锁(Global Interpreter Lock,GIL)”模型,即同一进程中多个线程共享一个解释器,而只有一个线程可以执行Python代码(线程安全的C扩展模块除外)。这意味着,Python多线程无法实现真正的并行执行,只能通过多个线程之间的切换来实现并发。 使用Python多线程可以带来以下优点: 1. 提高程序响应速度:可以让程序在等待I/O的同时继续执行其他任务,提高了程序的并发处理能力。 2. 更好地利用多核CPU:虽然Python多线程不能实现真正的并行执行,但可以在多核CPU上将任务分配到不同的核上执行,从而更好地利用CPU资源。 3. 代码可读性高:Python多线程的实现相对简单,代码易读易懂。 然而,在使用Python多线程的过程中,也需要注意以下问题: 1. 线程同步问题:由于多个线程可能同时访问共享的数据,因此需要使用锁、信号量等线程同步机制来避免线程冲突。 2. 可能会出现死锁和饥饿问题:死锁是指多个线程相互等待对方释放资源而无法继续执行,饥饿是指某个线程因调度算法等原因无法获得执行机会。这些问题需要仔细设计线程的互动方式来避免。 3. Python多线程无法利用CPU的多核处理能力:Python的GIL模型限制了同一时刻只有一个线程能执行Python代码,因此无法利用多核CPU实现真正的并行处理。 总之,Python多线程是一种方便实用的线程处理方式,可以有效提高程序的并发处理能力,但也需要注意线程同步等问题,对于高性能的性能要求,建议使用Python中的进程池等多进程方案。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值