手把手教你做项目多线程篇——基础知识详解(代码片段)

肥学大师 肥学大师     2022-12-08     105

关键词:

导读

在这里插入图片描述
随着暑假的推进,手把手教你做项目前边的准备也差不多了后续的项目也渐渐要开始了但是正式发出来可能要等一段时间前后端都是我一个人确实有点费力(毕竟我也是菜鸡),后面我在想想怎么提速吧嘻嘻 😃 详情请大家关注请大家关注手把手教你做项目专栏里面有更多资源等你来探索哦。好了我们今天我们照常学习新知识为以后的项目做铺垫,老规矩喊出我们的口号:“不肥身体,肥学问”来吧开始。

项目中多线程的目的

具体原理我觉得这应该是可以跳到操作系统里面了一点点小小总结大家还可以找本操作系统的书看看,老规矩找不到的话找我主页,有一个关于资源的文章里面有资源当然也可以私信我。
我们再说说我们项目里面用多线程的目的:
1.线程之间共享内存非常容易。
2·使用多线程来实现多任务并发执行比使用多进程的效率高
3·有时候可以节省运行时间,这个一会在下面就会知道
4·当你一个文件要同时执行多个功能时就可以用到多线程

python语言内置了多线程功能支持,而不是单纯地作为底层操作系统的调度方式,从而简化了python的多线程编程。

实战操作

说这么多不如实际动手练练,首先导入线程
特别注意:大家在见建文件的时候名字千万别和导入的包threading一样不然会出错的。

小知识

import threading

让我们先看看自己程序现在有多少个进程

import threading
def main():
    print(threading.current_thread())

if __name__ == '__main__':
    main()
结果:1#我的就一个你的呢?

如果你的进程不为一的话还可以这样查看每一个进程名

import threading
def main():
    print(threading.active_count())
    print(threading.enumerate())
   

if __name__ == '__main__':
    main()
>>>[<_MainThread(MainThread, started 36004)>]#返回的是一个列表因为我的目前就一个所以就一个主进程

还可以查看正在运行的进程

import threading
def main():
    print(threading.active_count())
    print(threading.enumerate())
    print(threading.current_thread())

if __name__ == '__main__':
    main()
>1
[<_MainThread(MainThread, started 36004)>]
<_MainThread(MainThread, started 36004)>

创建一个简单的线程

首先我们先介绍一下threading.Thread()里面的参数,大家学python每个模块的功能时最好还是看一下源文件内容,这样有助于提高你的编程能力:
在这里插入图片描述

需要注意的点我已经打上标记了

import threading
def first():
    print("frist active")
    print("frist finish")

def main():
    first_thread=threading.Thread(target=first,name="T1")
    first_thread.start()#开始的标志
    print("main")

if __name__ == '__main__':
    main()
结果:
第一次运行
frist active
main
frist finish
第二次运行
frist active
frist finish
main

每次的结果不一样就已经表明Frist和main是同时运行的了。
如果说效果不太明显的话,我们改进一下接下来我们引入

import time
import threading
import time
def first():
    print("frist active")
    time.sleep(3)
    print("frist finish")

def main():
    first_thread=threading.Thread(target=first,name="T1")
    first_thread.start()
    print("main")

if __name__ == '__main__':
    main()
结果:
frist active
main
frist finish

因为执行到Frist active的时候Frist线程要睡3秒这个时候main还在执行所以这样每次都是这个结果了。
特别强调target=first不是导入Frist函数从源文件我们就已经看出是通过run()方法进行的,这里解释我引用一位大佬解释
在这里插入图片描述

链接
这位大佬大家应该都熟悉,就是顶顶大名的雷学委各位大佬可以去他的主页看看可能会学到些新知识雷学委
当然如果你觉得这样不行的话你也可以重写threading.Thresd里的run方法来个自定义线程

    class MyThread(threading.Thread):
        def __init__(self,n):
            super(MyThread,self).__init__()   #重构run函数必须写
            self.n = n

        def run(self):
            print('task',self.n)
            time.sleep(1)
            print('2s')
            time.sleep(1)
            print('1s')
            time.sleep(1)
            print('0s')
            time.sleep(1)

    if __name__ == '__main__':
        t1 = MyThread('t1')
        t2 = MyThread('t2')
        t1.start()
        t2.start()
