如何更优雅地写python代码


Posted in Python onJuly 02, 2019

前言

Python 这门语言最大的优点之一就是语法简洁,好的代码就像伪代码一样,干净、整洁、一目了然。但有时候我们写代码,特别是 Python 初学者,往往还是按照其它语言的思维习惯来写,那样的写法不仅运行速度慢,代码读起来也费尽,给人一种拖泥带水的感觉,过段时间连自己也读不懂。

《计算机程序的构造和解释》的作者哈尔·阿伯尔森曾这样说:“Programs must be written for people to read, and only incidentally for machines to execute.”

要写出 Pythonic(优雅的、地道的、整洁的)代码,还要平时多观察那些大牛代码,Github 上有很多非常优秀的源代码值得阅读,比如:requests、flask、tornado,笔者列举一些常见的 Pythonic 写法,希望能给你带来一点启迪。

1、变量交换

大部分编程语言中交换两个变量的值时,不得不引入一个临时变量:

>>> a = 1
>>> b = 2
>>> tmp = a
>>> a = b
>>> b = tmp

pythonic

>>> a, b = b, a

2、循环遍历区间元素

for i in [0, 1, 2, 3, 4, 5]:
(print i)
# 或者
for i in range(6):
(print i)

pythonic

for i in xrange(6):
(print i)

xrange 返回的是生成器对象,生成器比列表更加节省内存,不过需要注意的是 xrange 是 python2 中的写法,python3 只有 range 方法,特点和 xrange 是一样的。

3、带有索引位置的集合遍历

遍历集合时如果需要使用到集合的索引位置时,直接对集合迭代是没有索引信息的,普通的方式使用:

colors = ['red', 'green', 'blue', 'yellow']
for i in range(len(colors)):
print (i, '--->', colors[i])

pythonic

for i, color in enumerate(colors):
print (i, '--->', color)

4、字符串连接

字符串连接时,普通的方式可以用 + 操作

names = ['raymond', 'rachel', 'matthew', 'roger',
'betty', 'melissa', 'judith', 'charlie']
s = names[0]
for name in names[1:]:
s += ', ' + name
print (s)

pythonic

print (', '.join(names))

join 是一种更加高效的字符串连接方式,使用 + 操作时,每执行一次+操作就会导致在内存中生成一个新的字符串对象,遍历8次有8个字符串生成,造成无谓的内存浪费。而用 join 方法整个过程只会产生一个字符串对象。

5、打开/关闭文件

执行文件操作时,最后一定不能忘记的操作是关闭文件,即使报错了也要 close。普通的方式是在 finnally 块中显示的调用 close 方法。

f = open('data.txt')
try:
data = f.read()
finally:
f.close()

pythonic

with open('data.txt') as f:
data = f.read()

使用 with 语句,系统会在执行完文件操作后自动关闭文件对象。

6、列表推导式

能够用一行代码简明扼要地解决问题时,绝不要用两行,比如

result = []
for i in range(10):
s = i*2
result.append(s)

pythonic

[i*2 for i in xrange(10)]

与之类似的还有生成器表达式、字典推导式,都是很 pythonic 的写法。

7、善用装饰器

装饰器可以把与业务逻辑无关的代码抽离出来,让代码保持干净清爽,而且装饰器还能被多个地方重复利用。比如一个爬虫网页的函数,如果该 URL 曾经被爬过就直接从缓存中获取,否则爬下来之后加入到缓存,防止后续重复爬取。

def web_lookup(url, saved={}):
if url in saved:
return saved[url]
page = urllib.urlopen(url).read()
saved[url] = page
return page

pythonic

import urllib #py2
#import urllib.request as urllib # py3
def cache(func):
saved = {}
def wrapper(url):
if url in saved:
return saved[url]
else:
page = func(url)
saved[url] = page
return page
return wrapper
@cache
def web_lookup(url):
return urllib.urlopen(url).read()

用装饰器写代码表面上感觉代码量更多,但是它把缓存相关的逻辑抽离出来了,可以给更多的函数调用,这样总的代码量就会少很多,而且业务方法看起来简洁了。

8、合理使用列表
列表对象(list)是一个查询效率高于更新操作的数据结构,比如删除一个元素和插入一个元素时执行效率就非常低,因为还要对剩下的元素进行移动

names = ['raymond', 'rachel', 'matthew', 'roger',
'betty', 'melissa', 'judith', 'charlie']
names.pop(0)
names.insert(0, 'mark')

pythonic

from collections import deque
names = deque(['raymond', 'rachel', 'matthew', 'roger',
'betty', 'melissa', 'judith', 'charlie'])
names.popleft()
names.appendleft('mark')

deque 是一个双向队列的数据结构,删除元素和插入元素会很快

9、序列解包

p = 'vttalk', 'female', 30, 'python@qq.com'
name = p[0]
gender = p[1]
age = p[2]
email = p[3]

