百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Python又称胶水语言,那如何通过Python使用DLL / COM对象?

zhezhongyun 2024-12-03 06:29 52 浏览


作为工程师,我们知道我们应该避免重新发明轮子。 如果可以的话,我们想使用其他人编写的库为我们做一些繁重的工作。 在本文中,我将与您分享一些有关如何利用DLL(或任何其他具有COM类型信息的文件,例如TLB或OCX文件)中的现有库的知识。 具体来说,我将分享我在旅途中学到的一些知识,以弄清楚如何使用Python发送传真。 因此,我们最终将展示如何使用Windows传真和扫描实用程序中功能背后的Windows DLL。

什么是DLL

DLL是动态链接库。 DLL有点像可执行文件。它们可以包含代码,数据和其他资源。关于DLL,有很多要说的,但是对于本篇文章,我们最关注DLL的代码部分。

因此,您现在所需要知道的就是,我们想在这些DLL文件中重用的代码(可以调用的方法,可以读取的属性等)由其他工程师编写。

我们如何使用Python中的DLL?

像前面提到的,DLL是类似的可执行文件。除非它们不是自己执行,而是由应用程序加载和使用。加载DLL后,只需正确调用其方法即可。

有几种方法可以解决此问题。

使用ctypes

ctypes是Python标准库的一部分,在某种程度上,它有助于我们使用DLL。

作为一个简单的示例,我将向您展示如何利用GetSystemMetrics方法,该方法存在于Windows \ System32 \ user32.dll中的User32.dll中。

第一步是加载DLL。这很简单。

import ctypes 
User32 = ctypes.WinDLL('User32.dll')

现在我们可以直接从此DLL开始调用方法!

>>> User32.GetSystemMetrics(1) # Get the height of the primary monitor 1440

太酷了! 但是您可能会问自己:我如何知道访问名为.GetSystemMetrics的方法,以及我如何知道将1作为参数传递来获取显示器的高度?

我如何知道存在哪些方法和属性?

这是一个难题。 在DLL的世界中,通常希望您提前了解这些方法。 这通常意味着被告知或阅读文档。 对于上面的示例,我们可以阅读Microsoft提供的文档,其中GetSystemMetrics是一种接受一个参数的方法,该参数表示要检索的系统信息索引的整数:

int GetSystemMetrics( 
    int nIndex 
);

该文档还指定了整数参数如何映射到系统指标。在表中,我们看到SM_CYSCREEN的索引为1,并被描述为“主显示监视器的屏幕高度,以像素为单位”。基于此信息,我们将可以组合在一起调用User32.GetSystemMetrics(1)来获取主监视器的高度。

如果我们没有文档怎么办?

有时,我们没有足够的运气提前知道或提供参考。即使有它们,它们也不是很方便。您会注意到,与许多普通的Python类不同,我们之前创建的User32对象不会告诉我们DLL中存在哪些方法。您可以尝试调用dir(User32),但不会产生任何有用的信息。

如果您冒险尝试如何在没有文档的情况下获取此信息,则可能会被告知使用DLL导出器或COM浏览器。输入pywin32。

使用PyWin32

Pywin32是一个了不起的库,可让您通过Python与Windows API进行交互。 win32com.client组件是它的众多功能之一,它使您可以与DLL进行交互。 PyWin32中鲜为人知的功能之一就是能够为所有DLL方法生成Python类。您也可以将PythonCOM用作COM浏览器。您可以使用

python -m pip install pywin32

要浏览系统上的COM库:

python -m win32com.client.combrowse

有更好的COM浏览器,但是使用它很方便。

但是真正有趣的部分是能够自动为COM接口生成Python类。 要开始生成Python文件,您可以从外壳运行此命令

python -m win32com.client.makepy -i

如果您不输入要生成的库,它将提示您选择一个。 我选择了传真服务COM类型库(Windows Fax and Scan工具后面的一个),您将看到类似以下的输出:

Microsoft Fax Service Extended COM Type Library {2BF34C1A-8CAC-419F-8547-32FDF6505DB8}, lcid=0, major=1, minor=0 >>> # Use these commands in Python code to auto generate .py support >>> from win32com.client import gencache >>> gencache.EnsureModule('{2BF34C1A-8CAC-419F-8547-32FDF6505DB8}', 0, 1, 0)

按照这些说明,我们可以执行以下操作:

