python 方法无法在线程中使用(附python获取网络流量)

  在python中,定义一个方法,直接调用可以,但是创建一个线程来调用就可能导致失败。这种现象多出现在使用com对象进行系统操作时,而且是以线程的形式调用。

  异常提示如下:syntax error。WMI returned a syntax error: you're probably running inside a thread without first calling pythoncom.CoInitialize[Ex] (no underlying exception)

  仔细观察的话,异常提示中已经给出了解决方案。在运行一个线程的时候需要调用pythoncom.CoInitialize()方法。鄙人在程序中使用了WMI模块,试图获取部分系统信息。

  异常的原因(推测)是com的机制问题。由于COM机制允许任意两组件之间相互通信而不必关心是在何种计算机上的何种操作系统下运行,也不用关心组件是使用何种语言编制的,这使COM技术拥有了强大的生命力。初始化COM环境的目的是使调用COM的API工作正常,也就是在COM操作之前调用CoInitialize或CoInitializeEX所以在线程函数中,如果使用com对象就必须调用CoInitialize或CoInitializeEX ,同时在退出时使用CoUninitialize来释放对象。
  案例:


c = wmi.WMI() interfaces = c.Win32_PerfRawData_Tcpip_NetworkInterface() print len(interfaces) rec = send = 0.0 for t in interfaces: print t.Name rec += float(t.BytesReceivedPersec) / 1024 / 1024 send += float(t.BytesSentPersec) / 1024 / 1024 print rec, send

 

  这段代码是根据电脑的网卡获取上下行的流量(即发送流量和接收流量),直接运行是可以的。但是在tornado的RequestHandler的get或者post方法中使用时,会抛出异常syntax error及相关提示。在RequestHandler中,post或者get请求被认为是一个后台的线程方法,所以需要在实例化WMI这个com组件之前先进行com实例化。

  为什么RequestHandler中post或者get请求被认为是一个后台的线程方法?这个问题从使用tornado的代码中可以窥得一二。代码如下:

if __name__ == '__main__':
    app = tornado.web.Application(
        handlers=[(r"/test/(\w+)", testHandler),
                  (r'/', MainHandler)]
    )
    server = tornado.httpserver.HTTPServer(app)
    server = server.listen(8848)
    tornado.ioloop.IOLoop.instance().start();

  在给tornado配置了路由规则之后,我们开启了httpserver的服务,最后创建进程来使tornado运行起来。而每次的post或者get请求则是通过路由调用相应的Handler,这些都是在线程中执行的。故案例中通过com组件WMI来获取系统流量放在get方法中就会报错。修改后如下:

    def get(self):
        res = {}
        pythoncom.CoInitialize()
        c = wmi.WMI()
        interfaces = c.Win32_PerfRawData_Tcpip_NetworkInterface()
        print len(interfaces)
        rec = send = 0.0
        for t in interfaces:
            print t.Name
            rec += float(t.BytesReceivedPersec) / 1024 / 1024
            send += float(t.BytesSentPersec) / 1024 / 1024
        print rec, send
        res["receive"] = "%.2f" % rec
        res["send"] = "%.2f" % send       
        self._write_json(res)

  关于WMI模块的使用可以参考如下:http://wutils.com/wmi/root/cimv2/win32_perfrawdata_tcpip_networkinterface/   模块中很有很多类型(类型也有很多属性),在具体使用的过程中查询起来很痛苦。案例中的统计流量用的是Win32_PerfRawData_Tcpip_NetworkInterface ,也有解决方案用的是Win32_PerfRawData_Tcpip_TCPv4 两个统计到的数据差别很大,请根据实际情况来选择。

  关于com机制的资料参考自:http://blog.csdn.net/chenglingsu6/article/details/5999134

posted @ 2017-07-27 09:43  单亚林  阅读(1888)  评论(0编辑  收藏  举报