欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解python并發(fā)獲取snmp信息及性能測(cè)試

 更新時(shí)間:2017年03月27日 10:17:39   作者:XuXinkun  
本篇文章主要介紹了詳解python并發(fā)獲取snmp信息及性能測(cè)試,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

python & snmp

用python獲取snmp信息有多個(gè)現(xiàn)成的庫(kù)可以使用,其中比較常用的是netsnmppysnmp兩個(gè)庫(kù)。網(wǎng)上有較多的關(guān)于兩個(gè)庫(kù)的例子。

本文重點(diǎn)在于如何并發(fā)的獲取snmp的數(shù)據(jù),即同時(shí)獲取多臺(tái)機(jī)器的snmp信息。

netsnmp

先說(shuō)netsnmp。python的netsnmp,其實(shí)是來(lái)自于net-snmp包。

python通過(guò)一個(gè)c文件調(diào)用net-snmp的接口獲取數(shù)據(jù)。

因此,在并發(fā)獲取多臺(tái)機(jī)器的時(shí)候,不能夠使用協(xié)程獲取。因?yàn)槭褂脜f(xié)程,在get數(shù)據(jù)的時(shí)候,協(xié)程會(huì)一直等待net-snmp接口返回?cái)?shù)據(jù),而不會(huì)像socket使用時(shí)那樣在等待數(shù)據(jù)時(shí)把CPU切換給其他協(xié)程使用。從這點(diǎn)上來(lái)說(shuō),使用協(xié)程和串行獲取沒(méi)有區(qū)別。

那么如何解決并發(fā)獲取的問(wèn)題呢?可以使用線程,多線程獲取(當(dāng)然也可以使用多進(jìn)程)。多個(gè)線程同時(shí)調(diào)用net-snmp的接口獲取數(shù)據(jù),然后cpu在多個(gè)線程之間不停切換。當(dāng)一個(gè)線程獲取一個(gè)結(jié)果后,可以繼續(xù)調(diào)用接口獲取下一個(gè)snmp數(shù)據(jù)。

這里我寫了一個(gè)樣例程序。首先把所有的host和oid做成任務(wù)放到隊(duì)列里,然后啟動(dòng)多個(gè)線程,去執(zhí)行獲取任務(wù)。程序樣例如下:

import threading
import time
import netsnmp
import Queue

start_time = time.time()
hosts = ["192.20.150.109", "192.20.150.110", "192.20.150.111", "192.20.150.112", "192.20.150.113", "192.20.150.114",
     "192.20.150.115", "192.20.150.116", "192.20.150.117", "192.20.150.118", "192.20.150.119", "192.20.150.120",
     "192.20.150.121", "192.20.80.148", "192.20.80.149", "192.20.96.59", "192.20.82.14", "192.20.82.15",
     "192.20.82.17", "192.20.82.19", "192.20.82.12", "192.20.80.139", "192.20.80.137", "192.20.80.136",
     "192.20.80.134", "192.20.80.133", "192.20.80.131", "192.20.80.130", "192.20.81.141", "192.20.81.140",
     "192.20.82.26", "192.20.82.28", "192.20.82.23", "192.20.82.21", "192.20.80.128", "192.20.80.127",
     "192.20.80.122", "192.20.81.159", "192.20.80.121", "192.20.80.124", "192.20.81.151", "192.20.80.118",
     "192.20.80.119", "192.20.80.113", "192.20.80.112", "192.20.80.116", "192.20.80.115", "192.20.78.62",
     "192.20.81.124", "192.20.81.125", "192.20.81.122", "192.20.81.121", "192.20.82.33", "192.20.82.31",
     "192.20.82.32", "192.20.82.30", "192.20.81.128", "192.20.82.39", "192.20.82.37", "192.20.82.35",
     "192.20.81.130", "192.20.80.200", "192.20.81.136", "192.20.81.137", "192.20.81.131", "192.20.81.133",
     "192.20.81.134", "192.20.82.43", "192.20.82.45", "192.20.82.41", "192.20.79.152", "192.20.79.155",
     "192.20.79.154", "192.25.76.235", "192.25.76.234", "192.25.76.233", "192.25.76.232", "192.25.76.231",
     "192.25.76.228", "192.25.20.96", "192.25.20.95", "192.25.20.94", "192.25.20.93", "192.24.163.14",
     "192.24.163.21", "192.24.163.29", "192.24.163.6", "192.18.136.22", "192.18.136.23", "192.24.193.2",
     "192.24.193.19", "192.24.193.18", "192.24.193.11", "192.20.157.132", "192.20.157.133", "192.24.212.232",
     "192.24.212.231", "192.24.212.230"]
