Python多线程实例教程_python

本文以实例形式较为详细的讲解了Python的多线程,是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有没有参与这个模块的开发,只是比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实例教程,以便于您获取更多的相关知识。

时间: 2024-12-30 23:23:40

Python多线程实例教程_python的相关文章

python中的多线程实例教程_python

本文以实例形式较为详细的讲述了Python中多线程的用法,在Python程序设计中有着比较广泛的应用.分享给大家供大家参考之用.具体分析如下: python中关于多线程的操作可以使用thread和threading模块来实现,其中thread模块在Py3中已经改名为_thread,不再推荐使用.而threading模块是在thread之上进行了封装,也是推荐使用的多线程模块,本文主要基于threading模块进行介绍.在某些版本中thread模块可能不存在,要使用dump_threading来代

Python中zip()函数用法实例教程_python

本文实例讲述了Python中zip()函数的定义及用法,相信对于Python初学者有一定的借鉴价值.详情如下: 一.定义: zip([iterable, ...])zip()是Python的一个内建函数,它接受一系列可迭代的对象作为参数,将对象中对应的元素打包成一个个tuple(元组),然后返回由这些tuples组成的list(列表).若传入参数的长度不等,则返回list的长度和参数中长度最短的对象相同.利用*号操作符,可以将list unzip(解压). 二.用法示例: 读者看看下面的例子,对

Python中apply函数的用法实例教程_python

一.概述: python apply函数的具体含义如下:  apply(func [, args [, kwargs ]]) 函数用于当函数参数已经存在于一个元组或字典中时,间接地调用函数.args是一个包含将要提供给函数的按位置传递的参数的元组.如果省略了args,任何参数都不会被传递,kwargs是一个包含关键字参数的字典.   apply()的返回值就是func()的返回值,apply()的元素参数是有序的,元素的顺序必须和func()形式参数的顺序一致 二.使用示例: 下面给几个例子来详

Python迭代用法实例教程_python

本文实例讲述了Python中迭代的用法,是一个非常实用的技巧.分享给大家供大家参考借鉴之用.具体分析如下: 如果给定一个list或tuple,我们可以通过for循环来遍历这个list或tuple,这种遍历我们成为迭代(Iteration). 在Python中,迭代是通过for ... in来完成的,而很多语言比如C或者Java,迭代list是通过下标完成的,比如Java代码: for (i=0; i<list.length; i++) { n = list[i]; } 可以看出,Python的f

Python内置函数的用法实例教程_python

本文简单的分析了Python中常用的内置函数的用法,分享给大家供大家参考之用.具体分析如下: 一般来说,在Python中内置了很多有用的函数,我们可以直接调用. 而要调用一个函数,就需要知道函数的名称和参数,比如求绝对值的函数abs,只有一个参数.可以直接从Python的官方网站查看文档:http://docs.python.org/2/library/functions.html#abs 也可以在交互式命令行通过help(abs)查看abs函数的帮助信息. 调用abs函数: >>> a

Python实现命令行通讯录实例教程_python

1.实现目标 编写一个命令行通讯录程序,可以添加.查询.删除通讯录好友及电话 2.实现方法 创建一个类来表示一个人的信息.使用字典存储每个人的对象,名字作为键. 使用pickle模块永久地把这些对象存储下来. 使用字典内建的方法添加.删除修改人员信息. 3.思维导图 4.编写伪代码 # 1.创建字典用来存储通讯录信息 # 2.创建人员类,包含姓名.关系.电话三个属性 # 3.创建操作类,包含增加.查询.删除人员,退出,保存并退出五个方法 # 4.程序运行 # 5.判断通讯录文件是否存在 # 6.

python海龟绘图实例教程_python

本文以实例形式介绍了python turtle模块即海龟绘图的使用方法,对于需要进行图形编程的朋友相信会有一定的借鉴价值. python turtle模块简介:  python2.6版本中引入的一个简单的绘图工具,叫做海龟绘图(Turtle Graphics) 1.使用海龟绘图首先我们需要导入turtle,如下所示: from turtle import * #将turtle中的所有方法导入 2.海龟绘图属性:  (1)位置 (2)方向 (3)画笔(画笔的属性,颜色.画线的宽度) 3.操纵海龟绘

Python Tkinter简单布局实例教程_python

本文实例展示了Python Tkinter实现简单布局的方法,示例中备有较为详尽的注释,便于读者理解.分享给大家供大家参考之用.具体如下: # -*- coding: utf-8 -*- from Tkinter import * root = Tk() # 80x80代表了初始化时主窗口的大小,0,0代表了初始化时窗口所在的位置 root.geometry('80x80+10+10') # 填充方向 ''' Label(root, text = 'l1', bg = 'red').pack(f

Python图算法实例分析_python

本文实例讲述了Python图算法.分享给大家供大家参考,具体如下: #encoding=utf-8 import networkx,heapq,sys from matplotlib import pyplot from collections import defaultdict,OrderedDict from numpy import array # Data in graphdata.txt: # a b 4 # a h 8 # b c 8 # b h 11 # h i 7 # h g