Event 简介

Event 事件 是线程间通信的最简单方法之一,主要用于线程同步。

处理机制

定义一个全局内置标志Flag,如果Flag为False,执行到 event.wait 时程序就会阻塞,如果Flag为True,event.wait 便不会阻塞

【注意如果处于阻塞状态,不管在哪使得Flag为true,wait都会继续执行】

接口

set()     将标志设置为True,并通知所有处于阻塞状态的线程恢复运行

clear()       将标志设置为False

isSet()  获取内置标志的状态,返回 True 或者 False

wait(timeout)   如果标志为False,将使得线程阻塞,如果为True,继续运行,默认为False

示例代码

示例代码--等通知

import threading
import time event = threading.Event() def chihuoguo(name):
# 等待事件,进入等待阻塞状态
print '%s 已经启动' % threading.currentThread().getName()
print '小伙伴 %s 已经进入就餐状态!'%name
time.sleep(1)
event.wait()
# 收到事件后进入运行状态
print '%s 收到通知了.' % threading.currentThread().getName()
print '小伙伴 %s 开始吃咯!'%name threads = []
thread1 = threading.Thread(target=chihuoguo, args=("a", ))
thread2 = threading.Thread(target=chihuoguo, args=("b", ))
threads.append(thread1)
threads.append(thread2) for thread in threads:
thread.start() time.sleep(0.1)
# 发送事件通知
print '主线程通知小伙伴开吃咯!'
event.set()

示例代码--互相通知

import threading
import time def producer():
print u'等人来买包子....'
event.wait()
#event.clear()
print event.isSet()
print u'chef:sb is coming for baozi...'
print u'chef:making a baozi for sb...'
time.sleep(5) print u'chef:你的包子好了...'
event.set() def consumer():
print u'chenchao:去买包子....'
event.set() time.sleep(2)
print 'chenchao:waiting for baozi to be ready...'
print event.wait()
print u'chenchao:哎呀真好吃....' event = threading.Event() p = threading.Thread(target=producer,args=())
c = threading.Thread(target=consumer,args=())
p.start()
c.start()

输出

等人来买包子....
chenchao:去买包子....
True
chef:sb is coming for baozi...
chef:making a baozi for sb...
chenchao:waiting for baozi to be ready...
True
chenchao:哎呀真好吃....
chef:你的包子好了...

上面实现了一个生产者-消费者模式,显然有错误,包子还没做好就吃上了。

稍微细心的缕下思路就会发现,消费者中的wait并没有阻塞线程,因为Flag此时为True

解决方法:

1. 用另一个 event2 来阻塞线程

2. 在生产者获得set时及时把Flag设置为False 【取消生产者中 event.clear() 的注释即可】

注意点1 

import threading
event = threading.Event()
print(1)
print(event.wait()) # 打印也会使线程阻塞
print(2)

注意点2

import time
import threading def myfunc():
while 1:
time.sleep(1)
print(1) event = threading.Event() ts = []
if len(ts) > 2:
event.wait() # 此时阻塞,已经开启的线程将继续运行 for i in range(12):
t = threading.Thread(target=myfunc)
t.start()
ts.append(t)

实战案例

多线程验证代理ip的有效性

问题:计算机并不能无休止的增加线程,每台计算机都有自己的上限

### 计算机能够执行的最大线程数

def myfunc():
time.sleep(2) count = 0
while 1:
count += 1
print(count)
t = threading.Thread(target=myfunc)
t.start()

超过上限,就会报错

thread.error: can't start new thread

思路:设置最大线程数,当启动的线程超过最大限制时,阻塞,不再生成新线程,并且持续跟踪线程数,一旦减小或者小于某个阈值,就取消阻塞,继续生成线程

class MyTestProxy(object):
def __init__(self):
self.sFile = 'ip.txt'
self.dFile = 'alive.txt'
self.url = 'https://www.qiushibaike.com/text/'
self.threadmax = 500 # 最大线程数
self.threadmin = 400 # 最低线程数
self.timeout = 3
self.regex = re.compile('qiushibaike.com')
self.aliveList = [] self.event = threading.Event()
self.event2 = threading.Event()
self.lock = threading.Lock() self.run() def run(self):
with open(self.sFile, 'rb') as fp:
lines = fp.readlines()
self.ts = 0 # 初始化线程数
while lines:
if self.ts > self.threadmax:
self.event.clear()
self.event.wait() # 超过设定线程就阻塞 line = lines.pop()
t = threading.Thread(target=self.linkWithProxy, args=(line, ))
t.start()
self.lock.acquire()
self.ts += 1 # 启动一个就加1,ts 被其他线程一直在更新,所以加锁
self.lock.release() self.event2.wait() # 处理完毕后统一存储
with open(self.dFile, 'w') as fp:
for i in range(len(self.aliveList)):
fp.write(self.aliveList[i]) def act(self):
# 执行完一个线程就减1,因为同时执行,要加锁
self.lock.acquire()
self.ts -= 1
self.lock.release()
print(self.ts)
if self.ts < self.threadmin:
self.event.set() # 小于最低线程取消阻塞
if self.ts == 0:
self.event2.set() def linkWithProxy(self, line):
# 爬虫
server = line.strip()
# print(server)
protocol = line.split(':')[0]
opener = urllib2.build_opener(urllib2.ProxyHandler({protocol:server}))
urllib2.install_opener(opener) try:
response = urllib2.urlopen(self.url, timeout=self.timeout)
except:
return
else:
try:
str = response.read()
if self.regex.search(str):
# print(str)
print('%s connect success'%server)
print(response.geturl())
self.aliveList.append(line)
except:
return
finally:
self.act() if __name__ == '__main__':
time.clock()
tp = MyTestProxy()
print(time.clock())