oids = [".1.3.6.1.4.1.2021.11.9.0",".1.3.6.1.4.1.2021.11.10.0",".1.3.6.1.4.1.2021.11.11.0",".1.3.6.1.4.1.2021.10.1.3.1",
    ".1.3.6.1.4.1.2021.10.1.3.2",".1.3.6.1.4.1.2021.10.1.3.3",".1.3.6.1.4.1.2021.4.6.0",".1.3.6.1.4.1.2021.4.14.0",
    ".1.3.6.1.4.1.2021.4.15.0"]
myq = Queue.Queue()
rq = Queue.Queue()

#把host和oid組成任務(wù)
for host in hosts:
  for oid in oids:
    myq.put((host,oid))

def poll_one_host():
  while True:
    try:
      #死循環(huán)從隊(duì)列中獲取任務(wù),直到隊(duì)列任務(wù)為空
      host, oid = myq.get(block=False)
      session = netsnmp.Session(Version=2, DestHost=host, Community="cluster",Timeout=3000000,Retries=0)
      var_list = netsnmp.VarList()
      var_list.append(netsnmp.Varbind(oid))
      ret = session.get(var_list)
      rq.put((host, oid, ret, (time.time() - start_time)))
    except Queue.Empty:
      break

thread_arr = []

#開(kāi)啟多線程
num_thread = 50
for i in range(num_thread):
  t = threading.Thread(target=poll_one_host, kwargs={})
  t.setDaemon(True)
  t.start()
  thread_arr.append(t)

#等待任務(wù)執(zhí)行完畢
for i in range(num_thread):
  thread_arr[i].join()

while True:
  try:
    info = rq.get(block=False)
    print info
  except Queue.Empty:
    print time.time() - start_time
    break

netsnmp除了支持get操作之外,還支持walk操作,即遍歷某個(gè)oid。

但是walk使用的時(shí)候需要謹(jǐn)慎,以免導(dǎo)致高延時(shí)等問(wèn)題,具體可以參見(jiàn)之前的一篇snmpwalk高延時(shí)問(wèn)題分析的博客。

pysnmp

pysnmp是用python實(shí)現(xiàn)的一套snmp協(xié)議的庫(kù)。其自身提供了對(duì)于異步的支持。

import time
import Queue
from pysnmp.hlapi.asyncore import *
t = time.time()
myq = Queue.Queue()

#回調(diào)函數(shù)。在有數(shù)據(jù)返回時(shí)觸發(fā)
def cbFun(snmpEngine, sendRequestHandle, errorIndication, errorStatus, errorIndex, varBinds, cbCtx):
   myq.put((time.time()-t, varBinds))
