线程的开启方法

进程是操作系统调度的最小单位,一个进程最少有一个主线程,而一个进程中可以开启多个线程

from threading import Thread

def task():
print('A child thread') if __name__ == '__main__': # 这里是可以不用写的,但是规范起见还是写了
t = Thread(target=task)
t.start()
print('==>main thread')

GIL锁

"""

 1. Python在设计之初就考虑到要在主循环中,同时只有一个线程在执行
2. Python 解释器中可以“运行”多个线程,但在任意时刻只有一个线程在解释器中运行。 1、GIL锁它是在python解释器中的, 只在cpython中有, pypy解释器是没有GIL锁的,
2、起一个垃圾回收线程, 一个是正常执行的线程
3、设置了一把锁(GIL锁), 一个线程要想执行,必须拿到这把锁
4、同一时刻,开启一个进程,一个进程中可以有多个线程, 只能有一个线程在执行 5、如果是计算密集型:要开进程
6、如果是io密集型:要开线程
"""

进程与线程的比较

线程与进程的区别可以归纳为以下4点:

  1. 地址空间和其它资源(如打开文件):进程间相互独立,同一进程的各线程间共享。某进程内的线程在其它进程不可见。
  2. 通信:进程间通信IPC,线程间可以直接读写进程数据段(如全局变量)来进行通信——需要进程同步和互斥手段的辅助,以保证数据的一致性。
  3. 调度和切换:线程上下文切换比进程上下文切换要快得多。
  4. 在多线程操作系统中,进程不是一个可执行的实体。

进程消耗的资源比线程要大的多,相对的,调用起来也比线程要慢

# 进程调用时间演示
import time
from multiprocessing import Process def task():
time.sleep(1)
print('A child process') if __name__ == '__main__':
start = time.time()
p = Process(target=task)
p.start()
p.join()
print('master process')
print(time.time() - start) # 输出结果
# A child process
# master process
# 1.069324016571045
# 线程调用时间演示
import time
from threading import Thread def task():
time.sleep(1)
print('A child thread') if __name__ == '__main__':
start = time.time()
t = Thread(target=task)
t.start()
t.join()
print('==>main thread')
print(time.time()-start)
# 输出结果
# A child thread
# ==>main thread
# 1.005803108215332

线程间的数据是共享的

def task():
global n
n=0 if __name__ == '__main__':
n=1
t=Thread(target=task)
t.start()
t.join()
print('主',n) # 查看结果为0,因为同一进程内的线程之间共享进程内的数据

Thread类的方法

t.is_alive() # 返回线程是否活动
t.getName() # 获取线程名
t.setName() # 更新线程名
threading.currentThread() # 返回当前的线程变量
threading.enumerate() # 返回一个包含正在运行的线程的list。正在运行指线程启动后、结束前,不包括启动前和终止后的线程
threading.activeCount() # 返回正在运行的线程数量,与len(threading.enumerate())有相同的结果

守护线程

from threading import Thread
import time
def task():
time.sleep(1)
print("我是子线程") if __name__ == '__main__':
t = Thread(target=task,)
t.setDaemon(True) # 开启守护线程, 主线程结束,子线程跟着结束
t.start() print("主线程")

线程互斥锁

from threading import Thread,Lock
import time
import random def task(lock):
#上锁
lock.acquire()
global n
# 10个线程同时涌入导致数据不安全
time.sleep(random.random())
n -= 1
# 释放锁
lock.release() if __name__ == '__main__':
start = time.time()
lock = Lock()
n = 100
l = []
for i in range(10):
t = Thread(target=task,args=(lock,))
t.start()
l.append(t)
for j in l:
j.join()
print('运算完毕:n = %s' %n)
print(time.time()-start)

信号量

from threading import Thread,Semaphore
import time def task(i, sm):
# 上锁
sm.acquire()
print("%s:这个人开始上厕所了" % i)
time.sleep(3)
print("%s:这个人上完了" % i)
# 释放锁
sm.release() # Semaphore:信号量可以理解为多把锁,同时允许多个线程来更改数据
if __name__ == '__main__':
sm = Semaphore(2) #
for i in range(10):
t = Thread(target=task, args=(i, sm))
t.start()

Event事件

from threading import Thread, Event

import time

def girl(event):
print("都女士正在恋爱中...")
time.sleep(3)
print("都女生分手了") # 发出分手的信号
event.set() def boy(i, event):
print("渣男:%s正在等待都女生分手" % i)
# 卡住
event.wait()
print("渣男:%s开始追了" % i) if __name__ == '__main__':
event = Event() t = Thread(target=girl, args=(event,))
t.start() for i in range(10):
b_t = Thread(target=boy, args=(i, event))
b_t.start()

释放出信号,而读数据课以收信号,一旦示范出可以读的信号该线程才会运行