结果:
task t1
task t2
2s
2s
1s
1s
0s
0s

守护线程

所谓’线程守护’,就是主线程不管该线程的执行情况,只要是其他子线程结束且主线程执行完毕,主线程都会关闭。也就是说:主线程不等待该守护线程的执行完再去关闭。

不好理解的话来看看例子

import threading
import time
def first():
    print("frist active")
    time.sleep(3)
    print("frist finish")

def second():
    print("second active")
    print("second finish")

def main():
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    first_thread.setDaemon(True)#一定要在start()前开始
    first_thread.start()
    second_thresd.start()
    print("main")

if __name__ == '__main__':
    main()
    

结果
frist active
second active
second finishjiemeijieshu
main

当主线程和其他子线程都结束不管守护线程first_thread 结没结束程序都结束
当设second_thresd为守护线程的时候情况是这样的

import threading
import time
def first():
    print("frist active")
    time.sleep(3)
    print("frist finish")

def second():
    print("second active")
    print("second finish")

def main():
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    second_thresd.setDaemon(True)#一定要在start()前开始
    first_thread.start()
    second_thresd.start()
    print("main")

if __name__ == '__main__':
    main()
    


frist active
second active
second finish
main
frist finish #尽管输出这个要等三秒

主进程等待子进程结束

为了让守护线程执行结束之后,主线程再结束,我们可以使用join方法,让主线程等待子线程执行

import threading
import time
def first():
    print("frist active")
    time.sleep(3)
    print("frist finish")

def second():
    print("second active")
    print("second finish")

def main():
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    first_thread.start()
    second_thresd.start()
    first_thread.join()

    print("main")

if __name__ == '__main__':
    main()
    


结果:
frist active
second active
second finish
frist finish
main
不加join是这样的结果
frist active
second active
second finish
main
frist finish

共享全局变量的特性

这里定义一个全局变量A

import threading
import time
def first():
    global A
    print("frist active")
    time.sleep(3)
    A=A+3
    print("frist:%d"%A)
    print("frist finish")

def second():
    global A
    print("second active")
    A=A+6
    print("second:%d"%A)
    print("second finish")

def main():
    global A
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    first_thread.start()
    second_thresd.start()
    #first_thread.join()

    print("main")
    A=A+3
    print("mian:%d"%A)

if __name__ == '__main__':
    A=0
    main()




来看一下结果

frist active
second active
second:6
second finish
main
mian:9
frist:12
frist finish

由上面的例子可以看出,输出A的值的时候不同进程之间的资源是共享的这就导致了变量A的值不固定造成了脏数据的情况,不理解的话我们就来个例子。
在这里插入图片描述

在没有互斥锁的情况下,假设账户有一万元钱,存钱和取钱同时进行可能账户余额会有一万一千元。这样我当然高兴只是银行不答应。为了避免这种情况我们引入锁的概念,下面我们简单的介绍几种编程里面常用的。

互斥锁

import threading
import time
def first():
    global A,lock
    lock.acquire()
    print("frist active")
    time.sleep(3)
    A=A+3
    print("frist:%d"%A)
    print("frist finish")
    lock.release()

def second():
    global A,lock
    lock.acquire()
    print("second active")
    A=A+6
    print("second:%d"%A)
    print("second finish")
    lock.release

def main():
    global A,lock
    lock=threading.Lock()
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    first_thread.start()
    second_thresd.start()
    #first_thread.join()

    print("main")
    A=A+3
    print("mian:%d"%A)

if __name__ == '__main__':
    A=0
    main()

结果

frist active
main
mian:3
frist:6
frist finish
second active
second:12
second finish

是不是这样看着就舒服多了,如果例子不够明显我们再来一个

import threading
import time
def first():
    global A,lock
    lock.acquire()
    print("frist active")
    time.sleep(3)
    A=A+3
    print("frist1:%d"%A)
    A = A + 3
    print("frist2:%d" % A)
    print("frist finish")
    lock.release()

def second():
    global A,lock
    lock.acquire()
    print("second active")
    A=A+6
    print("second1:%d"%A)
    A=A+6
    print("second2:%d"%A)

    print("second finish")
    lock.release()

