分享

Python 标准库学习 --- string

 文炳春秋 2020-10-18

想要代码写得好,除了参与开源项目、在大公司实习,最快捷高效的方法就是阅读 Python 标准库。学习 Python 标准库,不是背诵每一个标准库的用法,而是要过一遍留下印象,挑自己感兴趣的库重点研究。这样实际做项目的时候,我们就可以游刃有余地选择标准库。

作为这一系列的开始,第一个学习的是 string 模块。string 模块作为内置函数 str 的补充,提供了一些便利的函数。我会持续分享自己关于标准库的学习笔记与思考,争取一两周更新一篇标准库的内容。记得给公众号加个星标,不会错过精彩内容。还可以在 github 上给我提 issue,我尽力回答。

第一步

# 导入 string 模块import string

capwords

string 模块中提供了 capwords 函数,该函数使得字符串中的每个单词变为大写形式。我们来看看源码中是如何定义的:

def capwords(s, sep=None):
    return (sep or ' ').join(x.capitalize() for x in s.split(sep))

capwords 接受一个位置参数:待处理的字符串,和一个可选关键字参数:字符串的分隔符。字符串默认使用空格分隔,比如 ‘my name is python ’,也可以指定 seq 分隔,比如传入 seq 为 ‘-’:‘my-name-is-python’。这个函数使得被分隔的单词首字母大写。

>>> s = 'my name is python'>>> capwords(s)'My Name Is Python'>>> s = 'my-name-is-python'>>> capwords(s,'-')'My-Name-Is-Python'

总结一下子:我们需要首先向 capwords 函数中传入字符串。capwords 函数通过 str.split 方法将字符串分割成单词,再通过生成器表达式和 str.capitalize 方法,使得每一个单词首字母大写,最后再通过 str.join 方法将单词拼装为字符串。

上面是 cpython 的实现。对于标准库中比较简单的函数,我们可以考虑,如果是自己的话,会用什么方法写这个函数,最后再使用 timeit 模块比较一下这两者的性能。

我举个例子,比如说,这个函数还可以使用 map 函数重写,下面这两种方法实质上和 cpython 的实现等价的。一个使用了 str 的 capitalize 方法,另一个通过 methodcaller 方法调用字符串的 capitalize 方法。

def capwords1(s:str, seq:str=None)->str:
    return (seq or ' ').join(map(str.capitalize, s.split(seq)))from operator import methodcallerdef capwords2(s:str, seq:str=None)->str:
    return (seq or ' ').join(map(methodcaller('capitalize'), s.split(seq)))

我们再和标准实现比较性能,我是在 ipython 上测试的:

text = "your time is limted, so don't waste it living someone else's lives" * 10000%timeit capwords(text)24.9 ms ± 588 µs per loop (mean ± std. dev. of 7 runs, 10 loops each)%timeit capwords1(text)22.1 ms ± 721 µs per loop (mean ± std. dev. of 7 runs, 10 loops each)%timeit capwords2(text)28.4 ms ± 3.38 ms per loop (mean ± std. dev. of 7 runs, 10 loops each)

通过测试,我们可以发现,我们实现的第一个版本的函数,性能可能好一些;而第二个版本的实现则要逊色不少。

Template

下面是 Template 的基本用法,这是 string 模块提供给我们的字符串插值函数。该函数会将传进来的参数转化为字符串,然后进行插值,所以不支持格式化字符串 ,但是优点是更加安全。

首先建立一个模板接受 string 参数,string 的格式要求为:$ + 标识符(首个字符必须为 字母或者下划线,之后的字符只能是 字母、下划线、数字),使用 substitute 方法,我们就可以替换标识符。

匹配的样式:$$, %name, %{name}

from string import Templatestring = '姓名:$name 年龄:${age} 爱好:$hobby'template = Template(string)

substitue 的参数可以是字典:

>>> template.substitute({'name':'Python', 'age': 30, 'hobby':'all'})'姓名:Python 年龄:30 爱好:all'

还可以是关键字参数:

>>> template.substitute(name='Python', age=30, hobby='all')'姓名:Python 年龄:30 爱好:all'

关键字错误,解释器会报 KeyError:

>>> template.substitute(name='Python', age=20, hobb='all')KeyError: 'hobby'

这时候,我们可以使用 template 提供的另外一个方法 safe_subsitute 来防止编译器报错。当 safe_substitute 方法没有找到相应的关键字,会原封不动地返回标识符。

>>> template.safe_substitute(name='Python', age=30, hobb='all')'姓名:Python 年龄:30 爱好:$hobby'

Template 有四个类属性,其中 delimiter 为分隔符,默认为 $ ,后面接标识符。通过重写 delimiter,我们可以支持 % 等符号替换。类属性 idpattern 为标识符匹配规则,类属性 flags 表示忽略大小写。

class Template(metaclass=_TemplateMetaclass):
    """A string class for supporting $-substitutions."""

    delimiter = '$'
    idpattern = r'(?a:[_a-z][_a-z0-9]*)'
    braceidpattern = None
    flags = _re.IGNORECASE

比如说,我们可以重写类属性 delimiter 和 idpattern。

class MyTemplate(Template):
    delimiter = '%'
    idpattern = '[_][a-z]+_[a-z]+'

上面我们自定义了一个类,继承自 string.Template,并重写了 delimiter 和 idpattern 类属性。

>>> s = '%_name_main %age'>>> template = MyTemplate(s)>>> template.substitute(_name_main='Python', age = 30)ValueError: Invalid placeholder in string>>> template.safe_substitute(_name_main='Python', age = 30)'Python %age'

我们可以看到,分隔符已经换成了百分号,而标识符必须符合 _字母_字母的形式,否则会提示 valueError。

我们还可以从源码中学到一些技巧:

from collections import ChainMap as _Chainmapdef substitute(*args, **kws):
    mapping = _ChainMap(kws, args[0])

*args 接受一个字典, kws 接受关键字参数,Chainmap 函数将多个映射连接起来,就可以查找 args 和 kws 中的关键字。

以上就是我学习 Python 标准库的思考,还请大家多多转发支持。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多