Anders Wang


我所认识的每个人都是榜样,都有值得我去尊敬和学习的地方。


路径处理库pathlib使用详解

在编程中处理文件的读写操作时免不了要和文件路径打交道,甚至有时候为了完成某些场景功能会变的有些繁琐,以前在Python中操作文件路径,我们更多的时候是使用os模块。

而到了Python3时代后,Python3的系统标准库pathlib模块的 Path 对路径的操作会更简单。甚至可以说pathlib已经可以完全替代os.path,它完全采用面向对象的编程方式,因为Python 文档给它的定义是 Object-oriented filesystem paths(面向对象的文件系统路径),其语义适用于不同的操作系统,它继承纯路径但也提供I/O化操作,在处理配置路径方面十分简单。

这里要特殊说明下,我们所有的例子中都导入了pathlib2模块而不是pathlib模块,因为简单引用官方的话说就是老版本的pathlib模块已经只作为修复bug模式存在,而新版本的pathlib2是以修补更新的pathlib2而发布的,而且所有的新功能都可以兼容过去的旧版本python。

好了,来让我们看下常用的pathlib操作的例子汇总:

返回当前工作目录路径和Home路径

在pathlib里一切都是面向对象的,只需要调用指定的方法就可以了。

from pathlib2 import Path

# 获取当前目录
current_path = Path.cwd()  
print(current_path)

# 输出如下:
# /Users/Anders/Documents/

# 获取Home目录
home_path = Path.home()  
print(home_path)

# 输出如下:
# /Users/Anders
父目录操作

你可以看到想要获取一个路径下上级的父目录可以非常方便的直接使用面向对象的方式.parent就行了,如果还想上一级就继续以子对象继续操作parent属性就可以了。

from pathlib2 import Path

# 获取当前目录
current_path = Path.cwd()

# 获取上级父目录
print(current_path.parent)

# 获取上上级父目录
print(current_path.parent.parent)

# 获取上上上级父目录
print(current_path.parent.parent.parent)

# 获取上上上上级父目录
print(current_path.parent.parent.parent.parent)

# 获取上上上上级父目录
print(current_path.parent.parent.parent.parent.parent)

# 输出如下:
# /Users/Anders/Documents/Jupyter
# /Users/Anders/Documents
# /Users/Anders
# /Users
# /

当然路径是十分长的,而且在特定的场合我如果想获得每一级的父目录呢,贴心的pathlib已经帮我们想到了,使用parents属性就可以遍历整个父目录了,如下例子的效果和上面的例子是完全一样的,但是就变的非常简便。

from pathlib2 import Path

# 获取当前目录
current_path = Path.cwd()

for p in current_path.parents:  
    print(p)

# 输出如下:
# /Users/Anders/Documents/Jupyter
# /Users/Anders/Documents
# /Users/Anders
# /Users
# /
文件名操作

常用的文件名操作属性如下:

  • name 目录的最后一个部分

  • suffix 目录中最后一个部分的扩展名

  • suffixes 返回多个扩展名列表

  • stem 目录最后一个部分,没有后缀

  • with_name(name) 替换目录最后一个部分并返回一个新的路径

  • with_suffix(suffix) 替换扩展名,返回新的路径,扩展名存在则不变

from pathlib2 import Path

# 返回目录中最后一个部分的扩展名
example_path = Path('/Users/Anders/Documents/abc.gif')  
print(example_path.suffix)  
# 输出如下:
# .gif

# 返回目录中多个扩展名列表
example_paths = Path('/Users/Anders/Documents/abc.tar.gz')  
print(example_paths.suffixes)  
# 输出如下:
# ['.tar', '.gz']

# 返回目录中最后一个部分的文件名(但是不包含后缀)
example_path = Path('/Users/Anders/Documents/abc.gif')  
print(example_path.stem)  
# 输出如下:
# abc

# 返回目录中最后一个部分的文件名
example_path = Path('/Users/Anders/Documents/abc.gif')  
print(example_path.name)  
# 输出如下:
# abc.gif

# 替换目录最后一个部分的文件名并返回一个新的路径
new_path1 = example_path.with_name('def.gif')  
print(new_path1)  
# 输出如下:
# /Users/Anders/Documents/def.gif

# 替换目录最后一个部分的文件名并返回一个新的路径
new_path2 = example_path.with_suffix('.txt')  
print(new_path2)  
# 输出如下:
# /Users/Anders/Documents/abc.txt
路径拼接和分解
from pathlib2 import Path

#直接传进一个完整字符串
example_path1 = Path('/Users/Anders/Documents/powershell-2.jpg')

#也可以传进多个字符串
example_path2 = Path('/', 'Users', 'dongh', 'Documents', 'python_learn', 'pathlib_', 'file1.txt')

#也可以利用Path.joinpath()
example_path3 = Path('/Users/Anders/Documents/').joinpath('python_learn')

