0

0

一篇文章带你全面解析不一样的线程

Python当打之年

Python当打之年

发布时间:2023-08-10 16:13:35

|

713人浏览过

|

来源于Python当打之年

转载

iOS开发学习之iOS多线程和RunLoop 中文WORD版
iOS开发学习之iOS多线程和RunLoop 中文WORD版

iOS多线程编程对于iOS开发初学者来说,总是会觉得很难理解和掌握,现在通过几个实例来更加系统全面的理解iOS多线程编程,希望对大家有所帮助。 有些程序是一条直线,起点到终点;有些程序是一个圆,不断循环,直到将它切断。直线的如简单的Hello World,运行打印完,它的生命周期便结束了,像昙花一现那样;圆如操作系统,一直运行直到你关机。 一个运行着的程序就是一个进程或者叫做一个任务,一个进程至少包含一个线程,线程就是程序的执行流。Mac和iOS中的程序启动,创建好一个进程的同时,一个线程便开始运行,

下载
python当打之年" data-pluginname="mpprofile" data-signature="专注于各领域python技术,讲的是基础,打的是精锐!当打之年,打出精彩人生!">

前言

在将今天的知识点之前,大家是否了解线程,进程和协程了,那我们先来初步了解下吧。


线程

中央处理器的调度单元,简单点说就是程序中的末端执行者,相当于小弟的位置。

有人说python中的线程是个鸡肋,这是因为有了GIL,但是又不是一味的鸡肋,毕竟在执行io操作时还是挺管用的,只是在执行计算时就显得不尽人意。下面我们来看下线程的具体使用方法:

1.导入线程模块:

import threading as t

2.线程的用法

tt=t.Thread(group=None,target=None,name=None,args=(),kwargs={},name='',daemon=None)
group:线程组,必须是None
target:运行的函数
args:传入函数的参数元组
kwargs:传入函数的参数字典
name:线程名
daemon:线程是否随主线程退出而退出(守护线程)


Thread方法的返回值还有以下方法:
tt.start() : 激活线程,
tt.getName() : 获取线程的名称
tt.setName() :设置线程的名称 
tt.name : 获取或设置线程的名称
tt.is_alive() :判断线程是否为激活状态
tt.isAlive() :判断线程是否为激活状态
tt.setDaemon() 设置为守护线程(默认:False)
tt.isDaemon() :判断是否为守护线程
tt.ident :获取线程的标识符。只有在调用了start()方法之后该属性才有效
tt.join() :逐个执行每个线程,执行完毕后继续往下执行
tt.run() :自动执行线程对象


t的方法也有:
t.active_count(): 返回正在运行线程的数量
t.enumerate(): 返回正在运行线程的列表
t.current_thread().getName() 获取当前线程的名字
t.TIMEOUT_MAX 设置t的全局超时时间

下面我们来看下吧:

一篇文章带你全面解析不一样的线程


3.创建线程

线程可以使用Thread方法创建,也可以重写线程类的run方法实现,线程可分为单线程和多线程。

3.1 使用Thread方法来创建:

3.1.1 单线程
def xc():
    for y in range(100):
        print('运行中'+str(y))
tt=t.Thread(target=xc,args=()) #方法加入到线程
tt.start()  #开始线程
tt.join() #等待子线程结束
3.1.2 多线程
def xc(num):
    print('运行:'+str(num))
c=[]
for y in range(100):
    tt=t.Thread(target=xc,args=(y,))
    tt.start() #开始线程
    c.append(tt) #创建列表并添加线程
for x in c:
    x.join()  #等待子线程结束

3.2 重写线程的类方法

3.2.1 单线程
class Xc(t.Thread): #继承Thread类
    def __init__(self):
        super(Xc, self).__init__() 
    def run(self):  #重写run方法
        for y in range(100):
            print('运行中'+str(y))
x=Xc() 
x.start() #开始线程
x.join()  #等待子线程结束


