标签:tomat err 文件名 文件中 条目 常量 变化 info inf
此编码风格指南主要基于 Google Python Style Guide [中译版],结合百度python使用习惯和实际开发情况制定。
这份文档存在的意义是让大家写出统一风格的代码,让百度的模块可维护性和可读性更好;
文档内容可能会与您的喜好冲突, 请尽量用包容的心态来接受; 不合理之处, 请反馈给py-styleguide@baidu.com
本规范可分为三大部分,分别对Python语法、风格、编程实践作出规定与建议。
每一部分有若干专题,每一专题下有若干条目。
条目是规范的基本组成部分,每一条目由规定、定义、解释、示例、参考等项组成。
建议每位python开发人员细读”python之禅”,理解规范背后的思想
"""
The Zen of Python
Beautiful is better than ugly.
Explicit is better than implicit.
Simple is better than complex.
Complex is better than complicated.
Flat is better than nested.
Sparse is better than dense.
Readability counts.
Special cases aren‘t special enough to break the rules.
Although practicality beats purity.
Errors should never pass silently.
Unless explicitly silenced.
In the face of ambiguity, refuse the temptation to guess.
There should be one-- and preferably only one --obvious way to do it.
Although that way may not be obvious at first unless you‘re Dutch.
Now is better than never.
Although never is often better than |right| now.
If the implementation is hard to explain, it‘s a bad idea.
If the implementation is easy to explain, it may be a good idea.
Namespaces are one honking great idea -- let‘s do more of those!
-- by Tim Peters
"""
·[强制] 禁止使用from xxx import yyy语法直接导入类或函数(即yyy只能是module或package,不能是类或函数)。
·[强制] [PY002] 禁止使用from xxx import *
·[强制] [PY003] import时必须使用package全路径名(相对PYTHONPATH),禁止使用相对路径(相对当前路径)。
·[建议] 可以使用异常。但使用前请务必详细了解异常的行为,谨慎使用
·[强制] [PY004] 禁止使用双参数形式(raise MyException, ‘Error Message‘)或字符串形式(raise ‘Error Message‘)语法抛异常。
·[强制] 如果需要自定义异常,应该在模块内定义名为Error的异常基类。该基类必须继承自Exception。其它异常类都从该Error类派生而来。
·[强制] 除非重新抛出异常,禁止使用except:捕获所有异常。
·[建议] 除非重新抛出异常,否则不建议捕获Exception或StandardError。如果捕获,必须在日志中记录所捕获异常信息。
·[强制] [PY007] 捕捉异常时,应当使用as语法,禁止使用逗号语法。
·[建议] 建议try中的代码尽可能少。避免catch住未预期的异常,掩藏掉真正的错误。
·[建议] 建议使用finally子句来执行那些无论try块中有没有异常都应该被执行的代码,这对于清理资源常常很有用。例如:文件关闭。
解释
示例
YES:
raise MyException
raise MyException(‘Error Message‘)
class Error(Exception):
pass
try:
raise Error
except Error as error:
pass
NO:
raise ‘Error Message‘
raise MyException, ‘Error Message‘
try:
raise Error
except Error, error:
pass
·[强制] 如果定义全局变量,必须写在文件头部。
解释
·[建议] 类构造函数应该尽量简单,不能包含可能失败或过于复杂的操作
解释
·[强制] [PY004] 函数返回值必须小于等于3个。3个以上时必须通过class/namedtuple/dict等具名形式进行包装
解释
示例
YES:
def get_numbers():
return 1, 2, 3
a, b, c = get_numbers()
class Person(object):
def __init__(self, name, gender, age, weight):
self.name = name
self.gender = gender
self.age = age
self.weight = weight
or
import collections
Person = collections.namedtuple(‘Person‘, ‘name gender age weight‘)
def get_person_info():
return Person(‘jjp‘, ‘MALE‘, 30, 130)
person = get_person_info()
NO:
def get_numbers():
return 1, 2, 3, 4
a, b, c, d = get_numbers()
def get_person_info():
return ‘jjp‘, ‘MALE‘, 30, 130
name, gender, age, weight = get_person_info()
·[建议] 不推荐使用嵌套/局部/内部类或函数
解释
·[强制] [PY011] 可以使用列表推导。mapping、loop、filter部分单独成行,且最多只能写一行。禁止多层loop或filter。
解释
示例
YES:
result = []
for x in range(10):
for y in range(5):
if x * y > 10:
result.append((x, y))
for x in xrange(5):
for y in xrange(5):
if x != y:
for z in xrange(5):
if y != z:
yield (x, y, z)
return ((x, complicated_transform(x))
for x in long_generator_function(parameter)
if x is not None)
squares = [x * x for x in range(10)]
eat(jelly_bean for jelly_bean in jelly_beans
if jelly_bean.color == ‘black‘)
NO:
result = [(x, y) for x in range(10) for y in range(5) if x * y > 10]
return ((x, y, z)
for x in xrange(5)
for y in xrange(5)
if x != y
for z in xrange(5)
if y != z)
·[强制] 对容器或文件的只读遍历,应该使用内置的迭代方法,不要使用返回list的方式遍历。
·[强制] [PY013] 对容器类型,使用in或not in判断元素是否存在。而不是has_key。
解释
当文件或容器较小时,调用返回list的函数也可以接受。请reviewer视具体情况决定是否通过。
如果遍历过程中需要对容器进行增删,请使用返回list的方式遍历
示例
Yes:
for key in adict: ...
if key not in adict: ...
if obj in alist: ...
for line in afile: ...
for k, v in dict.iteritems(): ...
# 删除毕业学生
for id in students.keys():
if students[id].graduated:
del students[id]
NO:
for key in adict.keys(): ...
if not adict.has_key(key): ...
for line in afile.readlines(): ...
#删除毕业学生
for id in students:
if students[id].graduated:
del students[id] # 抛出RuntimeError异常
·[建议] 当返回较长列表数据时建议使用yield和generator函数。
解释
示例
YES:
#返回n以内的奇数
def odds(n):
for i in xrange(1, n + 1):
if i % 2 == 1:
yield i
for i in odds(1000):
print i
NO:
#返回n以内的奇数
def odds(n):
ret = []
for i in xrange(1, n + 1):
if i % 2 == 1:
ret.append(i)
return ret
for i in odds(1000):
print i
·[强制] [PY014] 可以使用lambda函数,但仅限一行之内。
解释
·[强制] 条件表达式仅用于一行之内,禁止嵌套使用
解释
示例
YES:
# 单行简单条件表达式
x = true_value if cond else false_value
# 改写成if形式
if cond:
x = true_value
else
x = false_value
# 复杂if表达式
if t < 60:
unit = "seconds"
elif t < 3600:
unit = "minutes"
else
unit = "hours"
NO:
#嵌套条件表达式
unit = "seconds" if t < 60 else "minutes" if t < 3600 else "hours"
·[强制] [PY016] 仅可使用以下基本类型字面常量或常量作为默认参数:整数、bool、浮点、字符串、None
解释
示例
YES:
def foo(a, b=None):
if b is None:
b = []
No: def foo(a, b=[]):
...
No: def foo(a, b=time.time()): # The time the module was loaded???
...
No: def foo(a, b=FLAGS.my_thing): # sys.argv has not yet been parsed...
...
·[强制] 可以使用property。但禁止在派生类里改写property实现。
解释
示例
YES:
import math
class Square(object):
"""A square with two properties: a writable area and a read-only perimeter.
To use:
>>> sq = Square(3)
>>> sq.area
9
>>> sq.perimeter
12
>>> sq.area = 16
>>> sq.side
4
>>> sq.perimeter
16
"""
def __init__(self, side):
self.side = side
def __get_area(self):
"""Calculates the ‘area‘ property."""
return self.side ** 2
def __set_area(self, area):
"""Sets the ‘area‘ property."""
self.side = math.sqrt(area)
area = property(__get_area, __set_area,
doc="""Gets or sets the area of the square.""")
@property
def perimeter(self):
return self.side * 4
NO:
class MySquare(Square):
def __get_area(self):
return math.pi * self.side ** 2 # overwrite doesn‘t work
def __set_area(self, area):
self.side = math.sqrt(area / math.pi) # overwrite doesn‘t work
·[建议] 建议显式转换到bool类型,慎用到bool类型的隐式转换。如使用隐式转换,你需要确保充分了解其语义
·[强制] [PY018] 禁止使用==或!=判断表达式是否为None,应该用is或is not None
·[强制] 当明确expr为bool类型时,禁止使用==或!=与True/False比较。应该替换为expr或not expr
·[强制] 判断某个整数表达式expr是否为零时,禁止使用not expr,应该使用expr == 0
解释
python中None、空字符串、0、空tuple、list、dict都会被隐式转换为False,这可能和用户预期的行为不一致。
为便于不太熟悉python语言的其它语言开发者理解python代码,建议”显式”表明到bool类型的转换语义
运算符==或!=的结果取决于__eq__函数,可能出现obj is not None,但obj==None的情况
当判断expr是否为0时,若expr为None,not expr也会返回True。一般这不是用户期望的行为
示例
YES:
if users is None or len(users) == 0:
print ‘no users‘
if foo == 0:
self.handle_zero()
if i % 10 == 0:
self.handle_multiple_of_ten()
Cautious:
if not users:
print "no users"
NO:
if len(users) == 0:
print ‘no users‘
if foo is not None and not foo:
self.handle_zero()
if not i % 10:
self.handle_multiple_of_ten()
解释
·[强制] [PY021] 禁止以分号结束语句
·[强制] [PY022] 一行只能写一条语句,没有例外情况
·[强制] [PY023] 每行不得超过100个字符
·[强制] [PY024] 函数长度不得超过100行
解释
示例
x = (‘This will build a very long long ‘
‘long long long long long long string‘)
·[建议] 除非用于明确算术表达式优先级、tuple或者隐式行连接,否则尽量避免冗余的括号。
解释
示例
YES:
if foo:
bar()
while x:
x = bar()
if x and y:
bar()
if not x:
bar()
return foo
for (x, y) in dict.iteritems(): ...
NO:
if (x):
bar()
if not(x):
bar()
return (foo)
·[强制] 使用4个空格缩进,禁止使用tab缩进。
·[强制] 把单行内容拆成多行写时,要么与首行保持对齐;要么首行留空,从第二行起统一缩进4个空格;为与后面的代码区分,可以使用8空格缩进。
解释
不同编辑器对TAB的设定可能不同,使用TAB容易造成在一些编辑器下代码混乱,所以建议一率转换成空格。
示例
YES:
# Aligned with opening delimiter
foo = long_function_name(var_one, var_two,
var_three, var_four)
# Aligned with opening delimiter in a dictionary
foo = {
long_dictionary_key: value1 +
value2,
...
}
# 4-space hanging indent; nothing on first line
foo = long_function_name(
var_one, var_two, var_three,
var_four)
# 4-space hanging indent in a dictionary
foo = {
long_dictionary_key:
long_dictionary_value,
...
}
NO:
# Stuff on first line forbidden
foo = long_function_name(var_one, var_two,
var_three, var_four)
# 2-space hanging indent forbidden
foo = long_function_name(
var_one, var_two, var_three,
var_four)
# No hanging indent in a dictionary
foo = {
long_dictionary_key:
long_dictionary_value,
...
}
·[强制] [PY027] 文件级定义(类或全局函数)之间隔两个空行,类方法之间隔一个空行
·[强制] [PY028] 圆括号、方括号、花括号内侧都不加空格
Yes: spam(ham[1], {eggs: 2}, [])
No: spam( ham[ 1 ], { eggs: 2 }, [ ] )
·[强制] [PY029] 参数列表, 索引或切片的左括号前不应加空格
Yes: spam(1)
Yes: dict[‘key‘] = list[index]
No: spam (1)
No: dict [‘key‘] = list [index]
·[强制] [PY030] 逗号、分号、冒号前不加空格,后边加一个空格
Yes:
if x == 4:
print x, y
x, y = y, x
No:
if x == 4 :
print x , y
x , y = y , x
·[强制] [PY031] 所有二元运算符前后各加一个空格
Yes: x == 1
No: x<1
·[强制] [PY032] 关键字参数或参数默认值里的等号前后不加空格
Yes: def complex(real, imag=0.0): return magic(r=real, i=imag)
No: def complex(real, imag = 0.0): return magic(r = real, i = imag)
·[强制] [PY033] 使用docstring描述module、function、class和method接口。docstring必须用三个双引号括起来。
·[强制] 对外接口部分必须用docstring描述,内部接口视情况自行决定是否写docstring。
·[强制][PY034] 接口的docstring描述至少包括功能简介、参数、返回值。如果可能抛出异常,必须注明。
·[强制] 每个文件都必须有文件声明,文件声明必须包括以下信息:版权声明,功能和用途简介,修改人及联系方式。
·[建议] TODO注释格式必须为:
# TODO: 干什么事情$负责人(邮箱前缀)$最终期限(YYYY-MM-DD)$
定义
解释
示例
################################################################################
#
# Copyright (c) 2014 Baidu.com, Inc. All Rights Reserved
#
################################################################################
"""
This module provide configure file management service in i18n environment.
Authors: jiangjinpeng(jiangjinpeng@baidu.com)
Date: 2014/04/05 17:23:06
"""
def fetch_bigtable_rows(big_table, keys, other_silly_variable=None):
"""Fetches rows from a Bigtable.
Retrieves rows pertaining to the given keys from the Table instance
represented by big_table. Silly things may happen if
other_silly_variable is not None.
Args:
big_table: An open Bigtable Table instance.
keys: A sequence of strings representing the key of each table row
to fetch.
other_silly_variable: Another optional variable, that has a much
longer name than the other args, and which does nothing.
Returns:
A dict mapping keys to the corresponding table row data
fetched. Each row is represented as a tuple of strings. For
example:
{‘Serak‘: (‘Rigel VII‘, ‘Preparer‘),
‘Zim‘: (‘Irk‘, ‘Invader‘),
‘Lrrr‘: (‘Omicron Persei 8‘, ‘Emperor‘)}
If a key from the keys argument is missing from the dictionary,
then that row was not found in the table.
Raises:
IOError: An error occurred accessing the bigtable.Table object.
"""
pass
class SampleClass(object):
"""Summary of class here.
Longer class information....
Longer class information....
Attributes:
likes_spam: A boolean indicating if we like SPAM or not.
eggs: An integer count of the eggs we have laid.
"""
def __init__(self, likes_spam=False):
"""Inits SampleClass with blah."""
self.likes_spam = likes_spam
self.eggs = 0
def public_method(self):
"""Performs operation blah."""
# TODO: Improve performance using concurrent operation. $jiangjinpeng$2014-04-05$
·[强制] [PY037] 每行只能导入一个库
YES:
import os
import sys
from third.party import lib
from third.party import foobar as fb
import my.own.module
NO:
import os, sys
from third.party import lib, foobar
·[强制] [PY039] 类(包括异常)名使用首字母大写驼峰式命名
·[强制] 常量使用全大写字母,单词间用下划线分隔
·[强制] 其它情况(目录/文件/package/module/function/method/variable/parameter)一律使用全小写字母,单词间用下划线分隔
·[强制] protected成员使用单下划线前缀,private成员使用双下划线前缀
·[强制] [PY043] 禁止使用双下划线开头,双下划线结尾的名字(类似__init__)
解释
双下划线开头、结尾的名字对python解释器有特殊意义,可能与内部关键字冲突
示例
ClassName, ExceptionName
GLOBAL_CONSTANT_NAME, CLASS_CONSTANT_NAME,
module_name, package_name, method_name, function_name, global_var_name, instance_var_name, function_parameter_name, local_var_name
_InternalClassName, _INTERNAL_CONSTANT_NAME, _internal_function_name, _protected_member_name, __private_member_name
·[建议] 模块的主程序必须以#!/usr/bin/env python开头。如果明确只支持某个python版本,请带上python版本号
·[建议] 模块可以自带某个特定版本的python环境一起发布。需要在程序的启动脚本中指定具体使用的python解释器程序
·[建议] 推荐使用2.7版本(含)以上的python解释器
解释
·[强制] 如果文件包含非ASCII字符,必须在文件前两行标明字符编码。
·[强制] 只能使用UTF-8或GB18030编码。推荐使用UTF-8编码,如果项目确有需要,可以使用GB18030
解释
# -*- coding: utf-8 -*-
·[强制] [PY046] 如果一个类没有基类,必须继承自object类。
解释
示例
YES:
class SampleClass(object):
pass
class OuterClass(object):
class InnerClass(object):
pass
class ChildClass(ParentClass):
"""Explicitly inherits from another class already."""
NO:
class SampleClass:
pass
class OuterClass:
class InnerClass:
pass
·[强制] 除了a+b这种最简单的情况外,应该使用%或format格式化字符串。
解释
Yes: x = a + b
x = ‘%s, %s!‘ % (imperative, expletive)
x = ‘{}, {}!‘.format(imperative, expletive)
x = ‘name: %s; score: %d‘ % (name, n)
x = ‘name: {}; score: {}‘.format(name, n)
No: x = ‘%s%s‘ % (a, b) # use + in this case
x = ‘{}{}‘.format(a, b) # use + in this case
x = imperative + ‘, ‘ + expletive + ‘!‘
x = ‘name: ‘ + name + ‘; score: ‘ + str(n)
·[强制] 不要使用+=拼接字符串列表,应该使用join。
解释
Yes: items = [‘<table>‘]
for last_name, first_name in employee_list:
items.append(‘<tr><td>%s, %s</td></tr>‘ % (last_name, first_name))
items.append(‘</table>‘)
employee_table = ‘‘.join(items)
No: employee_table = ‘<table>‘
for last_name, first_name in employee_list:
employee_table += ‘<tr><td>%s, %s</td></tr>‘ % (last_name, first_name)
employee_table += ‘</table>‘
·[强制] 用完文件或socket后必须显式关闭句柄。建议使用with语法简化开发
解释
示例
with open("hello.txt") as hello_file:
for line in hello_file:
print line
·[强制] 所有module都必须可导入。如需要执行主程序,必须检查__name__ == ‘__main__‘
解释
示例
def main():
...
if __name__ == ‘__main__‘:
main()
·[建议] 推荐使用PyUnit做单元测试。是否需要做单元测试以及目标单测覆盖率由项目负责人自行决定。
·[建议] 推荐测试代码放在单独的test目录中。如果被测试代码文件名为xxx.py,那么测试代码文件应该被命名为xxx_test.py
示例
#!/usr/bin/env python
# -*- coding: gb18030 -*-
import unittest
import math
class MathTestCase(unittest.TestCase):
def test_sqrt(self):
self.assertEqual(math.sqrt(4) * math.sqrt(4), 4)
if __name__ == "__main__":
unittest.main()
参考
·[建议] 推荐使用python自带的logging库打印日志。
·[建议] 推荐默认日志格式:"%(levelname)s: %(asctime)s: %(filename)s:%(lineno)d * %(thread)d %(message)s", 时间格式:"%Y-%m-%d %H:%M:%S"
·[建议] 推荐线上程序使用两个日志文件:一个专门记录warning/error/critical日志,另一个记录所有日志。
解释
示例
import os
import logging
import logging.handlers
def init_log(log_path, level=logging.INFO, when="D", backup=7,
format="%(levelname)s: %(asctime)s: %(filename)s:%(lineno)d * %(thread)d %(message)s",
datefmt="%m-%d %H:%M:%S"):
"""
init_log - initialize log module
Args:
log_path - Log file path prefix.
Log data will go to two files: log_path.log and log_path.log.wf
Any non-exist parent directories will be created automatically
level - msg above the level will be displayed
DEBUG < INFO < WARNING < ERROR < CRITICAL
the default value is logging.INFO
when - how to split the log file by time interval
‘S‘ : Seconds
‘M‘ : Minutes
‘H‘ : Hours
‘D‘ : Days
‘W‘ : Week day
default value: ‘D‘
format - format of the log
default format:
%(levelname)s: %(asctime)s: %(filename)s:%(lineno)d * %(thread)d %(message)s
INFO: 12-09 18:02:42: log.py:40 * 139814749787872 HELLO WORLD
backup - how many backup file to keep
default value: 7
Raises:
OSError: fail to create log directories
IOError: fail to open log file
"""
formatter = logging.Formatter(format, datefmt)
logger = logging.getLogger()
logger.setLevel(level)
dir = os.path.dirname(log_path)
if not os.path.isdir(dir):
os.makedirs(dir)
handler = logging.handlers.TimedRotatingFileHandler(log_path + ".log",
when=when,
backupCount=backup)
handler.setLevel(level)
handler.setFormatter(formatter)
logger.addHandler(handler)
handler = logging.handlers.TimedRotatingFileHandler(log_path + ".log.wf",
when=when,
backupCount=backup)
handler.setLevel(logging.WARNING)
handler.setFormatter(formatter)
logger.addHandler(handler)
import log
def main():
log.init_log("./log/my_program") # 日志保存到./log/my_program.log和./log/my_program.log.wf,按天切割,保留7天
logging.info("Hello World!!!")
参考
标签:tomat err 文件名 文件中 条目 常量 变化 info inf
原文地址:http://www.cnblogs.com/li-daphne/p/7245923.html