【跟我一起学Python吧】Python 多线程
其实自我感觉Python的多线程很类似于Java的多线程机制,但是比JAVA的多线程更灵活。在早期的Python多线程实现中,采用了thread模块。例如:
- from time import ctime,sleep
- from thread import start_new_thread
- def loop1():
- print "enter loop1:",ctime();
- sleep(3);
- print "leave loop1:",ctime();
- def loop2():
- print "enter loop2:",ctime();
- sleep(5);
- print "leave loop2:",ctime();
- def main():
- print "main begin:",ctime();
- start_new_thread(loop1, ());
- start_new_thread(loop2,());
- sleep(8);
- print "main end:",ctime();
- if __name__=="__main__":
- main();
简单介绍下这个代码块中的函数功能,sleep是线程睡眠时间,几乎等价于JAVA中的Thread.sleep(millionseconds)
start_new_thread是实例化一个线程并运行的方法,方法的第一个参数接受一个线程运行时所执行的函数对象,第二个参数是方法执行时所需要的参数,以一个元组的形式传入。
这大概是最早期的Python多线程实现了,注意代码中的main线程里的sleep(8)。这里的睡眠时间只能比3+5大,而不能小。如果小于这个时间,那么main主线程会提前退出,导致无论其子线程是否是后台线程,都将会中断,从而抛出线程中断异常,类似于Java的ThreadInterruptException。这个致命的影响几乎就是这个模块后期被抛弃的罪魁祸首。
当然在早期的Python多线程中,你可以利用加锁的机制来避免出现这个情况。稍微改动下以上代码:
- import thread;
- from time import sleep,ctime;
- from random import choice
- #The first param means the thread number
- #The second param means how long it sleep
- #The third param means the Lock
- def loop(nloop,sec,lock):
- print "Thread ",nloop," start and will sleep ",sec;
- sleep(sec);
- print "Thread ",nloop," end ",sec;
- lock.release();
- def main():
- seconds=[4,2];
- locks=[];
- for i in range(len(seconds)) :
- lock=thread.allocate_lock();
- lock.acquire();
- locks.append(lock);
- print "main Thread begins:",ctime();
- for i,lock in enumerate(locks):
- thread.start_new_thread(loop,(i,choice(seconds),lock));
- for lock in locks :
- while lock.locked() :
- pass;
- print "main Thread ends:",ctime();
- if __name__=="__main__" :
- main();
这里对Python线程运行时加入了锁监控机制,介绍下红色字体标志的几个方法(其实红色字体中的lock实质是thread.lockType实例。 ):
从以上介绍可以看出这个Lock类非常类似于JDK5.0中的java.util.concurrent.locks.Lock。不知道Doug Lea有没有参与这个模块的开发,哈哈~~(纯属YY),只是比JAVA中的LOCK类多了一个方法locked,用于检测Lock对象是否还处于加锁的状态。
所以上一个例子的工作原理就是在启动线程的时候,给每个线程都加了一把锁,直到线程运行介绍,再释放这个锁。同时在Python的main线程中用一个while循环来不停的判断每个线程锁已释放。这个方法虽然避免了最开始的例子中人为的时间控制,但是还不方便,高效。
所以在较新的Python版本中,都推荐使用threading模块。
看下threading模块的API,有过JAVA开发经验的会发现它和java.lang.Thread类非常接近。这里需要说的一点就是threading的run方法可以返回函数值,这点在用于跟踪和判断线程运行正常与否非常有作用。
threading模块支持三种方法来创建线程。而前两种方式都与其Thread类有关。看下它的简要说明:
- class Thread(_Verbose) :
- __init__(self, group=None, target=None, name=None, args=(), kwargs=None, verbose=None)
其中target指的是一个具体的函数,或者可调用的类实例(这里指实现了__call__方法的类实例)
第一种方法:指定线程运行的时候调用的函数。举例如下:
- from time import ctime,sleep
- import threading;
- from random import choice
- def loop(number,sec):
- print "Thread ",number," begins and will sleep ",sec," at ",ctime();
- sleep(sec);
- print "Thread ",number,"ends at ",ctime();
- def main():
- seconds=[2,4];
- threads=[];
- array=range(len(seconds));
- for i in array :
- t=threading.Thread(target=loop,args=(i,choice(seconds)));
- threads.append(t);
- print "main Thread begins at ",ctime();
- for t in threads :
- t.start();
- for t in threads :
- t.join();
- print "main Thread ends at ",ctime();
- if __name__=="__main__" :
- main();
从图上可以看出,这里target指向了一个具体的函数对象,而args传入了该方法调用时所必须的参数。这里传入了一个随即的睡眠时间。其中thread.join表示要等待该线程终止,和java中的Thread.join(long millionseconds)作用一样,如果不指定具体的时间的话,将会一直等待下去。
第二种方法就是指定一个可调用的类实例,实际上与前面一种非常的接近。如下所示:
- from time import ctime,sleep
- import threading;
- from random import choice
- class ThreadFunc(object):
- def __init__(self,func,args,name):
- self.func=func;
- self.args=args;
- self.name=name;
- def __call__(self):
- self.func(*self.args);
- def loop(number,sec):
- print "Thread ",number," begins and will sleep ",sec," at ",ctime();
- sleep(sec);
- print "Thread ",number,"ends at ",ctime();
- def main():
- seconds=[2,4];
- threads=[];
- array=range(len(seconds));
- for i in array :
- t=threading.Thread(target=ThreadFunc(loop,(i,choice(seconds)),loop.__name__));
- threads.append(t);
- print "main Thread begins at ",ctime();
- for t in threads :
- t.start();
- for t in threads :
- t.join();
- print "main Thread ends at ",ctime();
- if __name__=="__main__" :
- main();
这里只是将target指向从一个函数对象变成了一个可调用的类实例。
重点推荐下第三种方式,用继承threading.Thread的方式来实现线程,有过Java多线程应用的朋友一定会对下面的例子非常熟悉。
- from time import ctime,sleep
- import threading;
- from random import choice
- class MyThread(threading.Thread):
- def __init__(self,func,args,name):
- super(MyThread,self).__init__();
- self.func=func;
- self.args=args;
- self.name=name;
- def run(self):
- self.result=self.func(*self.args);
- def getResult(self):
- return self.result;
- def loop(number,sec):
- print "Thread ",number," begins and will sleep ",sec," at ",ctime();
- sleep(sec);
- print "Thread ",number,"ends at ",ctime();
- def main():
- seconds=[2,4];
- threads=[];
- array=range(len(seconds));
- for i in array :
- t=MyThread(loop,(i,choice(seconds)),loop.__name__);
- threads.append(t);
- print "main Thread begins at ",ctime();
- for t in threads :
- t.start();
- for t in threads :
- t.join();
- print "main Thread ends at ",ctime();
- if __name__=="__main__" :
- main();
从上面可以看出MyThread继承了threading.Thread类,并在初始化方法中执行了必要的参数赋值。值得注意的是在Java类的继承中,如果不显示的指定调用父类的构造方法,那么默认将调用父类的无参构造方法。而在Python中,就不会主动去调用。所以这里需要显示的调用父类的初始化方法。
推荐一个网站:程序人生
【跟我一起学Python吧】Python 多线程的更多相关文章
- Python爬虫之多线程
详情点我跳转 关注公众号"轻松学编程"了解更多. 多线程 在介绍Python中的线程之前,先明确一个问题,Python中的多线程是假的多线程! 为什么这么说,我们先明确一个概念,全 ...
- python高级之多线程
python高级之多线程 本节内容 线程与进程定义及区别 python全局解释器锁 线程的定义及使用 互斥锁 线程死锁和递归锁 条件变量同步(Condition) 同步条件(Event) 信号量 队列 ...
- python 类变量 在多线程下的共享与释放问题
最近被多线程给坑了下,没意识到类变量在多线程下是共享的,还有一个就是没意识到 内存释放问题,导致越累越大 1.python 类变量 在多线程情况 下的 是共享的 2.python 类变量 在多线程情况 ...
- python中的多线程【转】
转载自: http://c4fun.cn/blog/2014/05/06/python-threading/ python中关于多线程的操作可以使用thread和threading模块来实现,其中th ...
- Python之FTP多线程下载文件之分块多线程文件合并
Python之FTP多线程下载文件之分块多线程文件合并 欢迎大家阅读Python之FTP多线程下载系列之二:Python之FTP多线程下载文件之分块多线程文件合并,本系列的第一篇:Python之FTP ...
- Python之FTP多线程下载文件之多线程分块下载文件
Python之FTP多线程下载文件之多线程分块下载文件 Python中的ftplib模块用于对FTP的相关操作,常见的如下载,上传等.使用python从FTP下载较大的文件时,往往比较耗时,如何提高从 ...
- Python系列之多线程、多进程
线程是操作系统直接支持的执行单元,因此,高级语言通常都内置多线程的支持,Python也不例外,并且,Python的线程是真正的Posix Thread,而不是模拟出来的线程. Python的标准库提供 ...
- Python 简单理解多线程
进程,是一个或多个线程的集合,每个进程在内存中是相对独立的. 线程,是计算机最小的运算单元,每个进程至少要有一个线程,多个线程时,每个线程间之间共享内存. 分别举例常规运行和多线程运行: 0)常规运行 ...
- python中的多线程
一个程序可以理解为一个进程,这个进程有其代号,可以依据这个代号将其杀死. 一个进程肯定有且只有一个主线程,他可以有很多子线程. 运行一个任务如果可以有许多子线程同时去做,当然会提高效率. 但是,在py ...
- python单线程,多线程和协程速度对比
在某些应用场景下,想要提高python的并发能力,可以使用多线程,或者协程.比如网络爬虫,数据库操作等一些IO密集型的操作.下面对比python单线程,多线程和协程在网络爬虫场景下的速度. 一,单线程 ...
随机推荐
- lintcode:strStr 字符串查找
题目: 字符串查找 字符串查找(又称查找子字符串),是字符串操作中一个很有用的函数.你的任务是实现这个函数. 对于一个给定的 source 字符串和一个 target 字符串,你应该在 source ...
- SpringMVC学习总结(二)——DispatcherServlet详解
摘要: DispatcherServlet是前端控制器设计模式的实现,提供Spring Web MVC的集中访问点,而且负责职责的分派,而且与Spring IoC容器无缝集成,从而可以获得Spring ...
- 【多媒体封装格式详解】---MKV
http://blog.csdn.net/tx3344/article/details/8162656# http://blog.csdn.net/tx3344/article/details/817 ...
- React测试Mixin
1.test.jsx var randomNumberMixin = require("./randomNumberMixin.jsx"); describe("test ...
- set集合_定长
//set集合的操作 //便利初始化函数 NSSet *set1 = [[NSSet alloc] initWithObjects:@"aa", @&q ...
- 一些CSS技巧
1.网页LOGO背景居中 html { background: #f5f7f9 url(img/logo.png) no-repeat center center fixed; backgro ...
- 《数据通信与网络》笔记--TCP中的拥塞控制
1.拥塞窗口 发送方窗口的大小不仅取决于接收方,而.而且还取决于网络拥塞的情况. 发送方有2种信息:接收方通告的窗口大小和拥塞窗口的大小,实际的窗口大小事这两者中的最小者. 实际窗口大小 = min( ...
- 面试题_82_to_87_Date、Time 及 Calendar 的面试题
82)在多线程环境下,SimpleDateFormat 是线程安全的吗?(答案)不是,非常不幸,DateFormat 的所有实现,包括 SimpleDateFormat 都不是线程安全的,因此你不应该 ...
- WEBBROWSER中模拟鼠标点击(SendMessage/PostMessage)
好久没有写文章,发一篇顶顶博客访问量.别人建议转一些比较好的代码也贴过来,但是我打算这里主要发自己原创的代码,所以么..流量该多少就多少吧... 回到主题,在webbrowser中点击某链接网上几乎都 ...
- hdu 4911 Inversion (分治 归并排序 求逆序数)
题目链接 题意:给n个数,求交换k次相邻的数之后的最小的逆序数对. 用分治的方法,以前在poj上做过这种题,昨天比赛的时候忘了.... 下面的归并排序还是以前的模板. #include <ios ...