Python 用os和win32api库模仿DOS命令dir (完整实例)

简介: Python 用os和win32api库模仿DOS命令dir (完整实例)

补全了上一篇《Python 使用os库函数listdir() 模拟DOS命令dir》,基本上能够模仿 dir 和 dir/w 的显示结果:


    驱动器 D 中的卷是 文档

    卷的序列号是 109A-0446

    D:\test 的目录

   [.]             [..]            [01]            01.py           [02]            

   02.py           [03]            

                  2 个文件             2,748 字节

                  5 个目录 315,430,735,872 可用字节

    驱动器 D 中的卷是 文档

    卷的序列号是 109A-0446

    D:\test 的目录

   2021/05/02  23:48    <DIR>          .

   2021/05/02  23:48    <DIR>          ..

   2021/05/02  22:56    <DIR>          01

   2021/04/03  12:09             1,486 01.py

   2021/05/02  22:56    <DIR>          02

   2021/04/03  12:09             1,262 02.py

   2021/05/02  23:48    <DIR>          03

                  2 个文件             2,748 字节

                  5 个目录 315,430,735,872 可用字节

    驱动器 E 中的卷是 数据

    卷的序列号是 109A-0537

    E:\ 的目录

   2021/05/04  23:39    <DIR>          【Download】

   2021/05/04  21:19    <DIR>          【ScannedDoc】

   2021/05/04  21:18         2,410,344 杨程驾驶证证件照.png

                  1 个文件         2,410,344 字节

                  2 个目录 300,126,973,952 可用字节



代码:

import os,time
from win32api import GetVolumeInformation as DiskVolume
from win32api import GetDiskFreeSpaceEx as FreeSpace
def attrib(fn):
    from win32api import GetFileAttributes as attr
    ret=str(bin(attr(fn)))
    del attr
    if ret[-2]=='0':
        return True
    else:
        return False
def pyDir(path=None,para=None):
    if path==None: path=os.getcwd()
    dt=lambda x:time.strftime('%Y/%m/%d  %H:%M   ',time.localtime(x))
    cnCount=lambda str:len([i for i in str if '\u4e00'<=i<='\u9fff'])
    _time=dt(os.path.getmtime(path))
    print(u' 驱动器 '+path[0].upper()+ u' 中的卷是 '+DiskVolume(path[0:2])[0])
    tmp=str(hex(DiskVolume(path[0:2])[1])).upper()
    if tmp[:2]=='0X': tmp=tmp[2:]
    diskVol=tmp[:4]+'-'+tmp[4:]
    print(u' 卷的序列号是',diskVol)
    print('\n '+path+u' 的目录\n')
    if path[-2:]!=':\\'and para!='/w':
        print(_time,'<DIR>'.ljust(14),'.')
        print(_time,'<DIR>'.ljust(14),'..')
    if path[-1]!='\\':
        path+='\\'
    if para==None:
        files=[i for i in os.listdir(path) if attrib(path+i)]
        _time=[dt(os.path.getmtime(path+i)) for i in files]
        _size=[os.path.getsize(path+i) for i in files]
        for i,fname in enumerate(files):
            print(_time[i],'<DIR>'.ljust(14) if os.path.isdir(path+fname) else format(_size[i],',').rjust(14),fname)
    elif para=='/w':
        files=[i for i in os.listdir(path) if attrib(path+i)]
        lenths=max([len(i)+cnCount(i)+2 if os.listdir(path) else len(i)+cnCount(i) for i in os.listdir(path)])
        if lenths<=16:
            fw=16
        elif 16<lenths<=20:
            fw=20
        elif 20<lenths<=40:
            fw=40
        elif lenths>40:
            fw=80
        if path[-2:]!=':\\':
            print('[.]'.ljust(fw),end='')
            print('[..]'.ljust(fw),end='')
        for i,fname in enumerate(files):
            if (i+2)*fw%80==0: print()
            cnCount=lambda str:len([i for i in str if '\u4e00'<=i<='\u9fff'])
            print((fname.ljust(fw-cnCount(fname)) if os.path.isfile(path+fname) else '['+fname+']').ljust(fw-cnCount(fname)),end='')
        print()
    fileSize=format(sum([os.path.getsize(path+i) for i in files if os.path.isfile(path+i)]),',')
    fileCount=len([i for i in files if os.path.isfile(path+i)])
    dirCount=len([i for i in files if os.path.isdir(path+i)])
    if path[-2:]!=':\\':dirCount+=2
    diskSize=format(FreeSpace(path[:2])[0],',')
    print(f'{fileCount} 个文件'.rjust(20),f' {fileSize} 字节'.rjust(20))
    print(f'{dirCount} 个目录'.rjust(20),f'{diskSize} 可用字节'.rjust(20))
    print()
pyDir('D:\\test','/w')
pyDir('D:\\test')
pyDir('E:\\')



dir命令默认是不列出隐藏文件或目录的;如果要列出隐藏的,可以在上述代码找到两处 if attrib(path+i) 删除这个条件掉即可 ,此时相当于命令:dir/a

代码用GetVolumeInformation 、 GetDiskFreeSpaceEx 和 GetFileAttributes求硬盘分区的卷号序列号、剩余空间字节数、文件属性等。


GetVolumeInformation() 返回磁盘卷标信息,返回值是一个5元素的tuple,分别对应:


  TCHAR       lpVolumeNameBuffer[1024];            //硬盘卷标名称   
  DWORD     dwVolumeSerialNumber;                 //序列号   
  DWORD     dwMaximumComponentLength;     //文件名最大长度   
  DWORD     FileSystemFlags;                            //文件系统标志   
  TCHAR       lpFileSystemNameBuffer[1024];    //文件系统名称   



