前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Python 标准库学习 --- string

Python 标准库学习 --- string

作者头像
用户2870857
发布2019-12-23 16:06:22
6900
发布2019-12-23 16:06:22
举报
文章被收录于专栏:Python高效编程Python高效编程

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

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

github 地址: https://github.com/xflywind/Python-fighter

导入模块
代码语言:javascript
复制
# 导入 string 模块
import string
capwords

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

代码语言:javascript
复制
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’。这个函数使得被分隔的单词首字母大写。

代码语言:javascript
复制
>>> 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 方法。

代码语言:javascript
复制
def capwords1(s:str, seq:str=None)->str:
    return (seq or ' ').join(map(str.capitalize, s.split(seq)))


from operator import methodcaller

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

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

代码语言:javascript
复制
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}

代码语言:javascript
复制
from string import Template
string = '姓名:$name 年龄:${age} 爱好:$hobby'
template = Template(string)

substitue 的参数可以是字典:

代码语言:javascript
复制
>>> template.substitute({'name':'Python', 'age': 30, 'hobby':'all'})
'姓名:Python 年龄:30 爱好:all'

还可以是关键字参数:

代码语言:javascript
复制
>>> template.substitute(name='Python', age=30, hobby='all')
'姓名:Python 年龄:30 爱好:all'

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

代码语言:javascript
复制
>>> template.substitute(name='Python', age=20, hobb='all')
KeyError: 'hobby'

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

代码语言:javascript
复制
>>> template.safe_substitute(name='Python', age=30, hobb='all')
'姓名:Python 年龄:30 爱好:$hobby'

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

代码语言:javascript
复制
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。

代码语言:javascript
复制
class MyTemplate(Template):
    delimiter = '%'
    idpattern = '[_][a-z]+_[a-z]+'

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

代码语言:javascript
复制
>>> 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。

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

代码语言:javascript
复制
from collections import ChainMap as _Chainmap

def substitute(*args, **kws):
    mapping = _ChainMap(kws, args[0])

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

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

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-05-16,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Python高效编程 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 导入模块
  • capwords
  • Template
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档