def main():
    global A,lock
    lock=threading.Lock()
    first_thread=threading.Thread(target=first,name="T1")
    second_thresd=threading.Thread(target=second,name="T2")
    first_thread.start()
    second_thresd.start()
    #first_thread.join()

    print("main")
    A=A+3
    print("mian:%d"%A)

if __name__ == '__main__':
    A=0
    main()

结果

frist active
main
mian:3
frist1:6
frist2:9
frist finish
second active
second1:15
second2:21
second finish

去掉锁以后结果

frist active
second active
second1:6
second2:12
second finish
main
mian:15
frist1:18
frist2:21
frist finish

很明显去掉锁以后结果杂乱的很

信号量

我相信对操作系统有一定了解的肯定,在刚才提到锁的时候就想到了信号量毕竟考研题经常会出现,同步,互斥和信号量机制。我们就来说一说信号量锁,其实道理很简单假如你现在,在中国结婚了你只能娶一个老婆吧,尽管你可以去找别的女人但他们不能称为老婆她们会被称为小三,二奶啊等等,这里老婆这个信号量在中国就是只能有一个,别的再来就不可以了。

import threading
import time

def run(n,semaphore):
    semaphore.acquire()   #加锁
    time.sleep(3)
    print('run the thread:%s\\n' % n)
    semaphore.release()    #释放


if __name__== '__main__':
    num=0
    semaphore = threading.BoundedSemaphore(3)   #最多允许3个线程同时运行
    for i in range(10):
        t = threading.Thread(target=run,args=('t-%s' % i,semaphore))
        t.start()
    while threading.active_count() !=1:
        pass
    else:
        print('----------all threads done-----------')

结果

run the thread:t-2
run the thread:t-1


run the thread:t-0

run the thread:t-3
run the thread:t-5
run the thread:t-4



run the thread:t-6

run the thread:t-7

run the thread:t-8

run the thread:t-9

----------all threads done-----------

送点资源

肥学觉得python里面这些操作还是有点简单的,如果各位大佬想继续深究线程又没有资源的话可以我这里有一本关于操作系统的书,领取可以私信我哦,好了今天的学习就到这里吧别忘了点赞三联哦

手把手教你做项目mysql篇——从下载到命令总结(代码片段)

目录导读下载MySQL篇基础命令篇操作前的提示登录篇对数据库命令操作篇对表的操作删除表(这个单独说一下)对数据操作篇结尾送书导读有段时间没有好好整理文章了,这不是又该暑假啦嘛肥学准备在暑假来波大的&... 查看详情

c#多线程(16):手把手教你撸一个工作流(代码片段)

...注入实现工作流解析前言前面学习了很多多线程和任务的基础知识,这里要来实践一下啦。通过本篇教程,你可以写出一个简单的工作流引擎。本篇教程内容完成是基于任务的,只需要看过笔者的三篇关于异步的文章,掌握C#基... 查看详情

《手把手教你》系列基础篇(七十二)-java+selenium自动化测试-框架设计基础-testng简单介绍(详解教程)(代码片段)

1.简介前面文章细心的小伙伴会发现宏哥在运行测试用例的时候有的是在main方法下,而有的不需要用main方法去执行用例,那么为什么有的就不需要在main方法下就能够成功运行测试用例了。这就需要单元测试框架的支持,这篇宏... 查看详情

手把手教你做一个缓存工具(代码片段)

日常开发中,某些数据接口即使优化到极致,都难免还会存在计算量巨大导致响应过慢,多数情况单独做一个统计表用于存放这些处理后的数据用于读取,或者接入redis/memcache存数据,就是说单次响应本身是可以接受较慢一些的... 查看详情

csdn博客太火了也教你做一个——(期末web大作业)(代码片段)

...前端知识介绍最后对肥友说资料领取和详解展示整个项目手把手教你开发一个博客系统-2021-10-1408:26:10进入巨幕主面板左边给大家留出以下空间等学会了以后可以自己加一下图标或者展示一些其他功能。导读不光是这次博客系统&#... 查看详情

《手把手教你》系列基础篇(七十三)-java+selenium自动化测试-框架设计基础-testng实现启动不同浏览器(详解教程)(代码片段)