线程 Thread类 GIL锁 信号量 Event事件的更多相关文章

  1. 并发编程---死锁||递归锁---信号量---Event事件---定时器

    死锁 互斥锁:Lock(),互斥锁只能acquire一次 递归锁:  RLock(),可以连续acquire多次,每acquire一次计数器+1,只有计数为0时,才能被抢到acquire # 死锁 f ...

  2. python并发编程-多线程实现服务端并发-GIL全局解释器锁-验证python多线程是否有用-死锁-递归锁-信号量-Event事件-线程结合队列-03

    目录 结合多线程实现服务端并发(不用socketserver模块) 服务端代码 客户端代码 CIL全局解释器锁****** 可能被问到的两个判断 与普通互斥锁的区别 验证python的多线程是否有用需 ...

  3. 8.14 day32 TCP服务端并发 GIL解释器锁 python多线程是否有用 死锁与递归锁 信号量event事件线程q

    TCP服务端支持并发 解决方式:开多线程 服务端 基础版 import socket """ 服务端 1.要有固定的IP和PORT 2.24小时不间断提供服务 3.能够支 ...

  4. GIL全局解释器锁-死锁与递归锁-信号量-event事件

    一.全局解释器锁GIL: 官方的解释:掌握概念为主 """ In CPython, the global interpreter lock, or GIL, is a m ...

  5. [并发编程 - 多线程:信号量、死锁与递归锁、时间Event、定时器Timer、线程队列、GIL锁]

    [并发编程 - 多线程:信号量.死锁与递归锁.时间Event.定时器Timer.线程队列.GIL锁] 信号量 信号量Semaphore:管理一个内置的计数器 每当调用acquire()时内置计数器-1 ...

  6. 递归锁,event事件和信号量

    锁通常被用来实现对共享资源的同步访问.为每一个共享资源创建一个Lock对象,当你需要访问该资源时,调用acquire方法来获取锁对象(如果其它线程已经获得了该锁,则当前线程需等待其被释放),待资源访问 ...

  7. GIL 信号量 event事件 线程queue

    GIL全局解释器锁 官方解释: In CPython, the global interpreter lock, or GIL, is a mutex that prevents multiple n ...

  8. python基础--GIL全局解释器锁、Event事件、信号量、死锁、递归锁

    ps:python解释器有很多种,最常见的就是C python解释器 GIL全局解释器锁: GIL本质上是一把互斥锁:将并发变成串行,牺牲效率保证了数据的安全 用来阻止同一个进程下的多个线程的同时执行 ...

  9. python开发线程:死锁和递归锁&信号量&定时器&线程queue&事件evevt

    一 死锁现象与递归锁 进程也有死锁与递归锁,在进程那里忘记说了,放到这里一切说了额 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将 ...

随机推荐

  1. 11 shell中内置关键字[[]]:检查条件是否成立

    0.[[]] [] (())的对比 1.[[]]支持正则表达式 0.[[]] [] (())的对比 [[ ]] 对数字的比较不友好,所以使用 if 判断条件时,建议用(())来处理整型数字,用[[]] ...

  2. linux学习之路第六天(文件目录类第二部分)

    文件目录类 1.cat指令 作用:查看文件内容,是以只读的方式打开. 基本语法 cat [选项] 要查看的文件 常用选项 -n; 使用细节: cat只能浏览文件,而不能修改文件,通常会和more一起使 ...

  3. ESP32-默认事件循环

    默认的事件循环是一个事件循环的系统应用发布和处理事件(例如,Wi-Fi无线事件). 基于ESP-IDF4.1 1 #include "esp_log.h" 2 #include & ...

  4. Djiango 连接数据库mysql 的基本数据操作

    1.单表操作 (1) .增加操作 1 # 增加操作 2 def add_person(request): 3 # 1.使用save() 方法 4 # person_obj = models.Perso ...

  5. MYSQL_Join注入技巧

    Join注入技巧 join无名列报错注入 约束条件 在知到表名的前提下才能操作 注入语句 and extractvalue(1,concat(0x7e,(select * from (select * ...

  6. 使用Nginx和uwsgi部署Flask项目

    前言   之前用Flask框架开发了一个Python的Web项目,使用Nginx和uWSGI部署起来感觉挺麻烦,过程中还因为对Flask框架的不熟悉,花了好长时间才把应用完全部署起来.下面分享部署成功 ...

  7. 从redis数据库取数据存放到本地mysql数据库

    redis数据库属于非关系型数据库,数据存放在内存堆栈中,效率比较高. 其存储数据是以json格式字符串存储字典的,而类似的关系型数据库无法实现这种数据的存储. 在爬取数据时,将数据暂存到redis中 ...

  8. (JAVA1)学习,复习的好帮手

    学习准备:博客 博客,英文名Blog,它的正式名称为网络日记. 为什么要写博客 我们学习任何东西都需要总结和思考.有时候我们一直在赶路,却忘了放慢脚步 提升文笔组织能力 提升学习总结能力 提升逻辑思维 ...

  9. Gitea 1.4.0 目录穿越导致命令执行漏洞

    复现 POST /vulhub/repo.git/info/lfs/objects HTTP/1.1 Host: 192.168.49.2:3000 Accept-Encoding: gzip, de ...

  10. React Hooks的理解

    一.是什么 Hook 是 React 16.8 的新增特性.它可以让你在不编写 class 的情况下使用 state 以及其他的 React 特性 至于为什么引入hook,官方给出的动机是解决长时间使 ...