老男孩python学习自修第二十三天【多线程】
1. 线程的创建与运行
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ from threading import Thread def foo(param1, param2): print "{0}{1}".format(param1, param2) if __name__ == "__main__": print "main thread running" thread = Thread(target=foo, args=(123, "abc")) print "before new thread running" thread.start() print "after new thread running"
结果:
/Users/liudaoqiang/PycharmProjects/numpy/venv/bin/python /Users/liudaoqiang/Project/python_project/day23/thread_test.py main thread running before new thread running 123abcafter new thread running Process finished with exit code 0
2.线程常用API
thread.getName() 获取线程的名称,子线程的名称默认为Thread-n
thread.setName(name) 设置线程的名称
thread.isDaemon() 是否为守护线程,是守护线程则主线程结束,子线程结束;子线程不是守护线程则主线程等待子线程结束后才结束
thread.setDaemon(bool) 设置是否为守护线程
thread.start()
thread.run()
thread.join(timeout)
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ from threading import Thread import time def bar(): for item in range(100): print item time.sleep(1) if __name__ == "__main__": print "main thread running" thread = Thread(target=bar, args=()) thread.setDaemon(True) print "before %s running" % thread.getName() thread.start() print "after %s running" % thread.getName() time.sleep(5)
结果:
/Users/liudaoqiang/PycharmProjects/numpy/venv/bin/python /Users/liudaoqiang/Project/python_project/day23/thread_test.py main thread running before Thread-1 running after Thread-1 running 0 1 2 3 4 Process finished with exit code 0
3.join()的使用
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ from threading import Thread import time def bar(): for item in range(100): print item time.sleep(1) if __name__ == "__main__": print "main thread running" thread = Thread(target=bar, args=()) thread.setDaemon(True) print "before %s running" % thread.getName() thread.start() thread.join(10) print "after %s running" % thread.getName() time.sleep(5)
结果:
/Users/liudaoqiang/PycharmProjects/numpy/venv/bin/python /Users/liudaoqiang/Project/python_project/day23/thread_test.py main thread running before Thread-1 running 0 1 2 3 4 5 6 7 8 9 after Thread-1 running 10 11 12 13 14 Process finished with exit code 0
注意:
(1)thread.join(timeout) 表示主线程等待子线程运行timeout后,主线程再运行
4.自定义线程类
day23
__init__.py
mythread.py
mythread_test.py
mythread.py
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ from threading import Thread class MyThread(Thread): def run(self, ): print self.getName() + " is running" Thread.run(self)
mythread_test.py
#/usr/bin/env python # _*_ coding:UTF-8 _*_ from day23 import mythread def foo(param): print "Thread running with %s" % param if __name__ == "__main__": thread = mythread.MyThread(target=foo, args=("abc", )) thread.start()
结果:
/Users/liudaoqiang/PycharmProjects/numpy/venv/bin/python /Users/liudaoqiang/Project/python_project/day23/mythread_test.py Thread-1 is running Thread running with abc Process finished with exit code 0
注意:
(1)自定义线程类需要继承threading.Thread类并实现run(),在run()中调用父类的run()
(2)使用自定义的类需要导入,导入和使用为:
from package import module
mythread = module.MyThread()
5.生产者-消费者模型
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ from threading import Thread from Queue import Queue import time class Producer(Thread): def __init__(self, name, queue): self.__name = name self.__queue = queue super(Producer, self).__init__() def run(self): while True: if self.__queue.qsize() <= 10: self.__queue.put("包子") print "%s 生产了一个包子" % (self.__name) else: time.sleep(1) super(Producer,self).run() class Consumer(Thread): def __init__(self, name, queue): self.__name = name self.__queue = queue super(Consumer,self).__init__() def run(self): while True: if not self.__queue.empty(): self.__queue.get() print "%s 消费了一个包子" % (self.__name) super(Consumer,self).run() if __name__ == "__main__": queue = Queue(maxsize=200) for item in range(3): name = "producer %d" % item producer = Producer(name, queue) producer.start() for item in range(100): name = "consumer %d" % item consumer = Consumer(name, queue) consumer.start()
结果:
producer 1 生产了一个包子 consumer 42 消费了一个包子producer 1 生产了一个包子 consumer 42 消费了一个包子 producer 1 生产了一个包子consumer 42 消费了一个包子 producer 1 生产了一个包子 consumer 99 消费了一个包子producer 1 生产了一个包子 producer 1 生产了一个包子 consumer 94 消费了一个包子 producer 1 生产了一个包子
6.线程安全锁
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ import threading import time num = 0 def run(): lock.acquire() global num num += 1 lock.release() time.sleep(0.01) print "%s runing: num is %d" % (thread.getName(), num) if __name__ == "__main__": lock = threading.Lock() for item in range(2000): thread = threading.Thread(target=run, args=()) thread.setDaemon(False) thread.start()
部分结果:
Thread-1670 runing: num is 1669 Thread-1671 runing: num is 1669 Thread-1674 runing: num is 1674Thread-1674 runing: num is 1674 Thread-1674 runing: num is 1674 Thread-1681 runing: num is 1680 Thread-1681 runing: num is 1681 Thread-1685 runing: num is 1684 Thread-1685 runing: num is 1684 Thread-1686 runing: num is 1685 Thread-1687 runing: num is 1685 Thread-1697 runing: num is 1696Thread-1697 runing: num is 1696 Thread-1697 runing: num is 1697Thread-1698 runing: num is 1697
注意:
(1)一个线程没有没有进行sleep,则执行100条指令
(2)线程之间是共享内存资源的,为了线程的安全,需要增加线程锁
(3)双重加锁需要使用lock = threading.RLock()
其他锁:
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ import threading import time num = 0 def run(): lock.acquire() global num num += 1 lock.release() time.sleep(0.01) print "%s runing: num is %d" % (thread.getName(), num) if __name__ == "__main__": #这时普通锁 lock = threading.Lock() #这是双重锁 lock = threading.RLock() #这时多个线程共享锁 lock = threading.BoundedSemaphore(4) for item in range(2000): thread = threading.Thread(target=run, args=()) thread.setDaemon(False) thread.start()
7.生产者-消费者模型的通信事件
#!/usr/bin/env python # _*_ coding:UTF-8 _*_ import threading import time def producer(): print "生产者:正在等待消费者到来..." # 生产者在这里等待,等待消费者将标识位设置位True event.wait() # 生产者开始工作后,要将标识为清空,即设置位False,让消费者处于等待状态 event.clear() print "生产者:开始生产包子" time.sleep(5) print "生产者:包子生产完成" # 生产完成,将标识位设置位True,让消费者不处于等待状态 event.set() def consumer(): print "消费者:还未到来..." time.sleep(3) print "消费者:来了..." # 将标识位设置位True,让消费者开始工作 event.set() print "消费者:我要包子" # 这里需要延时 time.sleep(1) # 通过判断生产者是否将标识位设置位True while True: if event.isSet(): print "消费者:谢谢" break else: print "消费者:好了吗" time.sleep(1) if __name__ == "__main__": #定义生产者和消费者的通信事件 event = threading.Event() producer = threading.Thread(target=producer, args=()) producer.start() consumer = threading.Thread(target=consumer, args=()) consumer.start()
结果:
/Users/liudaoqiang/PycharmProjects/numpy/venv/bin/python /Users/liudaoqiang/Project/python_project/day23/thread_set_test.py 生产者:正在等待消费者到来... 消费者:还未到来... 消费者:来了... 消费者:我要包子生产者:开始生产包子 消费者:好了吗 消费者:好了吗 消费者:好了吗 消费者:好了吗 生产者:包子生产完成 消费者:谢谢 Process finished with exit code 0
注意:
(1)该通信事件的目的是生产者可以控制消费者是否等待;消费者也可以控制生产者是否等待
老男孩python学习自修第二十三天【多线程】的更多相关文章
- 老男孩python学习自修第二十四天【多进程】
1. 体验多进程的运行速度 #!/usr/bin/env python # _*_ coding:UTF-8 _*_ from multiprocessing import Pool import t ...
- 老男孩 python学习自修第二十二天【文件上传与下载】
1.使用socket实现文件上传 server.py #!/usr/bin/env python # _*_ coding:UTF-8 _*_ import os import SocketServe ...
- 老男孩python学习自修第十三天【md5加密】
示例代码如下: hashlib_test.py #!/usr/bin/env python # _*_ coding:UTF-8 _*_ import hashlib def genPasswd(na ...
- 老男孩python学习自修【第二天】字符串用法
实时处理增量日志最佳实践 主要使用f.seek()和f.tell()实现 字符串处理函数: s.find(substr, start, end) 查找子字符串,找不到则返回-1,找到则返回对应的索引 ...
- Python学习笔记第二十三周(Flask架构)
目录: 一.变量引用 内容: 备注:PyCharm小技巧,comm+alt+l 自动修改格式,comm+alt+return 向上添加新行 一.变量引用 1.url生成 from flask im ...
- 老男孩python学习自修第十六天【常用模块之sys和os】
例子: sys.argv 命令行参数List,第一个元素是程序本身路径 sys.exit(n) 退出程序,正常退出时exit(0) sys.version 获取Python解释程序的版本信息 sys. ...
- 老男孩python学习自修第十九天【异常处理】
1.常见的错误 TypeError 类型错误 NameError 没有该变量 ValueError 不期望的值 AttributeError 没有该属性 UnboundLocalError 没有该局部 ...
- 老男孩python学习自修第十八天【面向对象】
1.类与对象(构造方法与实例化) #!/usr/bin/env python # _*_ coding:UTF-8 _*_ class Province: def __init__(self, nam ...
- 老男孩python学习自修第十七天【装饰器】
装饰器:在某个方法执行前后去执行其他新定义的行为 例如: #!/usr/bin/env python # _*_ coding:UTF-8 _*_ def before_say_hello(): pr ...
随机推荐
- P2690 接苹果 (DP)
补一下dp的思路: dp[i][j]表示第 i 分钟转 j 次所得到的最大值.很容易得到这个dp的推导式. 图中¢()函数表示成立为1, 不成立为0的函数. #include<cmath> ...
- P2370 yyy2015c01的U盘(二分+背包)
思路:先说一下题意吧.就是给你n个文件大小为v,价值为c, 但是硬盘的大小为S, 而且要存的总价值大于等于p.问每次传输k大小的文件.问k的最大值是多少? 我们以k为二分对象. 直接讲检验函数吧. 假 ...
- Rsync服务实战
目录 1 安装rsync软件 2 配置 /etc/rsyncd.conf 3 创建用户(运行rsync服务的用户身份) 4 创建虚拟用户密码文件(客户端连接时候使用) 5启动 rsync 服务,并加入 ...
- # linux文件系统(inode block superblock)
先说一下格式化:每种操作系统所设置的文件属性/权限并不相同,为了存放这些文件所需的数据,因此就需要将分区格式化,以成为操作系统能够利用的文件系统格式.linux的文件格式为Ext2/Ext3,现在好像 ...
- 【ES6】函数
函数默认值问题 在ES6之前,不能直接为函数指定默认值,但是ES6允许为函数的参数设置默认值 之前实现方式 function log(x, y) { y = y || 'World'; console ...
- java 方法超时
public void getcd() { logger.info("任务开始!-------------------------------------"); final Exe ...
- 09 python初学 (字符串)
# 重复输出字符串 print('hello' * 2) # >>>hellohello # 字符串切片操作,最重要的!!!! print('hello'[2:]) # >&g ...
- nginx+tomcat9+redisson+redis+jdk1.8简单实现session共享
一.环境安装 由于资源限制,在虚拟机中模拟测试,一台虚拟机,所有软件均安装到该虚拟机内 安装系统:CentOS Linux release 7.4.1708 (Core) CentOS安装选择版本:B ...
- 用java语言写一个简易版本的登录页面,包含用户注册、用户登录、用户注销、修改密码等功能
package com.Summer_0421.cn; import java.util.Arrays; import java.util.Scanner; /** * @author Summer ...
- jvm调优-从eclipse开始
一.概述 什么是jvm调优呢?jvm调优就是根据gc日志分析jvm内存分配.回收的情况来调整各区域内存比例或者gc回收的策略:更深一层就是根据dump出来的内存结构和线程栈来分析代码中不合理的地方给予 ...