其实序列号转成16进制数后去掉0x标记再在中间插入-字符,就与DOS命令dir显示的一样:如“卷的序列号是 109A-0446”

GetDiskFreeSpaceEx() 返回磁盘可用空间,返回值是一个3元素的tuple,分别对应:

LARGE_INTEGER  lpFreeBytesAvailableToCaller    //调用者可用空间的字节数
LARGE_INTEGER  lpTotalNumberOfBytes              //磁盘总容量的字节数
LARGE_INTEGER  lpTotalNumberOfFreeBytes      //磁盘的可用空间的字节数


GetFileAttributes() 对应DOS命令attrib,可以查看和修改文件或目录的属性。返回值是以下列表中的单个数值或它们任意几个的和都可以。进制转换可得到一个最长为16位的二进制数,各位上0,1就表示是否有对应属性,如返回22即2+4+16表示系统隐藏目录。判断文件或目录是否隐藏,请参见上述代码中自定义函数 attrib(fn)


   1 FILE_ATTRIBUTE_READONLY 只读

   2 FILE_ATTRIBUTE_HIDDEN 隐藏

   4 FILE_ATTRIBUTE_SYSTEM 系统

   16 FILE_ATTRIBUTE_DIRECTORY 目录

   32 FILE_ATTRIBUTE_ARCHIVE 存档

   64 FILE_ATTRIBUTE_DEVICE 保留

   128 FILE_ATTRIBUTE_NORMAL 正常

   256 FILE_ATTRIBUTE_TEMPORARY 临时

   ......之后的不常用,一直到2^15




目录
相关文章
|
2月前
|
JSON 监控 API
掌握使用 requests 库发送各种 HTTP 请求和处理 API 响应
本课程全面讲解了使用 Python 的 requests 库进行 API 请求与响应处理,内容涵盖环境搭建、GET 与 POST 请求、参数传递、错误处理、请求头设置及实战项目开发。通过实例教学,学员可掌握基础到高级技巧,并完成天气查询应用等实际项目,适合初学者快速上手网络编程与 API 调用。
459 130
|
3月前
|
域名解析 JSON API
【干货满满】如何处理requests库调用API接口时的异常情况
在调用 API 时,网络波动、服务器错误、参数异常等情况难以避免。本文提供一套系统化的异常处理方案,涵盖 requests 库常见异常类型、处理策略、实战代码与最佳实践,通过分类处理、重试机制与兜底策略,提升接口调用的稳定性与可靠性。
|
1月前
|
Ubuntu API C++
C++标准库、Windows API及Ubuntu API的综合应用
总之,C++标准库、Windows API和Ubuntu API的综合应用是一项挑战性较大的任务,需要开发者具备跨平台编程的深入知识和丰富经验。通过合理的架构设计和有效的工具选择,可以在不同的操作系统平台上高效地开发和部署应用程序。
101 11
|
2月前
|
安全 Python
告别 os.path 的繁琐:拥抱 Python 的 pathlib
告别 os.path 的繁琐:拥抱 Python 的 pathlib
392 6
|
7月前
|
自动驾驶 程序员 API
告别重复繁琐!Apipost参数描述库让API开发效率飙升!
在API开发中,重复录入参数占用了42%的时间,不仅效率低下还易出错。Apipost推出的参数描述库解决了这一痛点,通过智能记忆功能实现参数自动填充,如版本号、分页控制、用户信息等常用字段,大幅减少手动输入。支持Key-Value与Raw-Json格式导入,一键提取响应结果至文档,将创建20参数接口文档时间从18分钟缩短至2分钟。相比Postman需手动搜索变量,Apipost的参数复用响应速度仅0.3秒,且支持跨项目共享与实时纠错,真正实现“一次定义,终身受益”。
|
9月前
|
监控 API 计算机视觉
CompreFace:Star6.1k,Github上火爆的轻量化且强大的人脸识别库,api,sdk都支持
CompreFace 是一个在 GitHub 上拥有 6.1k Star 的轻量级人脸识别库,支持 API 和 SDK。它由 Exadel 公司开发,基于深度学习技术,提供高效、灵活的人脸识别解决方案。CompreFace 支持多种模型(如 VGG-Face、OpenFace 和 Facenet),具备多硬件支持、丰富的功能服务(如人脸检测、年龄性别识别等)和便捷的部署方式。适用于安防监控、商业领域和医疗美容等多个场景。
1017 4
|
6月前
|
算法 Python
Apriori算法的Python实例演示
经过运行,你会看到一些集合出现,每个集合的支持度也会给出。这些集合就是你想要的,经常一起被购买的商品组合。不要忘记,`min_support`参数将决定频繁项集的数量和大小,你可以根据自己的需要进行更改。
253 18
|
7月前
|
Python
Python教程:os 与 sys 模块详细用法
os 模块用于与操作系统交互,主要涉及夹操作、路径操作和其他操作。例如,`os.rename()` 重命名文件,`os.mkdir()` 创建文件夹,`os.path.abspath()` 获取文件绝对路径等。sys 模块则用于与 Python 解释器交互,常用功能如 `sys.path` 查看模块搜索路径,`sys.platform` 检测操作系统等。这些模块提供了丰富的工具,便于开发中处理系统和文件相关任务。
328 14
|
8月前
|
Kubernetes API 网络安全
当node节点kubectl 命令无法连接到 Kubernetes API 服务器
当Node节点上的 `kubectl`无法连接到Kubernetes API服务器时,可以通过以上步骤逐步排查和解决问题。首先确保网络连接正常,验证 `kubeconfig`文件配置正确,检查API服务器和Node节点的状态,最后排除防火墙或网络策略的干扰,并通过重启服务恢复正常连接。通过这些措施,可以有效解决与Kubernetes API服务器通信的常见问题,从而保障集群的正常运行。
632 17