[TOC]
如果我们要操作文件、目录,可以在命令行下面输入操作系统提供的各种命令来完成。比如dir
、cp
等命令。
如果要在Python程序中执行这些目录和文件的操作怎么办?其实操作系统提供的命令只是简单地调用了操作系统提供的接口函数,Python内置的os
模块也可以直接调用操作系统提供的接口函数。
打开Python交互式命令行,我们来看看如何使用os
模块的基本功能:
1 | >>> import os |
如果是posix
,说明系统是Linux
、Unix
或Mac OS X
,如果是nt
,就是Windows
系统。
要获取详细的系统信息,可以调用uname()
函数:
1 | >>> os.uname() |
注意uname()
函数在Windows上不提供,也就是说,os
模块的某些函数是跟操作系统相关的。
环境变量
在操作系统中定义的环境变量,全部保存在os.environ
这个变量中,可以直接查看:
1 | >>> os.environ |
要获取某个环境变量的值,可以调用os.environ.get('key')
:
1 | >>> os.environ.get('PATH') |
操作文件和目录
操作文件和目录的函数一部分放在os
模块中,一部分放在os.path
模块中,这一点要注意一下。查看、创建和删除目录可以这么调用:
1 | # 查看当前目录的绝对路径: |
把两个路径合成一个时,不要直接拼字符串,而要通过os.path.join()
函数,这样可以正确处理不同操作系统的路径分隔符。在Linux/Unix/Mac下,os.path.join()
返回这样的字符串:
1 | part-1/part-2 |
而Windows下会返回这样的字符串:
1 | part-1\part-2 |
同样的道理,要拆分路径时,也不要直接去拆字符串,而要通过os.path.split()
函数,这样可以把一个路径拆分为两部分,后一部分总是最后级别的目录或文件名:
1 | >>> os.path.split('/Users/michael/testdir/file.txt') |
os.path.splitext()
可以直接让你得到文件扩展名,很多时候非常方便:
1 | >>> os.path.splitext('/path/to/file.txt') |
这些合并、拆分路径的函数并不要求目录和文件要真实存在,它们只对字符串进行操作。
文件操作使用下面的函数。假定当前目录下有一个test.txt
文件:
1 | # 对文件重命名: |
但是复制文件的函数居然在os
模块中不存在!原因是复制文件并非由操作系统提供的系统调用。理论上讲,我们通过上一节的读写文件可以完成文件复制,只不过要多写很多代码。
幸运的是shutil
模块提供了copyfile()
的函数,你还可以在shutil
模块中找到很多实用函数,它们可以看做是os
模块的补充。
最后看看如何利用Python的特性来过滤文件。比如我们要列出当前目录下的所有目录,只需要一行代码:
1 | >>> [x for x in os.listdir('.') if os.path.isdir(x)] |
要列出所有的.py
文件,也只需一行代码:
1 | >>> [x for x in os.listdir('.') if os.path.isfile(x) and os.path.splitext(x)[1]=='.py'] |
是不是非常简洁?
小结
Python的os
模块封装了操作系统的目录和文件操作,要注意这些函数有的在os
模块中,有的在os.path
模块中。
练习
利用
os
模块编写一个能实现dir -l
输出的程序。编写一个程序,能在当前目录以及当前目录的所有子目录下查找文件名包含指定字符串的文件,并打印出相对路径。
第一题:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16#!/usr/bin/env python3
# -*- coding: utf-8 -*-
from datetime import datetime
import os
pwd = os.path.abspath('.')
print(' Size Last Modified Name')
print('------------------------------------------------------------')
for f in os.listdir(pwd):
fsize = os.path.getsize(f)
mtime = datetime.fromtimestamp(os.path.getmtime(f)).strftime('%Y-%m-%d %H:%M')
flag = '/' if os.path.isdir(f) else ''
print('%10d %s %s%s' % (fsize, mtime, f, flag))第二题:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import os
path = "."
string = "out"
def p_file(path = "."):
for file in os.listdir(path):
sub_file = os.path.join(path,file)
if os.path.isfile(sub_file) and string in sub_file : #注意
print("filename:", sub_file)
elif os.path.isdir(sub_file):
p_file(sub_file)
if __name__ == '__main__':
p_file()
# 注意这里 --> os.path.isfile(sub_file)
# 为什么这里用的是sub_file 而不是 file呢, 因为path这时候可能不是".", 那如果用file没有指定路径 就是判断"." 目录下的这个file是文件还是目录