文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

利用python实现IP扫描

2023-01-31 07:20

关注

    需求:写一个脚本,判断192.168.11.0/24网络里,当前在线ip有哪些?

    知识点:

    1 使用subprocess模块,来调用系统命令,执行ping 192.168.11.xxx 命令

    2 调用系统命令执行ping命令的时候,会有返回值(ping的结果),需要用到stdout=fnull, stderr=fnull方法,屏蔽系统执行命令的返回值

     

    常规版本(代码)

    import os
    import time
    import subprocess
    def ping_call():
        start_time = time.time()
        fnull = open(os.devnull, 'w')
        for i in range(1, 256):
            ipaddr = 'ping 192.168.11.' + str(i)
            result = subprocess.call(ipaddr + ' -n 2', shell=True, stdout=fnull, stderr=fnull)
            current_time = time.strftime('%Y%m%d-%H:%M:%S', time.localtime())
            if result:
                print('时间:{} ip地址:{} ping fall'.format(current_time, ipaddr))
            else:
                print('时间:{} ip地址:{} ping ok'.format(current_time, ipaddr))
        print('程序耗时{:.2f}'.format(time.time() - start_time))
        fnull.close()
    ping_call()

    执行效果:

    blob.png


    上面的执行速度非常慢,怎么能让程序执行速度快起来?

    python提供了进程,线程,协程。分别用这三个对上面代码改进,提高执行效率,测试一波效率


    进程池异步执行 -- 开启20个进程

    import os
    import time
    import subprocess
    from multiprocessing import Pool
    def ping_call(num):
        fnull = open(os.devnull, 'w')
        ipaddr = 'ping 192.168.11.' + str(num)
        result = subprocess.call(ipaddr + ' -n 2', shell=True, stdout=fnull, stderr=fnull)
        current_time = time.strftime('%Y%m%d-%H:%M:%S', time.localtime())
        if result:
            print('时间:{} ip地址:{} ping fall'.format(current_time, ipaddr))
        else:
            print('时间:{} ip地址:{} ping ok'.format(current_time, ipaddr))
    
        fnull.close()
    
    
    if __name__ == '__main__':
        start_time = time.time()
        p = Pool(20)
        res_l = []
        for i in range(1, 256):
            res = p.apply_async(ping_call, args=(i,))
            res_l.append(res)
        for res in res_l:
            res.get()
        print('程序耗时{:.2f}'.format(time.time() - start_time))

    执行结果:

    blob.png



    线程池异步执行 -- 开启20个线程

    import os
    import time
    import subprocess
    from concurrent.futures import ThreadPoolExecutor
    def ping_call(num):
        fnull = open(os.devnull, 'w')
        ipaddr = 'ping 192.168.11.' + str(num)
        result = subprocess.call(ipaddr + ' -n 2', shell=True, stdout=fnull, stderr=fnull)
        current_time = time.strftime('%Y%m%d-%H:%M:%S', time.localtime())
        if result:
            print('时间:{} ip地址:{} ping fall'.format(current_time, ipaddr))
        else:
            print('时间:{} ip地址:{} ping ok'.format(current_time, ipaddr))
        fnull.close()
    
    if __name__ == '__main__':
        start_time = time.time()
        thread_pool = ThreadPoolExecutor(20)
        ret_lst = []
        for i in range(1, 256):
            ret = thread_pool.submit(ping_call, i)
            ret_lst.append(ret)
        thread_pool.shutdown()
        for ret in ret_lst:
            ret.result()
        print('线程池(20)异步-->耗时{:.2f}'.format(time.time() - start_time))

    执行结果:

    blob.png



    协程执行---(执行多个任务,遇到I/O操作就切换)

    使用gevent前,需要pip install gevent

    from gevent import monkey;monkey.patch_all()
    import gevent
    import os
    import time
    import subprocess
    
    def ping_call(num):
        fnull = open(os.devnull, 'w')
        ipaddr = 'ping 192.168.11.' + str(num)
        result = subprocess.call(ipaddr + ' -n 2', shell=True, stdout=fnull, stderr=fnull)
        current_time = time.strftime('%Y%m%d-%H:%M:%S', time.localtime())
        if result:
            print('时间:{} ip地址:{} ping fall'.format(current_time, ipaddr))
        else:
            print('时间:{} ip地址:{} ping ok'.format(current_time, ipaddr))
        fnull.close()
    
    def asynchronous(): # 异步
        g_l = [gevent.spawn(ping_call, i) for i in range(1, 256)]
        gevent.joinall(g_l)
    
    if __name__ == '__main__':
        start_time = time.time()
        asynchronous()
        print('协程执行-->耗时{:.2f}'.format(time.time() - start_time))

    执行结果:

    blob.png


    遇到I/O操作,协程的效率比进程,线程高很多!

    总结:python中,涉及到I/O阻塞的程序中,使用协程的效率最高


    最后附带协程池代码

    gevent.pool

    from gevent import monkey;monkey.patch_all()
    import gevent
    import os
    import time
    import subprocess
    import gevent.pool
    
    def ping_call(num):
        fnull = open(os.devnull, 'w')
        ipaddr = 'ping 192.168.11.' + str(num)
        result = subprocess.call(ipaddr + ' -n 2', shell=True, stdout=fnull, stderr=fnull)
        current_time = time.strftime('%Y%m%d-%H:%M:%S', time.localtime())
        if result:
            print('时间:{} ip地址:{} ping fall'.format(current_time, ipaddr))
        else:
            print('时间:{} ip地址:{} ping ok'.format(current_time, ipaddr))
        fnull.close()
    
    if __name__ == '__main__':
        start_time = time.time()
        res_l = []
        p = gevent.pool.Pool(100)
        for i in range(1, 256):
            res_l.append(p.spawn(ping_call, i))
        gevent.joinall(res_l)
        print('协程池执行-->耗时{:.2f}'.format(time.time() - start_time))

    执行结果:

    blob.png

     欢迎大家一起来玩好PY,一起交流。QQ群:198447500

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