也可以这么写:
Xc().run() 和上面的效果是一样的
3.2.2 多线程
class Xc(t.Thread): #继承Thread类
    def __init__(self):
        super(Xc, self).__init__() 
    def run(self,num):  #重写run方法
        print('运行:'+str(num))
x=Xc()
for y in range(10):
    x.run(y) #运行

4.线程锁

为什么要加锁,看了这个你就知道了:

一篇文章带你全面解析不一样的线程

多线程在运行时同时访问一个对象会产生抢占资源的情况,所以我们必须得束缚它,所以就要给他加一把锁把他锁住,这就是同步锁。要了解锁,我们得先创建锁,线程中有两种锁:Lock和RLock。

4.1 Lock

使用方法:

# 获取锁
当获取不到锁时,默认进入阻塞状态,设置超时时间,直到获取到锁,后才继续。非阻塞时,timeout禁止设置。如果超时依旧未获取到锁,返回False。
Lock.acquire(blocking=True,timeout=1)   


#释放锁,已上锁的锁,会被设置为unlocked。如果未上锁调用,会抛出RuntimeError异常。
Lock.release()

互斥锁,同步数据,解决多线程的安全问题:

n=10
lock=t.Lock()
def xc(num):
    lock.acquire()
    print('运行+:'+str(num+n))
    print('运行-:'+str(num-n))
    lock.release()
c=[]
for y in range(10):
    tt=t.Thread(target=xc,args=(y,))
    tt.start()
    c.append(tt)
for x in c:
    x.join()

这样就显得有条理了,而且输出也是先+后-。Lock在一个线程中多次使用同一资源会造成死锁。

死锁问题:

n=10
lock1=t.Lock()
lock2=t.Lock()
def xc(num):
  lock1.acquire()
  print('运行+:'+str(num+n))
  lock2.acquire()
  print('运行-:'+str(num-n))
  lock2.release()
  lock1.release()
c=[]
for y in range(10):
  tt=t.Thread(target=xc,args=(y,))
  tt.start()
  c.append(tt)
for x in c:
  x.join()


4.2 RLock

相比Lock它可以递归,支持在同一线程中多次请求同一资源,并允许在同一线程中被多次锁定,但是acquire和release必须成对出现。

使用递归锁来解决死锁:

n=10
lock1=t.RLock()
lock2=t.RLock()
def xc(num):
  lock1.acquire()
  print('运行+:'+str(num+n))
  lock2.acquire()
  print('运行-:'+str(num-n))
  lock2.release()
  lock1.release()
c=[]
for y in range(10):
  tt=t.Thread(target=xc,args=(y,))
  tt.start()
  c.append(tt)
for x in c:
  x.join()

这时候,输出变量就变得仅仅有条了,不在随意抢占资源。关于线程锁,还可以使用with更加方便:

#with上下文管理,锁对象支持上下文管理
with lock:   #with表示自动打开自动释放锁
  for i in range(10): #锁定期间,其他人不可以干活
    print(i)
  #上面的和下面的是等价的
if lock.acquire(1):#锁住成功继续干活,没有锁住成功就一直等待,1代表独占
  for i in range(10): #锁定期间,其他线程不可以干活
    print(i)
  lock.release() #释放锁


4.3 条件锁

等待通过,Condition(lock=None),可以传入lock或者Rlock,默认Rlock,使用方法:

Condition.acquire(*args)      获取锁


Condition.wait(timeout=None)  等待通知,timeout设置超时时间


Condition.notify(num)唤醒至多指定数目个数的等待的线程,没有等待的线程就没有任何操作


Condition.notify_all()  唤醒所有等待的线程 或者notifyAll()
def ww(c):
  with c:
    print('init')
    c.wait(timeout=5) #设置等待超时时间5
    print('end')
