win32com.client.Dispatch可以使用,但win32com.client.gencache.EnsureDispatch无法使用。

4
我正在学习使用Python的win32com技术,并遇到了一个奇怪的问题。我尝试将Outlook联系人导出为字典列表。我的代码在使用win32com.client.Dispatch("Outlook.Application")时工作正常,但是在使用速度更快、更“安全”的win32com.client.gencache.EnsureDispatch("Outlook.Application")时返回0个联系人。以下是我的代码:
class MapiImport():
    def __init__(self):
        self.olApp = win32com.client.Dispatch("Outlook.Application")
        self.namespace = self.olApp.GetNamespace(u"MAPI")
        # olFolderContacts = 10 :
        self.mapiContacts = self.namespace.GetDefaultFolder(10).Items

    def getContacts(self, *fields):
        contacts = []
        # Class == 40 is ContactItem
        # Class == 69 is DistListItem
        # Exclude ditribution list and others objects != ContactItem
        for contact in filter(lambda x: x.Class == 40,self.mapiContacts) :
            if not fields :
                ctact = dict((x.Name,x.Value) for x in contact.ItemProperties)
            else :
                ctact = {}
                for field in fields :
                    itemProp = contact.itemProperties[field]
                    ctact[field] = itemProp.Value
            contacts.append(ctact)
        return contacts

#====TEST SCRIPT====
myMAPI = MapiImport()
fields = (u"LastName",u"FirstName",u"Companies",
          u"HomeTelephoneNumber",u"Home2TelephoneNumber",
          u"MobileTelephoneNumber",
          u"BusinessTelephoneNumber",u"Business2TelephoneNumber",
          u"Email1Address",u"Email2Address",u"Email3Address",
          u"HomeAddress",u"BusinessAddress",
          u"Birthday",u"Anniversary",
          u"Body")
print(myMAPI.getContacts(*fields))

所以,当我替换时:
olApp = win32com.client.Dispatch("Outlook.Application")

随着:

olApp = win32com.client.gencache.EnsureDispatch("Outlook.Application")

它返回以下错误:
Traceback (most recent call last):
  File "D:\Documents and Settings\da7950\Mes documents\Dropbox\cheetahImporter\mapiImport.py", line 42, in <module>
    print(myMAPI.getContacts(*fields))
  File "D:\Documents and Settings\da7950\Mes documents\Dropbox\cheetahImporter\mapiImport.py", line 19, in getContacts
    for contact in filter(lambda x: x.Class == 40,self.mapiContacts) :
  File "D:\Documents and Settings\da7950\Mes documents\Python27\lib\site-packages\win32com\gen_py\00062FFF-0000-0000-C000-000000000046x0x9x2\_Items.py", line 122, in __getitem__
    return self._get_good_object_(self._oleobj_.Invoke(*(81, LCID, 1, 1, item)), "Item")
com_error: (-2147352567, "Une exception s'est produite.", (4096, u'Microsoft Office Outlook', u'Index de la matrice en dehors des limites.', None, 0, -2147352567), None)

该消息意味着“矩阵索引超出范围”。最奇怪的是,在调用EnsureDispatch之后,win32com.client.Dispatch不再起作用。我必须卸载pywin32并重新安装它...
我正在运行Python2.7.3 64位和Outlook 2007 32位。
谢谢。

早期绑定(EnsureDispatch)和晚期绑定(Disaptch)的优缺点是什么?如果我保持我的代码这样(它将在许多平台上运行),这样可以吗? - franckgaga
据我看,win32com.client.Dispatch 没有起作用是因为您在 Temp\gen_py 文件夹中生成了缓存。 - Winand
1个回答

4
我找到了一个解决方案。这是一个win32com makepy的bug...
主要问题在于Outlook对olContactItem的索引是基于1的(与Python基于0的索引不同)。
olApp = win32com.client.gencache.EnsureDispatch("Outlook.Application")           
namespace = olApp.GetNamespace(u"MAPI")           
# olFolderContacts = 10 :           
mapiContacts = namespace.GetDefaultFolder(10).Items
for i in range(1,len(mapiContacts)+1) :
    contact = self.mapiContacts[i]
    ...

contact.itemProperties 还有一个问题。在使用 EnsureDispatch 时,所有的属性都是区分大小写的,所以:

contact.ItemProperties("FullName").Value
#       ^

工作正常,但不是:

contact.itemProperties("FullName").Value
#       ^

为了准确获取名称,请参考: Microsoft Outlook ContactItem 在 MSDN 上的参考模型

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接