Python学习之进程和线程
对于操作系统来说,一个任务就是一个进程(Process),比如打开一个浏览器就是启动一个浏览器进程,打开一个记事本就启动了一个记事本进程,打开两个记事本就启动了两个记事本进程,打开一个Word就启动了一个Word进程。
有些进程还不止同时干一件事,比如Word,它可以同时进行打字、拼写检查、打印等事情。在一个进程内部,要同时干多件事,就需要同时运行多个“子任务”,我们把进程内的这些“子任务”称为线程(Thread)。
进程
Python的os模块封装了常见的系统调用,其中包括fork,可以在Python程序中轻松创建子进程:
importos
print('Process(%s)start...'%os.getpid())
#OnlyworksonUnix/Linux/Mac:
pid=os.fork()
ifpid==0:
print('Iamchildprocess(%s)andmyparentis%s.'%(os.getpid(),os.getppid()))
else:
print('I(%s)justcreatedachildprocess(%s).'%(os.getpid(),pid))
运行结果如下:
Process(876)start...
I(876)justcreatedachildprocess(877).
Iamchildprocess(877)andmyparentis876.
由于Windows没有fork调用,上面的代码在Windows上无法运行。由于Mac系统是基于BSD(Unix的一种)内核,所以,在Mac下运行是没有问题的,推荐大家用Mac学Python!
multiprocessing
如果你打算编写多进程的服务程序,Unix/Linux无疑是正确的选择。由于Windows没有fork调用,难道在Windows上无法用Python编写多进程的程序?
由于Python是跨平台的,自然也应该提供一个跨平台的多进程支持。multiprocessing模块就是跨平台版本的多进程模块。
multiprocessing模块提供了一个Process类来代表一个进程对象,下面的例子演示了启动一个子进程并等待其结束:
frommultiprocessingimportProcess
importos
#子进程要执行的代码
defrun_proc(name):
print('Runchildprocess%s(%s)...'%(name,os.getpid()))
if__name__=='__main__':
print('Parentprocess%s.'%os.getpid())
p=Process(target=run_proc,args=('test',))
print('Childprocesswillstart.')
p.start()
p.join()
print('Childprocessend.')
执行结果如下:
Parentprocess928.
Processwillstart.
Runchildprocesstest(929)...
Processend.
创建子进程时,只需要传入一个执行函数和函数的参数,创建一个Process实例,用start()方法启动,这样创建进程比fork()还要简单。
join()方法可以等待子进程结束后再继续往下运行,通常用于进程间的同步。
线程
Python的标准库提供了两个模块:_thread和threading,_thread是低级模块,threading是高级模块,对_thread进行了封装。绝大多数情况下,我们只需要使用threading这个高级模块。
启动一个线程就是把一个函数传入并创建Thread实例,然后调用start()开始执行:
importtime,threading
#新线程执行的代码:
defloop():
print('thread%sisrunning...'%threading.current_thread().name)
n=0
whilen<5:
n=n+1
print('thread%s>>>%s'%(threading.current_thread().name,n))
time.sleep(1)
print('thread%sended.'%threading.current_thread().name)
print('thread%sisrunning...'%threading.current_thread().name)
t=threading.Thread(target=loop,name='LoopThread')
t.start()
t.join()
print('thread%sended.'%threading.current_thread().name)
执行结果如下:
threadMainThreadisrunning...
threadLoopThreadisrunning...
threadLoopThread>>>1
threadLoopThread>>>2
threadLoopThread>>>3
threadLoopThread>>>4
threadLoopThread>>>5
threadLoopThreadended.
threadMainThreadended.
Lock
多线程和多进程最大的不同在于,多进程中,同一个变量,各自有一份拷贝存在于每个进程中,互不影响,而多线程中,所有变量都由所有线程共享,所以,任何一个变量都可以被任何一个线程修改,因此,线程之间共享数据最大的危险在于多个线程同时改一个变量,把内容给改乱了。
balance=0
lock=threading.Lock()
defrun_thread(n):
foriinrange(100000):
#先要获取锁:
lock.acquire()
try:
#放心地改吧:
change_it(n)
finally:
#改完了一定要释放锁:
lock.release()
当多个线程同时执行lock.acquire()时,只有一个线程能成功地获取锁,然后继续执行代码,其他线程就继续等待直到获得锁为止。
获得锁的线程用完后一定要释放锁,否则那些苦苦等待锁的线程将永远等待下去,成为死线程。所以我们用try...finally来确保锁一定会被释放。
ThreadLocal
importthreading
#创建全局ThreadLocal对象:
local_school=threading.local()
defprocess_student():
#获取当前线程关联的student:
std=local_school.student
print('Hello,%s(in%s)'%(std,threading.current_thread().name))
defprocess_thread(name):
#绑定ThreadLocal的student:
local_school.student=name
process_student()
t1=threading.Thread(target=process_thread,args=('Alice',),name='Thread-A')
t2=threading.Thread(target=process_thread,args=('Bob',),name='Thread-B')
t1.start()
t2.start()
t1.join()
t2.join()
执行结果:
Hello,Alice(inThread-A)
Hello,Bob(inThread-B)
全局变量local_school就是一个ThreadLocal对象,每个Thread对它都可以读写student属性,但互不影响。你可以把local_school看成全局变量,但每个属性如local_school.student都是线程的局部变量,可以任意读写而互不干扰,也不用管理锁的问题,ThreadLocal内部会处理。
可以理解为全局变量local_school是一个dict,不但可以用local_school.student,还可以绑定其他变量,如local_school.teacher等等。
ThreadLocal最常用的地方就是为每个线程绑定一个数据库连接,HTTP请求,用户身份信息等,这样一个线程的所有调用到的处理函数都可以非常方便地访问这些资源。
以上内容为大家介绍了Python学习之进程和线程,希望对大家有所帮助,如果想要了解更多Python相关知识,请关注IT培训机构:千锋教育。http://www.mobiletrain.org/
相关推荐HOT
更多>>Ppython await是什么?
await的解释:await用来声明程序挂起。比如异步程序执行到某一步时需要等待的时间很长,就将此挂起,去执行其他的异步程序。await后面只能跟异...详情>>
2023-11-06 20:47:43Python企业应用的优缺点
Python是软件开发领域一朵诱人的奇葩:人们经常把Python看作是一种起到缝合作用的严格脚本语言,不过却很少有人意识到Python已经足够强大到应用...详情>>
2023-11-06 18:04:55Python自定义模块导入方法
Python模块是一个Python文件,以.py结尾,包括了Python对象定义和Python语句,能让Python代码段更有逻辑性、更好用、更易懂,既然Python模块有...详情>>
2023-11-06 12:51:53Python中yield的用法详解
首先我要吐槽一下,看程序的过程中遇见了yield这个关键字,然后百度的时候,发现没有一个能简单的让我懂的,讲起来真TM的都是头头是道,什么参...详情>>
2023-11-06 11:43:31