1.简介上一篇文章中,从TestNg的特点我们知道支持变量,那么我们这一篇就通过变量参数来启动不同的浏览器进行自动化测试。那么如何实现同时启动不同的浏览器对脚本进行测试,且听我娓娓道来。2.项目实战2.1创... 查看详情

手把手教你做ios逆向分析,突破微信的群发多选数量限制(代码片段)

👇👇关注后回复 “进群” ,拉你进程序员交流群👇👇作者丨大帅老猿转自:掘金 https://juejin.cn/post/7006067681350647844前言很久没碰iOS开发了,最近都在web前端持续输出,加了很多推文群,每... 查看详情

一基础折线图详解《手把手教你echarts数据可视化详解》(代码片段)

注:本系列教程需要对应JavaScript、html、css基础,否则将会导致阅读时困难,本教程将会从ECharts的官方示例出发,详解每一个示例实现,从中学习ECharts。ECharts官方示例:https://echarts.apache.org/examples/zh/index.html一、折线图我们打开... 查看详情

一基础折线图详解《手把手教你echarts数据可视化详解》(代码片段)

注:本系列教程需要对应JavaScript、html、css基础,否则将会导致阅读时困难,本教程将会从ECharts的官方示例出发,详解每一个示例实现,从中学习ECharts。ECharts官方示例:https://echarts.apache.org/examples/zh/index.h... 查看详情

一基础折线图详解《手把手教你echarts数据可视化详解》(代码片段)

注:本系列教程需要对应JavaScript、html、css基础,否则将会导致阅读时困难,本教程将会从ECharts的官方示例出发,详解每一个示例实现,从中学习ECharts。ECharts官方示例:https://echarts.apache.org/examples/zh/index.h... 查看详情

手把手教你做一个安卓点餐系统(代码片段)

最近有小伙伴要学习安卓,今天给大家展示一个安卓系统的开发,希望能帮到你学习!传统的点餐方式有:1)手工点餐模式。在这种模式下,流程简单,店员很快知道要点什么,但随着点餐人数多&... 查看详情

手把手教你做视频播放器-设计方案(代码片段)

前言通过“计算器”应用我们已经熟悉了安卓应用开发的大致流程,具备了开发的初步知识。接下来,我们将开始制作一个“视频播放器”应用,进一步加深对程序开发的学习。当完成这个“视频播放器”应用后࿰... 查看详情

二基础平滑面积折线图与折线堆叠面积堆叠《手把手教你echarts数据可视化详解》(代码片段)

...ECharts官方示例:https://echarts.apache.org/examples/zh/index.html《手把手教你ECharts数据可视化详解》目录一、基础折线图详解一、平滑的折线图在上一节的折线图中,我 查看详情

三教你搞懂渐变堆叠面积图《手把手教你echarts数据可视化详解》(代码片段)

...ECharts官方示例:https://echarts.apache.org/examples/zh/index.html《手把手教你ECharts数据可视化详解》目录一、基础折线图详解二、基础平滑、面积折线图与折线堆叠、面 查看详情

手把手教你做一个电子相册(代码片段)

1、概述首先介绍下什么是HaaS:2020年9月17日,在云栖大会上阿里云IoT团队正式发布了HaaS(HardwareasaServie)。HaaS是一种物联网设备云端一体开发框架,它的目的是通过数量收敛的硬件积木(比如主控板:... 查看详情

手把手教你做一个网页(代码片段)

摘要:搞嵌入式的要学习一点前端吗?那么前端是什么?是网页是网站吗?是也不全是。前端技术一般分为前端设计和前端开发,前端设计一般可以理解为网站的视觉设计,前端开发则是网站的前台代码实... 查看详情

手把手教你做一个网页(代码片段)

摘要:搞嵌入式的要学习一点前端吗?那么前端是什么?是网页是网站吗?是也不全是。前端技术一般分为前端设计和前端开发,前端设计一般可以理解为网站的视觉设计,前端开发则是网站的前台代码实... 查看详情

二基础平滑面积折线图与折线堆叠面积堆叠《手把手教你echarts数据可视化详解》(代码片段)

...rts官方示例:https://echarts.apache.org/examples/zh/index.html《手把手教你ECh 查看详情