pythonic

name, gender, age, email = p

10、遍历字典的 key 和 value

方法一速度没那么快,因为每次迭代的时候还要重新进行hash查找 key 对应的 value。

方法二遇到字典非常大的时候,会导致内存的消耗增加一倍以上

# 方法一
for k in d:
print (k, '--->', d[k])
# 方法二
for k, v in d.items():
print (k, '--->', v)

pythonic

for k, v in d.iteritems():
print (k, '--->', v)

iteritems 返回迭代器对象,可节省更多的内存,不过在 python3 中没有该方法了,只有 items 方法,等值于 iteritems。

​当然还有很多 pythonic 写法,在此不再一一列举。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python图像灰度变换及图像数组操作
Jan 27 Python
Python多进程库multiprocessing中进程池Pool类的使用详解
Nov 24 Python
详解Python判定IP地址合法性的三种方法
Mar 06 Python
关于django 数据库迁移(migrate)应该知道的一些事
May 27 Python
Django-Rest-Framework 权限管理源码浅析(小结)
Nov 12 Python
Python3 集合set入门基础
Feb 10 Python
Python pyautogui模块实现鼠标键盘自动化方法详解
Feb 17 Python
django 读取图片到页面实例
Mar 27 Python
使用python matplotlib 画图导入到word中如何保证分辨率
Apr 16 Python
使用python批量修改XML文件中图像的depth值
Jul 22 Python
anaconda升级sklearn版本的实现方法
Feb 22 Python
Pytorch反向传播中的细节-计算梯度时的默认累加操作
Jun 05 Python
Python pip替换为阿里源的方法步骤
Jul 02 #Python
python set内置函数的具体使用
Jul 02 #Python
ZABBIX3.2使用python脚本实现监控报表的方法
Jul 02 #Python
浅谈python 中类属性共享的问题
Jul 02 #Python
如何通过Python实现标签云算法
Jul 02 #Python
对python特殊函数 __call__()的使用详解
Jul 02 #Python
对python 调用类属性的方法详解
Jul 02 #Python
You might like
php生成二维码时出现中文乱码的解决方法
2014/12/18 PHP
PHP编译configure时常见错误的总结
2017/08/17 PHP
PHP实现自动发送邮件功能代码(qq 邮箱)
2017/08/18 PHP
phpfpm的作用和用法
2019/10/10 PHP
javascript 关闭IE6、IE7
2009/06/01 Javascript
window.parent调用父框架时 ie跟火狐不兼容问题
2009/07/30 Javascript
Mootools 1.2教程(2) DOM选择器
2009/09/14 Javascript
jQuery 动态酷效果实现总结
2009/12/27 Javascript
对jQuery的事件绑定的一些思考(补充)
2013/04/20 Javascript
JS对象转换为Jquery对象示例
2014/01/26 Javascript
javascript中bind函数的作用实例介绍
2014/09/28 Javascript
jQuery实现跨域iframe接口方法调用
2015/03/14 Javascript
JavaScript获取两个数组交集的方法
2015/06/09 Javascript
jQuery实现右下角可缩放大小的层完整实例
2016/06/20 Javascript
AngularJS出现$http异步后台无法获取请求参数问题的解决方法
2016/11/03 Javascript
AngularJS 实现点击按钮获取验证码功能实例代码
2017/07/13 Javascript
vue移动端轻量级的轮播组件实现代码
2018/07/12 Javascript
vue+element的表格实现批量删除功能示例代码
2018/08/17 Javascript
JS/HTML5游戏常用算法之碰撞检测 包围盒检测算法详解【凹多边形的分离轴检测算法】
2018/12/13 Javascript
JS数组方法slice()用法实例分析
2020/01/18 Javascript
js实现页面图片消除效果
2020/03/24 Javascript
详解JavaScript匿名函数和闭包
2020/07/10 Javascript
解决Django migrate No changes detected 不能创建表的问题
2018/05/27 Python
Python 访问限制 private public的详细介绍
2018/10/16 Python
python用plt画图时,cmp设置方法
2018/12/13 Python
对python中词典的values值的修改或新增KEY详解
2019/01/20 Python
Python学习笔记之pandas索引列、过滤、分组、求和功能示例
2019/06/03 Python
如何通过python画loss曲线的方法
2019/06/26 Python
Python HTMLTestRunner如何下载生成报告
2020/09/04 Python
viagogo法国票务平台:演唱会、体育比赛、戏剧门票
2017/03/27 全球购物
英国网络托管和域名领导者:Web Hosting UK
2017/10/15 全球购物
西班牙土拨鼠床垫公司,感觉在云端:Marmota
2019/03/18 全球购物
自荐信格式的六要素
2013/09/21 职场文书
大学生毕业求职自荐书范文
2014/02/04 职场文书
质量负责人岗位职责
2015/02/15 职场文书
python tkinter模块的简单使用
2021/04/07 Python