from win32com.client import gencache 
faxcomex = gencache.EnsureModule('{2BF34C1A-8CAC-419F-8547-32FDF6505DB8}', 0, 1, 0) 
print(dir(faxcomex)) # Unlike before, we can actually see some method names 
print(repr(faxcomex)) # You'll notice the generated filename there, if you're curious to look.

生成的代码有趣的是它附带了文档字符串。

class IFaxServer(DispatchBaseClass):
  'IFaxServer Interface'
  CLSID = IID('{D73733C7-CC80-11D0-B225-00C04FB6C2F5}')
  coclass_clsid = IID('{D73733C8-CC80-11D0-B225-00C04FB6C2F5}')
  def Connect(self, ServerName=defaultNamedNotOptArg):
    'Makes a connection to a fax server'
    return self._oleobj_.InvokeTypes(1, LCID, 1, (24, 0), ((8, 0),),ServerName
      )
  def CreateDocument(self, FileName=defaultNamedNotOptArg):
    'Creates a fax document to send'
    return self._ApplyTypes_(4, 1, (12, 0), ((8, 0),), 'CreateDocument', None,FileName
      )

因此,您甚至可以在交互式外壳中执行诸如help(faxcomex.IFaxServer.Connect)之类的操作。 漂亮整齐! 如果您愿意,甚至可以将生成的代码复制到faxcomex.py之类的文件中,而不用复制faxcomex = gencache.EnsureModule('{2BF34C1A-8CAC-419F-8547-32FDF6505DB8}',0,1,0)的代码, 可以简单地使用导入Faxcomex

发送传真

探索生成的Python代码并阅读FaxComEx接口的Microsoft文档,我提出了以下功能。

from faxcomex import FaxDocument, FaxServer
def send_fax(number, subject, recipient_name='', servername='', body_doc='C:\\Path\\To\\SomeFile.tiff') 
    doc = FaxDocument()
    doc.Body = body_doc
    doc.Subject = subject
    doc.Recipients.Add(number, recipient_name)
    server = FaxServer()
    server.Connect(servername)
    doc.ConntectedSubmit(server)
    server.Disconnect()

假设您有一个在本地运行的传真服务器,并且Windows传真和扫描工具中已经有传真调制解调器设置,则servername =''将假定为本地服务器。

这有效!

为了使事情变得更好,更像Python,我们可以通过子类化生成的Python类来隐藏COM库的某些构件。

from faxcomex import FaxServer, FaxDocument
class PyFaxServer(FaxServer):
    def __init__(self, servername=''):
        """get servername on object creation, so its not needed later"""
        super().__init__()
        self.__servername = servername
    def connect(self):
        """a python-naming-convention-compliant alias for `Connect`"""
        return self.Connect()
    def Connect(self):
        """override this so we can call connect without arguments"""
        return super().Connect(self.__servername)
    def _connection_manager(self):
        """manage connection and disconnection in a context manager"""
        try:
            yield self.connect()
        finally:
            self.Disconnect()
    def send(self, doc):
        """convenience method to connect to the server and send a document"""
        with self._connection_manager():
            doc.ConnectedSubmit(self)
class PyFaxDocument(FaxDocument):
    def __init__(self, *recipients, subject, body):
        super().__init__()
        for recipient_number, recipient_name in recipients:
            self.Recipients.Add(recipient_number, recipient_name)
        self.Subject = subject
        self.Body = body
    def submit(self, server):
        """Convenience method to submit document to a PyFaxServer object)"""
        server.send(self)

现在我们的send_fax函数看起来和感觉上更像Python,即使它是一个内部的MS DLL。

def send_fax(number, subject, body_doc, recipient_name='', servername=''):
    server = PyFaxServer(servername)
    recipient = (number, recipient_name)
    document = PyFaxDocument(recipient, subject=subject, body=body_doc)
    server.send(document)

因此,我们实现了一个不错的传真服务器接口,而无需编写我们自己的几乎任何代码。 挺酷的!

这就是我现在所拥有的。

