当前位置: 代码网 > it编程>前端脚本>Python > Python使用ctypes调用Windows API清空回收站

Python使用ctypes调用Windows API清空回收站

2026年04月28日 Python 我要评论
引言很多朋友刚接触 windows 编程时,总觉得调用系统底层 api 是一件很高深、非常复杂的事情,感到很害怕。其实 python 自带的 ctypes 库就能让咱们轻松调用 c 语言导出的动态链接

引言

很多朋友刚接触 windows 编程时,总觉得调用系统底层 api 是一件很高深、非常复杂的事情,感到很害怕。其实 python 自带的 ctypes 库就能让咱们轻松调用 c 语言导出的动态链接库函数,本文就以实现 windows 上的 “清空回收站”功能为例,从零开始详解 python 如何通过 ctypes 库跟 windows api 进行交互调用。

一、整体思路

清空回收站,不能直接调用 shemptyrecyclebin 就完事了,我们要编写一个更智能更稳健的程序,能准确判断是否真正成功清空:

  1. 清空前先查询回收站里有多少文件,就知道有没有必要清空了。
  2. 调用 windows 提供的清空回收站 api,执行清理。
  3. 清空后再次查询文件数量,对比前后变化,判断用户是真的清空了,还是中途取消了(比如在系统弹出的确认对话框里点了“否”)。

这种执行后查询验证结果的思路,在实际开发中非常实用,我们不能盲目依赖 api 的返回值,比如 shemptyrecyclebin 的返回值,仅表示函数是不是成功调用了,不表示用户点击了确定清空,或执行过程中是否取消了操作,所以说,它的结果并不表示业务是否成功,也就是是否真正清空了回收站。

二、ctypes 是什么?怎么用?

ctypes 是 python 内置的库,这个库能把 python 的数据类型转换成 c 语言的数据类型,然后调用 dll 或共享库中的函数。简单说就是让 python 可以直接调用 windows 系统底层函数

2.1 加载 dll

windows api 大多存放在 kernel32.dlluser32.dllshell32.dll 等系统核心动态链接库文件中。这里咱们要用的两个函数都在 shell32.dll 里。

import ctypes shell32 = ctypes.windll('shell32', use_last_error=true) 
  • windll 表示加载一个 windows dll,默认使用 stdcall 调用约定(windows api 的标准)。
  • use_last_error=true 让 ctypes 在出错时保存 windows 错误码,方便调试。

2.2 定义函数原型(参数类型和返回值类型)

这是新手最容易踩坑的地方。windows api 函数是用 c 写的,python 并不知道它接收什么参数,所以我们必须显式声明参数类型(argtypes)和返回值类型(restype)。

例如咱们用到的清空函数声明(来自 win32 sdk):

hresult shemptyrecyclebinw(hwnd hwnd, lpcwstr pszrootpath, dword dwflags);

对应到 python 就是:

shell32.shemptyrecyclebinw.argtypes = [wintypes.hwnd, wintypes.lpcwstr, wintypes.dword]
shell32.shemptyrecyclebinw.restype = ctypes.hresult

在这里

  • hwnd 是窗口句柄(可理解为窗口的身份证),传 none 表示没有父级窗口。
  • lpcwstr 是宽字符串指针,对应 python 的 str (ctypes 会自动转成 wchar_t*)。
  • dword 是 32 位无符号整数。
  • hresult 是一个 32 位整数,一般来讲 0(即 s_ok)表示成功。

2.3 定义结构体

很多 windows api 都要传结构体,比如我们这里用到的查询回收站信息函数要用到 shqueryrbinfo 结构体。在 ctypes 中定义c语言结构体还是相对比较容易的:只要继承 ctypes.structure,然后写一个 _fields_ 列表,每个元素是 (字段名, 字段类型)

class shqueryrbinfo(ctypes.structure):
    _fields_ = [
        ("cbsize", wintypes.dword),       # 结构体自身大小
        ("i64size", ctypes.c_longlong),   # 总大小(字节)
        ("i64numitems", ctypes.c_longlong) # 项目总数
    ]

c 语言的 __int64 对应 python 的 ctypes.c_longlong(8 字节有符号整数)。
另外,考虑到向后兼容,这个结构体的第一个成员 cbsize 必须在调用前赋值为结构体占用的字节数,很多 windows api 都这样设计。

rb_info = shqueryrbinfo()
rb_info.cbsize = ctypes.sizeof(shqueryrbinfo)

三、查询回收站信息

封装一个 get_recycle_bin_count() 函数,返回所有驱动器回收站里的文件总数。

  • 调用 shqueryrecyclebinw(none, pointer_to_struct)
    第一个参数传 none 表示“查询所有驱动器的总回收站”。也可以传 "c:\\" 这样的路径,查询指定盘。
  • 第二个参数需要传结构体的指针,用 ctypes.byref() 获得。
  • 函数执行后,结构体的 i64numitems 字段就被填上了项目总数。
hr = shell32.shqueryrecyclebinw(none, ctypes.byref(rb_info))
if hr == 0:   # 成功
    return rb_info.i64numitems
else:
    return -1

为什么函数名最后总要带上 w 字母?
那是因为 windows api 有两套字符编码:a(ansi)和 w(unicode)。现代 windows 内部完全使用 unicode,所以咱们直接调用 w 版本,用 python 的 str 传参即可。

