[新手上路]批处理新手入门导读[视频教程]批处理基础视频教程[视频教程]VBS基础视频教程[批处理精品]批处理版照片整理器
[批处理精品]纯批处理备份&还原驱动[批处理精品]CMD命令50条不能说的秘密[在线下载]第三方命令行工具[在线帮助]VBScript / JScript 在线参考
返回列表 发帖

[转载教程] Python调用com的技巧以及com事件

python调用有事件发生的com时,需要一些技巧。

我们以ie这个com为例来讲解一下我的经验。

首先我们需要pywin32这个模块的支持,它提供了我们调用com便利直接方法。你可以www.sf.net搜索并下载它。

先运行如下代码:

  1. # -*- coding: cp936 -*-
  2. import win32gui
  3. import win32com
  4. import win32com.client
  5. import pythoncom
  6. import time
  7. class test:
  8.     def runtest(self):
  9.         print 'tuntest'
  10. class EventHandler:
  11.     def OnVisible(self, visible):
  12.         global bVisibleEventFired
  13.         bVisibleEventFired = 1
  14.     def OnDownloadBegin(self):
  15.         print "DownloadBegin"
  16.     def OnDownloadComplete(self):
  17.         print "DownloadComplete"
  18.     def OnDocumentComplete(self, pDisp = pythoncom.Missing , URL = pythoncom.Missing):
  19.         print "documentComplete of %s" % URL
  20.         #在这里我们再调用test的runtest方法,看是否继承成功。
  21.         self.runtest()
  22. class runcom(test):
  23.     def __init__(self):
  24.         ie = win32com.client.DispatchWithEvents("InternetExplorer.Application", EventHandler)
  25.         ie.Visible = 1
  26.         ie.Navigate("www.aawns.com")
  27.         #这里调用test的runtest方法,看是否继承成功。
  28.         self.runtest()
  29.         pythoncom.PumpMessages()
  30.         ie.Quit()
  31. a=runcom()
复制代码


我们看到,程序运行正常,能打开我们指定的站点,并各事件被触发后都能作出正确的反映。

但是假如我们希望在事件发生后,能调用我们继承下来的一些方法和属性。你会发现无法使用。

如下代码将展示这个例子

  1. # -*- coding: cp936 -*-
  2. import win32gui
  3. import win32com
  4. import win32com.client
  5. import pythoncom
  6. import time
  7. class test:
  8.     def runtest(self):
  9.         print 'tuntest'
  10. class EventHandler:
  11.     def OnVisible(self, visible):
  12.         global bVisibleEventFired
  13.         bVisibleEventFired = 1
  14.     def OnDownloadBegin(self):
  15.         print "DownloadBegin"
  16.     def OnDownloadComplete(self):
  17.         print "DownloadComplete"
  18.     def OnDocumentComplete(self, pDisp = pythoncom.Missing , URL = pythoncom.Missing):
  19.         print "documentComplete of %s" % URL
  20.         #在这里我们再调用test的runtest方法,看是否继承成功。
  21.         self.runtest()
  22. class runcom(test):
  23.     def __init__(self):
  24.         ie = win32com.client.DispatchWithEvents("InternetExplorer.Application", EventHandler)
  25.         ie.Visible = 1
  26.         ie.Navigate("www.aawns.com")
  27.         #这里调用test的runtest方法,看是否继承成功。
  28.         self.runtest()
  29.         pythoncom.PumpMessages()
  30.         ie.Quit()
  31. a=runcom()
复制代码


运行结果是错误的。

tuntest
DownloadBegin
DownloadComplete
DownloadBegin
DownloadComplete
documentComplete of http://www.aawns.com/
pythoncom error: Python error invoking COM method.
Traceback (most recent call last):
  File "C:\Python23\Lib\site-packages\win32com\server\policy.py", line 283, in _
Invoke_
    return self._invoke_(dispid, lcid, wFlags, args)
  File "C:\Python23\Lib\site-packages\win32com\server\policy.py", line 288, in _