资源

  • Tim Golden's PyWin32 docs(http://timgolden.me.uk/pywin32-docs/html/com/win32com/HTML/QuickStartClientCom.html#StaticDispatch)
  • Microsoft's FaxComEx API documentation(https://docs.microsoft.com/en-us/windows/win32/api/faxcomex/)
  • Python ctypes documentation.(https://docs.python.org/3.8/library/ctypes.html#loading-dynamic-link-libraries)

相关推荐

JPA实体类注解,看这篇就全会了

基本注解@Entity标注于实体类声明语句之前,指出该Java类为实体类,将映射到指定的数据库表。name(可选):实体名称。缺省为实体类的非限定名称。该名称用于引用查询中的实体。不与@Tab...

Dify教程02 - Dify+Deepseek零代码赋能,普通人也能开发AI应用

开始今天的教程之前,先解决昨天遇到的一个问题,docker安装Dify的时候有个报错,进入Dify面板的时候会出现“InternalServerError”的提示,log日志报错:S3_USE_A...

用离散标记重塑人体姿态:VQ-VAE实现关键点组合关系编码

在人体姿态估计领域,传统方法通常将关键点作为基本处理单元,这些关键点在人体骨架结构上代表关节位置(如肘部、膝盖和头部)的空间坐标。现有模型对这些关键点的预测主要采用两种范式:直接通过坐标回归或间接通过...

B 客户端流RPC (clientstream Client Stream)

客户端编写一系列消息并将其发送到服务器,同样使用提供的流。一旦客户端写完消息,它就等待服务器读取消息并返回响应gRPC再次保证了单个RPC调用中的消息排序在客户端流RPC模式中,客户端会发送多个请...

我的模型我做主02——训练自己的大模型:简易入门指南

模型训练往往需要较高的配置,为了满足友友们的好奇心,这里我们不要内存,不要gpu,用最简单的方式,让大家感受一下什么是模型训练。基于你的硬件配置,我们可以设计一个完全在CPU上运行的简易模型训练方案。...

开源项目MessageNest打造个性化消息推送平台多种通知方式

今天介绍一个开源项目,MessageNest-可以打造个性化消息推送平台,整合邮件、钉钉、企业微信等多种通知方式。定制你的消息,让通知方式更灵活多样。开源地址:https://github.c...

使用投机规则API加快页面加载速度

当今的网络用户要求快速导航,从一个页面移动到另一个页面时应尽量减少延迟。投机规则应用程序接口(SpeculationRulesAPI)的出现改变了网络应用程序接口(WebAPI)领域的游戏规则。...

JSONP安全攻防技术

关于JSONPJSONP全称是JSONwithPadding,是基于JSON格式的为解决跨域请求资源而产生的解决方案。它的基本原理是利用HTML的元素标签,远程调用JSON文件来实现数据传递。如果...

大数据Doris(六):编译 Doris遇到的问题

编译Doris遇到的问题一、js_generator.cc:(.text+0xfc3c):undefinedreferenceto`well_known_types_js’查找Doris...

网页内嵌PDF获取的办法

最近女王大人为了通过某认证考试,交了2000RMB,官方居然没有给线下教材资料,直接给的是在线教材,教材是PDF的但是是内嵌在网页内,可惜却没有给具体的PDF地址,无法下载,看到女王大人一点点的截图保...

印度女孩被邻居家客人性骚扰,父亲上门警告,反被围殴致死

微信的规则进行了调整希望大家看完故事多点“在看”,喜欢的话也点个分享和赞这样事儿君的推送才能继续出现在你的订阅列表里才能继续跟大家分享每个开怀大笑或拍案惊奇的好故事啦~话说只要稍微关注新闻的人,应该...

下周重要财经数据日程一览 (1229-0103)

下周焦点全球制造业PMI美国消费者信心指数美国首申失业救济人数值得注意的是,下周一希腊还将举行第三轮总统选举需要谷歌日历同步及部分智能手机(安卓,iPhone)同步日历功能的朋友请点击此链接,数据公布...

PyTorch 深度学习实战(38):注意力机制全面解析

在上一篇文章中,我们探讨了分布式训练实战。本文将深入解析注意力机制的完整发展历程,从最初的Seq2Seq模型到革命性的Transformer架构。我们将使用PyTorch实现2个关键阶段的注意力机制变...

聊聊Spring AI的EmbeddingModel

序本文主要研究一下SpringAI的EmbeddingModelEmbeddingModelspring-ai-core/src/main/java/org/springframework/ai/e...

前端分享-少年了解过iframe么

iframe就像是HTML的「内嵌画布」,允许在页面中加载独立网页,如同在画布上叠加另一幅动态画卷。核心特性包括:独立上下文:每个iframe都拥有独立的DOM/CSS/JS环境(类似浏...