参考:

python3多线程——官方教程中文版

python多线程-1

python多线程-2.1

python多线程-2.2

python多线程-3

python多线程-4


python中的多线程无法在多个核上同时进行,这是因为:

Python的线程虽然是真正的线程,但解释器执行代码时,有一个GIL锁:Global Interpreter Lock,任何Python线程执行前,必须先获得GIL锁,然后,每执行100条字节码,解释器就自动释放GIL锁,让别的线程有机会执行。这个GIL全局锁实际上把所有线程的执行代码都给上了锁,所以,多线程在Python中只能交替执行,即使100个线程跑在100核CPU上,也只能用到1个核。

GIL是Python解释器设计的历史遗留问题,通常我们用的解释器是官方实现的CPython,要真正利用多核,除非重写一个不带GIL的解释器。

所以,在Python中,可以使用多线程,但不要指望能有效利用多核。如果一定要通过多线程利用多核,那只能通过C扩展来实现,不过这样就失去了Python简单易用的特点。

不过,也不用过于担心,Python虽然不能利用多线程实现多核任务,但可以通过多进程实现多核任务。多个Python进程有各自独立的GIL锁,互不影响。


Thread类构造方法

Thread(group = None, target = None, name = None, args = (), kwargs = {})

  • group: 线程组,目前尚未实现,必须是None。

  • target:要执行的方法。

  • name:线程名。(默认是Thread-number)

  • args/kwargs:要传入方法的参数。

实例方法

  • isAlive():返回线程是否在运行。

  • get/setName:获取/设置线程名。

  • start():启动线程

  • is/setDaemon(bool):获取/设置是后台进程(默认是前台进程(False))。

    • 如果是后台进程,主线程执行过程中,后台线程也在进行,主线程执行完毕之后,后台线程不论成功与否,主线程和后台线程均停止。

    • 如果是前台线程,主线程执行过程中,前台线程也在执行,主线程执行完毕后,等待前台线程执行完成后,程序停止。

  • join():阻塞当前上下文环境的线程,知道调用此方法的线程终止或者到达指定的timeout(可选参数)。

创建线程的两种方法

  • 方法1:将想要执行的方法传给Thread类中的构造方法。

  • 方法2:定义一个新类继承Thread类,重写run方法。

# 创建线程的两种方法

# 第一种:将要执行的方法传给Thread的构造方法
import threading
import time def run(n):
time.sleep(1)
print(n) for i in range(5):
t = threading.Thread(target = run, args = (i,))
t.start() # 第二种,继承Thread类,重写run()方法 class Mythread(threading.Thread):
def __init__(self, arg):
threading.Thread.__init__(self)
# super().__init__(self)
self.arg = arg def run(self):
time.sleep(1)
print(self.arg) for i in range(5,10):
t = Mythread(i)
t.start()

setDeamon方法

设置是后台进程(True)还是前台进程(False)。(默认是False)

例题如下:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import threading
import time class Mythread(threading.Thread):
def __init__(self, arg, name):
threading.Thread.__init__(self, name=name)
# super().__init__(self)
self.arg = arg def run(self):
time.sleep(3)
print(self.arg) thread = Mythread(10, 'TestThread') # thread.setDaemon(True)
thread.start() for i in range(2):
print(threading.current_thread().name + ': ', i)
# 判断子线程是不是存活
print(thread.name + ': ', thread.isAlive())
time.sleep(1)

输出:

MainThread:  0
TestThread: True
MainThread: 1
TestThread: True
10

在线程启动前,设置:

thread.setDaemon(True)

输出:

MainThread:  0
TestThread: True
MainThread: 1
TestThread: True

可以看到,子线程并没有完全进行完毕,因为这时主线程已经结束了,子线程也要跟着结束。

join方法

阻塞线程上下文,直到调用此方法的线程终止或到达指定的timeout,即使设置了setDeamon(True)的主线程也要等待子线程结束。

实例如下:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import threading
import time class Mythread(threading.Thread):
def __init__(self, arg, name):
threading.Thread.__init__(self, name=name)
# super().__init__(self)
self.arg = arg def run(self):
time.sleep(1)
print(self.name + ': ', self.arg) thread = Mythread(10, 'TestThread') for i in range(5):
t = Mythread(i, 'Thread' + str(i))
t.start()
# t.join()