效率还是不错的

参考资料:

http://www.cnblogs.com/huxi/archive/2010/06/26/1765808.html

高效编程之 多线程Event的更多相关文章

  1. C# 高效编程笔记2

    C# 高效编程笔记2 1.理解GetHashCode()的陷阱 (1)作用:作为基于散列集合定义键的散列值,如:HashSet<T>,Dictionary<K,V>容器等 (2 ...

  2. C# 高效编程笔记1

    C# 高效编程笔记1 1.使用属性而不是可访问的数据成员 (1).NET Framework中的数据绑定类仅支持属性,而不支持共有数据成员 (2)属性相比数据成员更容易修改 2.用运行时常量(read ...

  3. Python高效编程的19个技巧

    初识Python语言,觉得python满足了我上学时候对编程语言的所有要求.python语言的高效编程技巧让我们这些大学曾经苦逼学了四年c或者c++的人,兴奋的不行不行的,终于解脱了.高级语言,如果做 ...

  4. 架构师速成-如何高效编程 for java

    引子 赵云大喝一声,挺枪骤马杀入重围,左冲右突,如入无人之境.那枪浑身上下,若舞梨花:遍体纷纷,如飘瑞雪. 赵云是所有历史人物中我最喜欢的一个,如果放到现代,他走了it的道路,一定可以成为一个编程高手 ...

  5. C# winform编程中多线程操作控件方法

    private void Form1_Load(object sender, EventArgs e) { Thread newthread = new Thread(new ThreadStart( ...

  6. IOS编程之多线程

    IOS编程之多线程 目录 概述——对多线程的理解 IOS中实现多线程的三种方式 NSThread 线程创建 线程的同步与锁 线程间的交互 线程的操作方法 NSOperation and NSOpera ...

  7. C语言高效编程的几招(绝对实用,绝对经典)

    编写高效简洁的C语言代码,是许多软件工程师追求的目标.废话不说,走起! 第一招:以空间换时间 计算机程序中最大的矛盾是空间和时间的矛盾,那么,从这个角度出发逆向思维来考虑程序的效率问题 eg.字符串的 ...

  8. [收藏转贴]struct探索·extern "C"含义探索 ·C++与C的混合编程·C 语言高效编程的几招

    一.C/C++语言 struct深层探索 1.自然对界 struct是一种复合数据类型,其构成元素既可以是基本数据类型(如 int.long.float等)的变量,也可以是一些复合数据类型(如 arr ...

  9. [.NET] 《C# 高效编程》(一) - C# 语言习惯

    C# 语言习惯 目录 一.使用属性而不是可访问的数据成员 二.使用运行时常量(readonly)而不是编译时常量(const) 三.推荐使用 is 或 as 操作符而不是强制类型转换 四.使用 Con ...

随机推荐

  1. PHP基础教程 php 网络上关于设计模式一些总结

    1.单例模式 单例模式顾名思义,就是只有一个实例.作为对象的创建模式,单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式的要点有三个: 一是某个类只能有一个实例; 二 ...

  2. Quartz监听器

    1.概念Quartz的监听器用于当任务调度中你所关注事件发生时,能够及时获取这一事件的通知.类似于任务执行过程中的邮件.短信类的提醒.Quartz监听器主要有JobListener.TriggerLi ...

  3. matlab中句柄@的用法

    @是Matlab中的句柄函数的标志符,即间接的函数调用方法. 1 句柄函数 主要有两种语法: handle = @functionname handle = @(arglist)anonymous_f ...

  4. jQuery_完成复选框的全选与全不选

    别的不多说,直接上代码,用于完成复选框的全选与全不选. <!DOCTYPE html> <html> <head> <meta charset="U ...

  5. 箭头函数详解()=>{}

    摘要:箭头函数有几个使用注意点. (1)函数体内的this对象,就是定义时所在的对象,而不是使用时所在的对象,箭头函数继承而来的this指向永远不变. (2)不可以当作构造函数,也就是说,不可以使用n ...

  6. (转载)Kaggle_Titanic生存预测 -- 详细流程吐血梳理

    Kaggle_Titanic生存预测 -- 详细流程吐血梳理 https://blog.csdn.net/Koala_Tree/article/details/78725881 Kaggle中Tita ...

  7. spring cloud:服务网关 Spring Cloud GateWay 入门

    Spring 官方最终还是按捺不住推出了自己的网关组件:Spring Cloud Gateway ,相比之前我们使用的 Zuul(1.x) 它有哪些优势呢?Zuul(1.x) 基于 Servlet,使 ...

  8. springboot 基于@Scheduled注解 实现定时任务

    前言 使用SpringBoot创建定时任务非常简单,目前主要有以下三种创建方式: 一.基于注解(@Scheduled) 二.基于接口(SchedulingConfigurer) 前者相信大家都很熟悉, ...

  9. python - 代码调试的好帮手sys._getframe()

    python 的调试,令人非常忧伤,通过将输出路径打印的方式,可以提高很大的方便性: import sys #coding=utf-8 def get_cur_info(): print sys._g ...

  10. Spring bean的自动装配属性

    bean的自动装配属性能简化xml文件配置. bean 的自动装配属性分为四种: 1.byName 2.byTyoe 3.constructor 4. autodetect byName: 它查找配置 ...