# #利用 / 可以创建子路径
example_path4 = Path('/Users/Anders/Documents')  
example_path5 = example_path4 / 'python_learn/pic-2.jpg'  
遍历文件夹

我们可以在路径对象后面直接使用iterdir()方法,该方法返回一个生成器,我们可以循环遍历出所有指定目录下的目录路径。

from pathlib2 import Path

# 返回目录中最后一个部分的扩展名
example_path = Path('/Users/Anders/Documents')  
[path for path in example_path.iterdir()]

# 输出如下:
# [PosixPath('/Users/Anders/Documents/abc.jpg'),
#  PosixPath('/Users/Anders/Documents/book-master'),
#  PosixPath('/Users/Anders/Documents/Database'),
#  PosixPath('/Users/Anders/Documents/Git'),
#  PosixPath('/Users/Anders/Documents/AppProjects')]
文件操作

文件操作是使用率非常高的操作,在pathlib里如果要打开一个文件也十分的简单,只需要open方法就可以,它的操作语法是:open(mode='r', bufferiong=-1, encoding=None, errors=None, newline=None)

from pathlib2 import Path

example_path = Path('/Users/Anders/Documents/information/JH.txt')

with example_path.open(encoding = 'GB2312') as f:  
    print(f.read())

对于简单的文件读写,在pathlib库中有几个简便的方法:

  • .read_text(): 以文本模式打开路径并并以字符串形式返回内容。
  • .read_bytes(): 以二进制/字节模式打开路径并以字节串的形式返回内容。
  • .write_text(): 打开路径并向其写入字符串数据。
  • .write_bytes(): 以二进制/字节模式打开路径并向其写入数据。

比如可以把之前的例子改写如下:

from pathlib2 import Path

example_path = Path('/Users/Anders/Documents/information/JH.txt')  
example_path.read_text(encoding='GB2312')  
创建文件夹和删除文件夹

关于这里的创建文件目录mkdir方法接收两个参数:

  • parents:如果父目录不存在,是否创建父目录。
  • exist_ok:只有在目录不存在时创建目录,目录已存在时不会抛出异常。
from pathlib2 import Path

example_path = Path('/Users/Anders/Documents/test1/test2/test3')

# 创建文件目录,在这个例子中因为本身不存在test1,test2,test3,由于parents为True,所以都会被创建出来。
example_path.mkdir(parents = True, exist_ok = True)  
# 删除路径对象目录,如果要删除的文件夹内包含文件就会报错
example_path.rmdir()  
判断文件及文件夹对象是否存在

关于文件的判断还有很多相关属性,罗列如下:

  • is_dir() 是否是目录

  • is_file() 是否是普通文件

  • is_symlink() 是否是软链接

  • is_socket() 是否是socket文件

  • isblockdevice() 是否是块设备

  • ischardevice() 是否是字符设备

  • is_absolute() 是否是绝对路径

  • resolve() 返回一个新的路径,这个新路径就是当前Path对象的绝对路径,如果是软链接则直接被解析

  • absolute() 也可以获取绝对路径,但是推荐resolve()

  • exists() 该路径是否指向现有的目录或文件:

部分例子可以参照下面:

from pathlib2 import Path

example_path = Path('/Users/Anders/Documents/pic-2.jpg')

# 判断对象是否存在
print(example_path.exists())  
# 输出如下:
# True

# 判断对象是否是目录
print(example_path.is_dir())  
# 输出如下:
# False

# 判断对象是否是文件
print(example_path.is_file())  
# 输出如下:
# True
文件的信息

只需要通过.stat()方法就可以返还指定路径的文件信息。

from pathlib2 import Path

example_path = Path('/Users/Anders/Documents/pic.jpg')  
print(example_path.stat())  
# 输出如下:
# os.stat_result(st_mode=33188, st_ino=8598206944, st_dev=16777220, st_nlink=1, st_uid=501, st_gid=20, st_size=38054, st_atime=1549547190, st_mtime=1521009880, st_ctime=1521009883)

print(example_path.stat().st_size)  
# 输出如下:
# 38054

如上,已经基本列举了路径处理中常用的各类操作,其实pathlib的方便和强大远不止于此,具体信息可以参见官方文档:https://docs.python.org/3.4/library/pathlib.html

最近的文章

在matplotlib中创建子图的多种方式

在用Matplotlib绘制图像时,有时候需要从多个角度对数据进行对比。为此子图的概念便提了出来,子图可以在较大的图形中同时放置一组较小的坐标轴。这些子图可能是画中画、网格图,或者是其他更复杂的布局形…

Python, 技术博文详细阅读
更早的文章

匿名函数lambda

在Python中有个匿名函数特性非常的便捷和有用,用关键字lambda就可以声明一个匿名函数,所以很多时候直接称呼为lambda函数。 每次介绍新特性的时候,我们都要反问什么是lambda函数呢?因为…

Python, 技术博文详细阅读
comments powered by Disqus