invoke_
    return S_OK, -1, self._invokeex_(dispid, lcid, wFlags, args, None, None)
  File "C:\Python23\Lib\site-packages\win32com\server\policy.py", line 550, in _
invokeex_
    return func(*args)
  File "D:\python\test.py", line 24, in OnDocumentComplete
    self.runtest()
  File "C:\Python23\Lib\site-packages\win32com\client\__init__.py", line 454, in
__getattr__
    raise AttributeError, "'%s' object has no attribute '%s'" % (repr(self), att
r)
exceptions.AttributeError: '' object has no attribute 'runtest'


我们看到test类的runtest方法并没有被继承进去,为什么呢?我的理解是因为com的事件模式让你无法继承python中的self,因为在调用 ie的时候并不是用EventHandler的实例而是将这个类作为了事件处理的方法(不知道这里理解是否正确,如果有更好的理解。我们交流)

经过查找了很多资料和试探了很多方法,只有采用全局变量的方式才能在事件和各个类之间传递数据。代码变更成了这样

  1. # -*- coding: cp936 -*-
  2. import win32gui
  3. import win32com
  4. import win32com.client
  5. import pythoncom
  6. import time
  7. class EventHandler:
  8.     def OnVisible(self, visible):
  9.         global bVisibleEventFired
  10.         bVisibleEventFired = 1
  11.     def OnDownloadBegin(self):
  12.         print "DownloadBegin"
  13.         #先继承全局变量增加一个字符串
  14.         global testlist
  15.         testlist.append("DownloadBegin")
  16.     def OnDownloadComplete(self):
  17.         print "DownloadComplete"
  18.         #先继承全局变量增加一个字符串
  19.         global testlist
  20.         testlist.append("DownloadComplete")
  21.     def OnDocumentComplete(self, pDisp = pythoncom.Missing , URL = pythoncom.Missing):
  22.         print "documentComplete of %s" % URL
  23.         #先继承全局变量再打印
  24.         global testlist
  25.         print testlist
  26. class runcom:
  27.     def __init__(self):
  28.         global testlist
  29.         ie = win32com.client.DispatchWithEvents("InternetExplorer.Application", EventHandler)
  30.         ie.Visible = 1
  31.         ie.Navigate("www.aawns.com")
  32.         #打印全局变量
  33.         print testlist
  34.         pythoncom.PumpMessages()
  35.         ie.Quit()
  36. testlist=[]
  37. a=runcom()
复制代码


可以看到,用全局变量的方式解决了于事件内传输数据的问题。

没有解决的问题:使用Twisted的时候也有相应的事件,如何保证Twisted 和com中的事件都被触发?

后来经过拐拐龙底咚朋友的提醒,知道了可以通过继承的方式将类的事件和属性继承下来。例子代码:

  1. import win32gui
  2. import win32com
  3. import win32com.client
  4. import pythoncom
  5. import time
  6. class Test:
  7.     def runtest(self):
  8.         print 'test'
  9. class EventHandler:
  10.     def OnVisible(self,visible):
  11.         global bVisibleEventFired
  12.         bVisibleEventFired = 1
  13.     def OnDownloadBegin(self):
  14.         print 'DownloadBegin'
  15.         self.runtest()
  16.         self.value = 1
  17.     def OnDownloadComplete(self):
  18.         print 'DownloadComplete'
  19.         self.value += 1
  20.     def OnDocumentComplete(self,pDisp=pythoncom.Missing,URL=pythoncom.Missing):
  21.         print 'documentComplete of %s' %URL
  22.         print self.value
  23. class H(Test,EventHandler):
  24.     pass
  25. ie = win32com.client.DispatchWithEvents('InternetExplorer.Application',H)
  26. ie.Visible = 1
  27. ie.Navigate("www.sohu.com")
  28. pythoncom.PumpMessages()
  29. ie.Quit()
复制代码

原文:http://bbs.chinaunix.net/thread-558195-1-1.html

返回列表