输出(比较乱,因为停顿1s后,所有的线程都在准备就绪的状态): ```python
Thread2: 2
Thread4: Thread1: Thread0: 0Thread3: 3 1
4

我们在每个线程的后面,用join方法进行阻塞,如下:

t.join()

输出:

Thread0:  0
Thread1: 1
Thread2: 2
Thread3: 3
Thread4: 4

Lock类、Rlock类

由于线程之间的随机调度:某个线程可能在执行n条后,CPU接着执行其他线程。为了防止多个线程同时操作一个内存资源时造成混乱,我们使用了锁。

例如,我们同时使用两个线程操作一个全局变量,如下:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import threading class Mythread(threading.Thread):
def __init__(self, name, i):
threading.Thread.__init__(self, name=name)
self.i = i def run(self): global count
for i in range(100000):
count -= self.i
count += self.i count = 10 t1 = Mythread('Thread' + str(1) + ': ', 100)
t2 = Mythread('Thread' + str(2) + ': ', 50)
t1.start()
t2.start()
t1.join()
t2.join() print('count = ', count)

在运行次数比较少的时候,可以输出正确答案,但是当循环次数变大之后,每次输出的结果就不一样了。

但是,当我们使用了Lock类之后,如下:

#!/usr/bin/env python
# -*- coding:utf-8 -*-
import threading
import time class Mythread(threading.Thread):
def __init__(self, name, i):
threading.Thread.__init__(self, name=name)
self.i = i def run(self): global count
for i in range(100000):
lock.acquire() # 获取锁
count -= self.i
count += self.i
lock.release() # 释放锁 count = 10
lock = threading.Lock()
t1 = Mythread('Thread' + str(1) + ': ', 100)
t2 = Mythread('Thread' + str(2) + ': ', 50)
t1.start()
t2.start()
t1.join()
t2.join() print('count = ', count)

Lock类与RLock类的区别。

这两种锁的主要区别是:在同一线程内,对RLock进行多次acquire()操作,程序不会阻塞。

  • threading.Lock() 加载线程的锁对象,是一个基本的锁对象,一次只能一个锁定,其余锁请求,需等待锁释放后才能获取

  • threading.RLock() 多重锁,在同一线程中可用被多次acquire。 如果使用RLock,那么acquire和release必须成对出现。

初识python多线程的更多相关文章

  1. Python导出Excel为Lua/Json/Xml实例教程(一):初识Python

    Python导出Excel为Lua/Json/Xml实例教程(一):初识Python 相关链接: Python导出Excel为Lua/Json/Xml实例教程(一):初识Python Python导出 ...

  2. Python开发【第一篇】:初识Python

    初识python 一.python简介 python的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏姆为了在阿姆斯特丹打发时间,决心开发一个新的脚本解 ...

  3. python多线程学习记录

    1.多线程的创建 import threading t = t.theading.Thread(target, args--) t.SetDeamon(True)//设置为守护进程 t.start() ...

  4. python多线程编程

    Python多线程编程中常用方法: 1.join()方法:如果一个线程或者在函数执行的过程中调用另一个线程,并且希望待其完成操作后才能执行,那么在调用线程的时就可以使用被调线程的join方法join( ...

  5. Python 多线程教程:并发与并行

    转载于: https://my.oschina.net/leejun2005/blog/398826 在批评Python的讨论中,常常说起Python多线程是多么的难用.还有人对 global int ...

  6. python多线程

    python多线程有两种用法,一种是在函数中使用,一种是放在类中使用 1.在函数中使用 定义空的线程列表 threads=[] 创建线程 t=threading.Thread(target=函数名,a ...

  7. Python开发【第二篇】:初识Python

    Python开发[第二篇]:初识Python   Python简介 Python前世今生 python的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏 ...

  8. python 多线程就这么简单(转)

    多线程和多进程是什么自行google补脑 对于python 多线程的理解,我花了很长时间,搜索的大部份文章都不够通俗易懂.所以,这里力图用简单的例子,让你对多线程有个初步的认识. 单线程 在好些年前的 ...

  9. python 多线程就这么简单(续)

    之前讲了多线程的一篇博客,感觉讲的意犹未尽,其实,多线程非常有意思.因为我们在使用电脑的过程中无时无刻都在多进程和多线程.我们可以接着之前的例子继续讲.请先看我的上一篇博客. python 多线程就这 ...

随机推荐

  1. LODOP打印维护适应不同的客户端

    之前的博文:Lodop打印设计.维护.预览.直接打印简单介绍,介绍了打印设计.打印维护.打印预览,直接打印等的区别和使用. 如上面以前博文描述的,打印维护是针对客户端进行调整的,开放打印维护给客户端, ...

  2. jenkins:新增节点是启动方式没有Launch agent by connecting it to the master

    默认在这里的配置是禁用 所以启动方式只有两种,缺少Launch agent by connecting it to the master

  3. Kubernetes 之 MySQL 持久存储和故障转移(十一)

    目录 一.规划 二.部署 1.创建 PV 和 PVC 2.部署 MySQL 3.更新 MySQL 数据 4.故障转移 一.规划 我们接着之前的文档的架构规划进行下面的操作. IP 角色 192.168 ...

  4. Python中IO编程-StringIO和BytesIO

    Python在内存中读写数据,用到的模块是StringIO和BytesIO StringIO >>> from io import StringIO >>> f = ...

  5. 将自定义jar包上传github并制作成maven仓库

    参照:https://www.jianshu.com/p/98a141701cc7 第一阶段 :配置github 1.创建mvn-repo分支     首先在你的github上创建一个maven-re ...

  6. EditPlus配置ftp连接linux

    选择文件/FTP下面的设置FTP服务器 1.点击添加 2.填写名称.ftp服务器.用户名.密码信息 3.点击高级设置 4.选择加密方式为sftp,端口22,如果不填端口号,默认也是22,确定 5.确定 ...

  7. [十一集训] Day1 (2018-2019 ACM-ICPC Nordic Collegiate Programming Contest (NCPC 2018))

    A Altruistic Amphibians 原题 题目大意: n只青蛙在高度为d的井中,每只有跳跃距离.重量和高度,每只青蛙可以借助跳到别的青蛙的背上而跳出井,每只青蛙能承受的最大重量是自身重量, ...

  8. Python中的条件判断、循环以及循环的终止

    条件判断 条件语句是用来判断给定条件是否满足,并根据判断所得结果从而决定所要执行的操作,通常的逻辑思路如下图: 单次判断 形式 if <判断条件>: <执行> else: &l ...

  9. 【LEETCODE】57、数组分类,适中级别,题目:969、442、695

    package y2019.Algorithm.array.medium; import java.util.ArrayList; import java.util.List; /** * @Proj ...

  10. VisualStudio 连接 MySql 实现增删查改

    首先创建数据库,建立一个用户登录表 2.visualStudio默认是不支持MySql的,要想通过Ado.Net 操作MySql 需要在管理NeGet包添加对MySql.Data  和 MySql.D ...