hosts = ["192.20.150.109", "192.20.150.110", "192.20.150.111", "192.20.150.112", "192.20.150.113", "192.20.150.114",
     "192.20.150.115", "192.20.150.116", "192.20.150.117", "192.20.150.118", "192.20.150.119", "192.20.150.120",
     "192.20.150.121", "192.20.80.148", "192.20.80.149", "192.20.96.59", "192.20.82.14", "192.20.82.15",
     "192.20.82.17", "192.20.82.19", "192.20.82.12", "192.20.80.139", "192.20.80.137", "192.20.80.136",
     "192.20.80.134", "192.20.80.133", "192.20.80.131", "192.20.80.130", "192.20.81.141", "192.20.81.140",
     "192.20.82.26", "192.20.82.28", "192.20.82.23", "192.20.82.21", "192.20.80.128", "192.20.80.127",
     "192.20.80.122", "192.20.81.159", "192.20.80.121", "192.20.80.124", "192.20.81.151", "192.20.80.118",
     "192.20.80.119", "192.20.80.113", "192.20.80.112", "192.20.80.116", "192.20.80.115", "192.20.78.62",
     "192.20.81.124", "192.20.81.125", "192.20.81.122", "192.20.81.121", "192.20.82.33", "192.20.82.31",
     "192.20.82.32", "192.20.82.30", "192.20.81.128", "192.20.82.39", "192.20.82.37", "192.20.82.35",
     "192.20.81.130", "192.20.80.200", "192.20.81.136", "192.20.81.137", "192.20.81.131", "192.20.81.133",
     "192.20.81.134", "192.20.82.43", "192.20.82.45", "192.20.82.41", "192.20.79.152", "192.20.79.155",
     "192.20.79.154", "192.25.76.235", "192.25.76.234", "192.25.76.233", "192.25.76.232", "192.25.76.231",
     "192.25.76.228", "192.25.20.96", "192.25.20.95", "192.25.20.94", "192.25.20.93", "192.24.163.14",
     "192.24.163.21", "192.24.163.29", "192.24.163.6", "192.18.136.22", "192.18.136.23", "192.24.193.2",
     "192.24.193.19", "192.24.193.18", "192.24.193.11", "192.20.157.132", "192.20.157.133", "192.24.212.232",
     "192.24.212.231", "192.24.212.230"]

oids = [".1.3.6.1.4.1.2021.11.9.0",".1.3.6.1.4.1.2021.11.10.0",".1.3.6.1.4.1.2021.11.11.0",".1.3.6.1.4.1.2021.10.1.3.1",
    ".1.3.6.1.4.1.2021.10.1.3.2",".1.3.6.1.4.1.2021.10.1.3.3",".1.3.6.1.4.1.2021.4.6.0",".1.3.6.1.4.1.2021.4.14.0",
    ".1.3.6.1.4.1.2021.4.15.0"]
    
snmpEngine = SnmpEngine()

#添加任務(wù)
for oid in oids:
  for h in hosts:
    getCmd(snmpEngine,
      CommunityData('cluster'),
      UdpTransportTarget((h, 161), timeout=3, retries=0,),
      ContextData(),
      ObjectType(ObjectIdentity(oid)),
      cbFun=cbFun)
time1 = time.time() - t

#執(zhí)行異步獲取snmp
snmpEngine.transportDispatcher.runDispatcher()

#打印結(jié)果
while True:
  try:
    info = myq.get(block=False)
    print info
  except Queue.Empty:
    print time1
    print time.time() - t
    break

pysnmp本身只支持最基礎(chǔ)的get和getnext命令,因此如果想使用walk,需要自己進(jìn)行實(shí)現(xiàn)。

性能測(cè)試

在同一個(gè)環(huán)境下,對(duì)兩者進(jìn)行了性能測(cè)試。兩者對(duì)198個(gè)host,10個(gè)oid進(jìn)行采集。

測(cè)試組 耗時(shí)(sec)
netsnmp(20線程) 6.252
netsnmp(50線程) 3.269
netsnmp(200線程) 3.265
pysnmp 4.812

可以看到netsnmp的采集速度跟線程數(shù)有關(guān)。當(dāng)線程數(shù)增大到一定程度,采集時(shí)間不再縮短。因?yàn)殚_(kāi)辟線程同樣會(huì)消耗時(shí)間。而已有的線程已經(jīng)足夠處理。

pysnmp性能較之略差一下。詳細(xì)分析pysnmp在添加任務(wù)(執(zhí)行g(shù)etCmd時(shí))消耗了約1.2s,之后的采集約消耗3.3秒。