四、调用函数清空回收站

hr = shell32.shemptyrecyclebinw(none, none, 0)
  • 第一个参数 hwndnone 表示没有父窗口。
  • 第二个参数 pszrootpathnone 表示清空所有驱动器的回收站。
  • 第三个参数 dwflags0 表示使用系统默认行为,也就是弹出确认对话框并显示进度。

聪明的你一定想到了:如果用户在确认对话框点了“否”,api 会返回什么?
经过实测,shemptyrecyclebinw 在这种情况下依然返回 s_ok(成功)!所以仅靠返回值无法区分“用户取消了”和“真的清空了”。

因此我们的代码要做更严谨的判断:

  1. 清空前记录文件数 count_before
  2. 调用清空 api。
  3. 清空后再次查询文件数 count_after
  4. 综合判断:
    • 如果 hr == 0count_after < count_before(文件数减少了),说明真的删除了,于是提示“清空成功”。
    • 如果 hr == 0count_after == count_before,说明用户取消了确认对话框所以提示“操作已取消”。
    • 如果 hr != 0,说明 api 调用失败(例如权限不足),应显示错误代码。

另,如果清空前文件数就是 0,直接弹窗告知“无需操作”。

五、完整代码

可以直接把下面的代码复制保存为 empty_recycle_bin.pyw 双击运行,需要安装 python3。

empty_recycle_bin.pyw:

import ctypes
from ctypes import wintypes
# <-定义结构体 ->
# shqueryrbinfo 结构体用于接收回收站信息
class shqueryrbinfo(ctypes.structure):
    _fields_ = [
        ("cbsize", wintypes.dword),      # 结构体大小
        ("i64size", ctypes.c_longlong),  # 回收站内文件总大小 (字节)
        ("i64numitems", ctypes.c_longlong) # 回收站内项目总数
    ]
# <- 定义常量 ->
mb_ok = 0x00000000
mb_iconinformation = 0x00000040
mb_iconwarning = 0x00000030
mb_iconerror = 0x00000010
def get_recycle_bin_count():
    """获取所有驱动器回收站的文件总数"""
    shell32 = ctypes.windll('shell32', use_last_error=true)
    # 定义 shqueryrecyclebinw
    # hresult shqueryrecyclebinw(lpcwstr pszrootpath, lpshqueryrbinfo pshqueryrbinfo);
    shell32.shqueryrecyclebinw.argtypes = [wintypes.lpcwstr, ctypes.pointer(shqueryrbinfo)]
    shell32.shqueryrecyclebinw.restype = ctypes.hresult
    rb_info = shqueryrbinfo()
    rb_info.cbsize = ctypes.sizeof(shqueryrbinfo)
    # pszrootpath 为 none 表示查询所有驱动器
    hr = shell32.shqueryrecyclebinw(none, ctypes.byref(rb_info))
    if hr == 0:
        return rb_info.i64numitems
    return -1
def empty_recycle_bin():
    shell32 = ctypes.windll('shell32', use_last_error=true)
    user32 = ctypes.windll('user32', use_last_error=true)
    # 定义 shemptyrecyclebinw
    shell32.shemptyrecyclebinw.argtypes = [wintypes.hwnd, wintypes.lpcwstr, wintypes.dword]
    shell32.shemptyrecyclebinw.restype = ctypes.hresult
    # 记录执行前的项目数
    count_before = get_recycle_bin_count()
    if count_before == 0:
        user32.messageboxw(none, "回收站已经是空的,无需操作。", "提示", mb_ok | mb_iconinformation)
        return
    # 调用 api 执行清空 (dwflags=0, 显示系统确认对话框)
    hr = shell32.shemptyrecyclebinw(none, none, 0)
    # 记录执行后的项目数
    count_after = get_recycle_bin_count()
    # 综合判断
    # 逻辑:api必须成功 and (文件数减少了 or 文件数变为了0)
    if hr == 0:
        if count_after < count_before and count_after >= 0:
            user32.messageboxw(none, f"清空成功!\n文件数从 {count_before} 降至 {count_after}。", "成功", mb_ok | mb_iconinformation)
        elif count_before != -1 and count_after == count_before:
            # api 返回了成功,但数量没变,说明用户在系统对话框点了“否”
            user32.messageboxw(none, "操作已取消或未执行删除。", "提示", mb_ok | mb_iconwarning)
        else:
            user32.messageboxw(none, "回收站状态未发生显著变化。", "提示", mb_ok | mb_iconwarning)
    else:
        # 如果返回了非 0 的 hresult
        err_hex = hex(hr & 0xffffffff)
        user32.messageboxw(none, f"操作失败。错误代码: {err_hex}", "错误", mb_ok | mb_iconerror)
if __name__ == "__main__":
    empty_recycle_bin()

: 执行脚本的 “python.exe” 要和调用的 dll 位数相同,比如都是64位的。

通过上面这个例子,就应该意识到了,python + ctypes 和 comtypes 几乎可以调用所有 windows api,借助python 调用 c dll 把系统底层能力融入到自己的程序中,其实也没有想象中的那么难。

以上就是python使用ctypes调用windows api清空回收站的详细内容,更多关于python ctypes调用windows api的资料请关注代码网其它相关文章!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2026  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com