我的编程空间,编程开发者的网络收藏夹
学习永远不晚

web开发中的线程是怎样的

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

web开发中的线程是怎样的

web开发中的线程是怎样的,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

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

有人说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的全局超时时间

下面我们来看下吧:

web开发中的线程是怎样的

3.创建线程

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

一、使用Thread方法来创建:

1.单线程

def xc():     for y in range(100):         print('运行中'+str(y)) tt=t.Thread(target=xc,args=()) #方法加入到线程 tt.start()  #开始线程 tt.join() #等待子线程结束

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()  #等待子线程结束

二、重写线程的类方法

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() 和上面的效果是一样的

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.线程锁

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

web开发中的线程是怎样的

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

一、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()

二、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() #释放锁

三、条件锁

等待通过,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.信号量

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

一、有界信号量

它不允许使用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

web开发中的线程是怎样的

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

二、无界信号量

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

web开发中的线程是怎样的

可以看到虽然多了个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)

web开发中的线程是怎样的

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

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方法设为全局变量吗?我们来看下:

web开发中的线程是怎样的

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

8.Timer

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

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

下面我们来看下:

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

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

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注编程网行业资讯频道,感谢您对编程网的支持。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

web开发中的线程是怎样的

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

理想的Java Web开发框架是怎样的

今天就跟大家聊聊有关理想的Java Web开发框架是怎样的,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。理想的Java Web开发框架,应该有一个好的IDE开发工具,架构设计清晰简单
2023-06-17

web开发安全框架中的Apache Shiro的应用是怎样的

今天给大家介绍一下web开发安全框架中的Apache Shiro的应用是怎样的。文章的内容小编觉得不错,现在给大家分享一下,觉得有需要的朋友可以了解一下,希望对大家有所帮助,下面跟着小编的思路一起来阅读吧。web开发安全框架中的Apache
2023-06-02

Java多线程发展史是怎样的

本篇内容主要讲解“Java多线程发展史是怎样的”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java多线程发展史是怎样的”吧!JDK 1.01996年1月的JDK1.0版本,从一开始就确立了Ja
2023-06-17

基于Django Web开发的系统清单是怎样的

这篇文章给大家介绍基于Django Web开发的系统清单是怎样的,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一 简介 最近在和 同事 一起开发一套数据库管理平台 ,该平台使用Django 作为web 框架。和大
2023-06-04

web前端中弹窗是怎么开发的

这篇“web前端中弹窗是怎么开发的”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“web前端中弹窗是怎么开发的”文章吧。状态驱
2023-07-05

自定义用于Web开发的开源PHP框架Codeigniter是怎么样的

自定义用于Web开发的开源PHP框架Codeigniter是怎么样的,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Codeigniter 是一个 PHP 框架,可以使公司进行开
2023-06-16

web开发中怎么去掉内联样式

小编给大家分享一下web开发中怎么去掉内联样式,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!经常修改magento的时候,有遇到element.style。 这个
2023-06-08

jvm线程变化是怎样的

这篇文章主要介绍“jvm线程变化是怎样的”,在日常操作中,相信很多人在jvm线程变化是怎样的问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”jvm线程变化是怎样的”的疑惑有所帮助!接下来,请跟着小编一起来学习吧
2023-06-03

编程热搜

目录