在增加了oid數(shù),在進(jìn)行實(shí)驗(yàn)。host仍然是198個(gè),oid是42個(gè)。

測(cè)試組 耗時(shí)(sec)
netsnmp(20線程) 30.935
netsnmp(50線程) 12.914
netsnmp(200線程) 4.044
pysnmp 11.043

可以看到差距被進(jìn)一步拉大。在線程足夠多的情況下,netsnmp的效率要明顯強(qiáng)于pysnmp。

因?yàn)槎叨贾С挚梢圆⑿胁杉鄠€(gè)host,從易用性來(lái)說(shuō),netsnmp更為簡(jiǎn)單一些,且netsnmp支持walk功能。本文更加推薦netsnmp。

安裝netsnmp需要安裝net-snmp。如果centos,則使用yum會(huì)較為方便。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

  • 在Python中os.fork()產(chǎn)生子進(jìn)程的例子

    在Python中os.fork()產(chǎn)生子進(jìn)程的例子

    今天小編就為大家分享一篇在Python中os.fork()產(chǎn)生子進(jìn)程的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-08-08
  • Python中關(guān)于函數(shù)的具體用法范例以及介紹

    Python中關(guān)于函數(shù)的具體用法范例以及介紹

    函數(shù)是組織好的,可重復(fù)使用的,用來(lái)實(shí)現(xiàn)單一,或相關(guān)聯(lián)功能的代碼段。函數(shù)能提高應(yīng)用的模塊性,和代碼的重復(fù)利用率。你已經(jīng)知道Python提供了許多內(nèi)建函數(shù),比如print()。但你也可以自己創(chuàng)建函數(shù),這被叫做用戶自定義函數(shù)
    2021-09-09
  • python實(shí)現(xiàn)PID算法及測(cè)試的例子

    python實(shí)現(xiàn)PID算法及測(cè)試的例子

    今天小編就為大家分享一篇python實(shí)現(xiàn)PID算法及測(cè)試的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-08-08
  • 使用Matplotlib將圖片保存為.tiff格式

    使用Matplotlib將圖片保存為.tiff格式

    這篇文章主要介紹了使用Matplotlib將圖片保存為.tiff格式問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Python這樣操作能存儲(chǔ)100多萬(wàn)行的xlsx文件

    Python這樣操作能存儲(chǔ)100多萬(wàn)行的xlsx文件

    這篇文章主要介紹了Python這樣操作能存儲(chǔ)100多萬(wàn)行的xlsx文件的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-04-04
  • python學(xué)習(xí)之面向?qū)ο蟆救腴T初級(jí)篇】

    python學(xué)習(xí)之面向?qū)ο蟆救腴T初級(jí)篇】

    Python從第一天開(kāi)始就是面向?qū)ο蟮恼Z(yǔ)言。正因?yàn)槿绱?,?chuàng)建和使用類和對(duì)象是非常地容易。這篇文章主要介紹了python中面向?qū)ο蟮南嚓P(guān)資料,屬于python面向?qū)ο髮W(xué)習(xí)的初級(jí)篇,本章將幫助您在使用Python面向?qū)ο缶幊痰募夹g(shù)方面所有提高,有需要的朋友可以參考借鑒。
    2017-01-01
  • Python調(diào)用MySQLdb插入中文亂碼的解決

    Python調(diào)用MySQLdb插入中文亂碼的解決

    這篇文章主要介紹了Python調(diào)用MySQLdb插入中文亂碼的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • Python實(shí)現(xiàn)遍歷windows所有窗口并輸出窗口標(biāo)題的方法

    Python實(shí)現(xiàn)遍歷windows所有窗口并輸出窗口標(biāo)題的方法

    這篇文章主要介紹了Python實(shí)現(xiàn)遍歷windows所有窗口并輸出窗口標(biāo)題的方法,涉及Python調(diào)用及遍歷windows窗口句柄的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-03-03
  • 最新評(píng)論