def xx(c):
  with c:
    print('nono')
    c.notifyAll() #唤醒所有线程
    print('start')
    c.notify(1) #唤醒一个线程
    print('21')
c=t.Condition() #创建条件
t.Thread(target=ww,args=(c,)).start()
t.Thread(target=xx,args=(c,)).start()

这样就可以在等待的时候唤醒函数里唤醒其他函数里所存在的其他线程了。


5.信号量

信号量可以分为有界信号量和无解信号量,下面我们来具体看看他们的用法:

5.1 有界信号量

它不允许使用release超出初始值的范围,否则,抛出ValueError异常。

#构造方法。value为初始信号量。value小于0,抛出ValueError异常
b=t.BoundedSemaphore(value=1)  


#获取信号量时,计数器减1,即_value的值减少1。如果_value的值为0会变成阻塞状态。获取成功返回True
BoundedSemaphore.acquire(blocking=True,timeout=None)  


#释放信号量,计数器加1。即_value的值加1,超过初始化值会抛出异常ValueError。
BoundedSemaphore.release()  


#信号量,当前信号量
BoundedSemaphore._value

一篇文章带你全面解析不一样的线程

可以看到了多了个release后报错了。


5.2 无界信号量

它不检查release的上限情况,只是单纯的加减计数器。

一篇文章带你全面解析不一样的线程

可以看到虽然多了个release,但是没有问题,而且信号量的数量不受限制。


6.Event

线程间通信,通过线程设置的信号标志(flag)的False 还是True来进行操作,常见方法有:

event.set()      flag设置为True
event.clear()  flag设置为False
event.is_set()  flag是否为True,如果 event.isSet()==False将阻塞线程;
设置等待flag为True的时长,None为无限等待。等到返回True,未等到超时则返回False
event.wait(timeout=None)

下面通过一个例子具体讲述:

import time
e=t.Event()
def ff(num):
  while True:
    if num<5:
      e.clear()   #清空信号标志
      print('清空')
    if num>=5:
      e.wait(timeout=1) #等待信号标志为真
      e.set()
      print('启动')
      if e.isSet(): #如果信号标志为真则清除标志
        e.clear()
        print('停止')
    if num==10:
      e.wait(timeout=3)
      e.clear()
      print('退出')
      break
    num+=1
    time.sleep(2)
ff(1)

一篇文章带你全面解析不一样的线程

设置延迟后可以看到效果相当明显,我们让他干什么事他就干什么事。


7.local

可以为各个线程创建完全属于它们自己的变量(线程局部变量),而且它们的值都在当前调用它的线程当中,以字典的形式存在。下面我们来看下:

l=t.local()  #创建一个线程局部变量
def ff(num):
  l.x=100  #设置l变量的x方法的值为100
  for y in range(num):
    l.x+=3 #改变值
  print(str(l.x))


for y in range(10):
  t.Thread(target=ff,args=(y,)).start() #开始执行线程

那么,可以将变量的x方法设为全局变量吗?我们来看下:

一篇文章带你全面解析不一样的线程

可以看出他报错了,产生错误的原因是因为这个类中没有属性x,我们可以简单的理解为局部变量就只接受局部。


8.Timer

设置定时计划,可以在规定的时间内反复执行某个方法。他的使用方法是:

t.Timer(num,func,*args,**kwargs) #在指定时间内再次重启程序

下面我们来看下:

def f():
  print('start')
  global t #防止造成线程堆积导致最终程序退出
  tt= t.Timer(3, f)
  tt.start()
f()

这样就达到了每三秒执行一次f函数的效果。


总结

通过对线程的全面解析我们了解到了线程的重要性,它可以将我们复杂的问题变得简单化,对于喜欢玩爬虫的小伙伴们可以说是相当有用了,本文基本覆盖了线程的所有概念,希望能帮到大家。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

93

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

106

2025.09.18

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

377

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

32

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

29

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

103

2026.02.06

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

99

2025.12.01

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 4.9万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号