侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

实例探究Python以并发方式编写高性能端口扫描器的方法

Python  /  管理员 发布于 7年前   196

关于端口扫描器
端口扫描工具(Port Scanner)指用于探测服务器或主机开放端口情况的工具。常被计算机管理员用于确认安全策略,同时被攻击者用于识别目标主机上的可运作的网络服务。

端口扫描定义是客户端向一定范围的服务器端口发送对应请求,以此确认可使用的端口。虽然其本身并不是恶意的网络活动,但也是网络攻击者探测目标主机服务,以利用该服务的已知漏洞的重要手段。端口扫描的主要用途仍然只是确认远程机器某个服务的可用性。

扫描多个主机以获取特定的某个端口被称为端口清扫(Portsweep),以此获取特定的服务。例如,基于SQL服务的计算机蠕虫就会清扫大量主机的同一端口以在 1433 端口上建立TCP连接。

Python实现

端口扫描器原理很简单,无非就是操作socket,能connect就认定这个端口开放着。

import socket def scan(port):   s = socket.socket()   if s.connect_ex(('localhost', port)) == 0:     print port, 'open'   s.close() if __name__ == '__main__':   map(scan,range(1,65536)) 

这样一个最简单的端口扫描器出来了。
等等喂,半天都没反应,那是因为socket是阻塞的,每次连接要等很久才超时。
我们自己给它加上的超时。

s.settimeout(0.1)

再跑一遍,感觉快多了。

多线程版本

import socket import threading def scan(port):   s = socket.socket()   s.settimeout(0.1)   if s.connect_ex(('localhost', port)) == 0:     print port, 'open'   s.close()  if __name__ == '__main__':   threads = [threading.Thread(target=scan, args=(i,)) for i in xrange(1,65536)]   map(lambda x:x.start(),threads) 

运行一下,哇,好快,快到抛出错误了。thread.error: can't start new thread。
想一下,这个进程开启了65535个线程,有两种可能,一种是超过最大线程数了,一种是超过最大socket句柄数了。在linux可以通过ulimit来修改。
如果不修改最大限制,怎么用多线程不报错呢?
加个queue,变成生产者-消费者模式,开固定线程。

多线程+队列版本

import socket import threading from Queue import Queue def scan(port):   s = socket.socket()   s.settimeout(0.1)   if s.connect_ex(('localhost', port)) == 0:     print port, 'open'   s.close()  def worker():   while not q.empty():     port = q.get()     try:       scan(port)     finally:       q.task_done()  if __name__ == '__main__':   q = Queue()   map(q.put,xrange(1,65535))   threads = [threading.Thread(target=worker) for i in xrange(500)]   map(lambda x:x.start(),threads)   q.join() 

这里开500个线程,不停的从队列取任务来做。

multiprocessing+队列版本
总不能开65535个进程吧?还是用生产者消费者模式

import multiprocessing def scan(port):   s = socket.socket()   s.settimeout(0.1)   if s.connect_ex(('localhost', port)) == 0:     print port, 'open'   s.close()  def worker(q):   while not q.empty():     port = q.get()     try:       scan(port)     finally:       q.task_done()  if __name__ == '__main__':   q = multiprocessing.JoinableQueue()   map(q.put,xrange(1,65535))   jobs = [multiprocessing.Process(target=worker, args=(q,)) for i in xrange(100)]   map(lambda x:x.start(),jobs) 

注意这里把队列作为一个参数传入到worker中去,因为是process safe的queue,不然会报错。
还有用的是JoinableQueue(),顾名思义就是可以join()的。

gevent的spawn版本

from gevent import monkey; monkey.patch_all(); import gevent import socket ... if __name__ == '__main__':   threads = [gevent.spawn(scan, i) for i in xrange(1,65536)]   gevent.joinall(threads) 

注意monkey patch必须在被patch的东西之前import,不然会Exception KeyError.比如不能先import threading,再monkey patch.

gevent的Pool版本

from gevent import monkey; monkey.patch_all(); import socket from gevent.pool import Pool ... if __name__ == '__main__':   pool = Pool(500)   pool.map(scan,xrange(1,65536))   pool.join() 

concurrent.futures版本

import socket from Queue import Queue from concurrent.futures import ThreadPoolExecutor ... if __name__ == '__main__':   q = Queue()   map(q.put,xrange(1,65536))   with ThreadPoolExecutor(max_workers=500) as executor:     for i in range(500):       executor.submit(worker,q) 

您可能感兴趣的文章:

  • python基于gevent实现并发下载器代码实例
  • python实现可以断点续传和并发的ftp程序
  • Python多进程并发(multiprocessing)用法实例详解
  • Python控制多进程与多线程并发数总结
  • python实现多线程的方式及多条命令并发执行
  • python并发编程之多进程、多线程、异步和协程详解
  • python监控网站运行异常并发送邮件的方法
  • Python实现多并发访问网站功能示例
  • Python socket实现的文件下载器功能示例
  • python使用urllib模块开发的多线程豆瓣小站mp3下载器
  • python 并发下载器实现方法示例


  • 上一条:
    Python中死锁的形成示例及死锁情况的防止
    下一条:
    Python使用dis模块把Python反编译为字节码的用法详解
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在python语言中Flask框架的学习及简单功能示例(0个评论)
    • 在Python语言中实现GUI全屏倒计时代码示例(0个评论)
    • Python + zipfile库实现zip文件解压自动化脚本示例(0个评论)
    • python爬虫BeautifulSoup快速抓取网站图片(1个评论)
    • vscode 配置 python3开发环境的方法(0个评论)
    • 近期文章
    • 智能合约Solidity学习CryptoZombie第四课:僵尸作战系统(0个评论)
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2018-04
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2022-01
    • 2023-07
    • 2023-10
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客