前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Google Python风格规范(上)

Google Python风格规范(上)

作者头像
王强
发布于 2018-08-09 09:29:25
发布于 2018-08-09 09:29:25
71100
代码可运行
举报
文章被收录于专栏:Python爬虫实战Python爬虫实战
运行总次数:0
代码可运行

好的代码风格,给人舒服的感觉,今天介绍一下谷歌的Python风格规范,由于规范比较多,将分为两次介绍。

1 分号

不要在行尾加分号, 也不要用分号将两条命令放在同一行。

2 行长度

每行不超过80个字符。

例外:

  1. 长的导入模块语句
  2. 注释里的URL

不要使用反斜杠连接行。

Python会将圆括号、中括号和花括号的行隐式的连接起来,可以利用这个特点。如果需要,可以在表达式外围增加一堆额外的圆括号。

如果一个文本字符串在一行放不下, 可以使用圆括号来实现隐式行连接:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
x = ('This will build a very long long '
     'long long long long long long string')

在注释中,如果必要,将长的URL放在一行上。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Yes:  
# See details at
# http://www.example.com/us/developer/documentation/api/content/v2.0/csv_file_name_extension_full_specification.html


No:  
# See details at
# http://www.example.com/us/developer/documentation/api/content/\
# v2.0/csv_file_name_extension_full_specification.html

注意上面例子中的元素缩进; 你可以在本文的 缩进部分找到解释。

3 括号

宁缺毋滥得使用括号。

除非是用于实现行连接,否则不要在返回语句或条件语句中使用括号。不过在元组两遍使用括号是可以的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Yes: 
if foo:
    bar() 
while x:
    x = bar()     
if x and y:
    bar()     
if not x:
    bar()     
return foo     
    
for (x, y) in dict.items(): 
    ...

No:  
if (x):
    bar()     
if not(x):
    bar()     
return (foo)

4 缩进

用四个空格来缩进代码。 绝对不要用tab,也不要tab和空格混用。对于行连接的情况,应该垂直对其换行的元素,或者使用4空格的悬挂式缩进:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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 dictionaryfoo = {
    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,
        ...
    }

5 空行

顶级定义(比如函数或者类定义)之间空两行,方法定义之间空一行。

6 空格

按照标准的排版规范来使用标点两边的空格。

  1. 括号内不要有空格。 Yes: spam(ham[1], {eggs: 2}, []) No: spam( ham[ 1 ], { eggs: 2 }, [ ] )
  2. 不要在逗号,分号,冒号前面加空格,但应该在它们后面加(除了在行尾)。 Yes: if x == 4: print x, y x, y = y, x No: if x == 4 : print x , y x , y = y , x
  3. 参数列表, 索引或切片的左括号前不应加空格。 Yes: spam(1) no: spam (1) Yes: dict['key'] = list[index] No: dict ['key'] = list [index]
  4. 在二元操作符两边都加上一个空格, 比如赋值(=), 比较(==, <, >, !=, <>, <=, >=, in, not in, is, is not), 布尔(and, or, not)。 至于算术操作符两边的空格该如何使用, 需要你自己好好判断。 不过两侧务必要保持一致。 Yes: x == 1 No: x<1
  5. 当’=’用于指示关键字参数或默认参数值时, 不要在其两侧使用空格。 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)
  6. 不要用空格来垂直对齐多行间的标记, 因为这会成为维护的负担(适用于:, #, =等): Yes: foo = 1000 # comment long_name = 2 # comment that should not be aligned dictionary = { "foo": 1, "long_name": 2, } No: foo = 1000 # comment long_name = 2 # comment that should not be aligned dictionary = { "foo" : 1, "long_name": 2, }

7 Shebang

大部分.py文件不必以#!作为文件的开始. 根据 PEP-394, 程序的main文件应该以 #!/usr/bin/python2或者 #!/usr/bin/python3开始。

注:在计算机科学中,Shebang(也称为 Hashbang )是一个由井号和叹号构成的字符序列 #! ,其出现在文本文件的第一行的前两个字符。 在文件中存在 Shebang 的情况下,类 Unix 操作系统的程序载入器会分析 Shebang 后的内容,将这些内容作为解释器指令,并调用该指令,并将载有 Shebang 的文件路径作为该解释器的参数。

#!先用于帮助内核找到Python解释器, 但是在导入模块时, 将会被忽略. 因此只有被直接执行的文件中才有必要加入#!。

8 注释

确保对模块, 函数, 方法和行内注释使用正确的风格。

  1. 文档字符串 Python有一种独一无二的的注释方式:使用文档字符串。 文档字符串是包, 模块, 类或函数里的第一个语句。 这些字符串可以通过对象的doc成员被自动提取,并且被pydoc所用。 我们对文档字符串的惯例是使用三重双引号‘’‘ ’‘’( PEP-257 )。 一个文档字符串应该这样组织:首先是一行以句号、问号或惊叹号结尾的概述(或者该文档字符串单纯只有一行)、 接着是一个空行, 接着是文档字符串剩下的部分,它应该与文档字符串的第一行的第一个引号对齐。 下面有更多文档字符串的格式化规范。
  2. 模块 每个文件应该包含一个许可样板。根据项目使用的许可(例如: Apache 2.0,,BSD,LGPL, GPL), 选择合适的样板。
  3. 函数和方法 下文所指的函数,包括函数, 方法, 以及生成器。 一个函数必须要有文档字符串, 除非它满足以下条件: 文档字符串应该包含函数做什么,以及输入和输出的详细描述。通常, 不应该描述”怎么做”,除非是一些复杂的算法。 文档字符串应该提供足够的信息, 当别人编写代码调用该函数时,他不需要看一行代码,只要看文档字符串就可以了。 对于复杂的代码, 在代码旁边加注释会比使用文档字符串更有意义。 关于函数的几个方面应该在特定的小节中进行描述记录, 这几个方面如下文所述。 每节应该以一个标题行开始。 标题行以冒号结尾。除标题行外, 节的其他内容应被缩进2个空格。 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
    • Args: 列出每个参数的名字, 并在名字后使用一个冒号和一个空格,分隔对该参数的描述。如果描述太长超过了单行80字符,使用2或者4个空格的悬挂缩进(与文件其他部分保持一致)。描述应该包括所需的类型和含义。 如果一个函数接受*foo(可变长度参数列表)或者**bar (任意关键字参数), 应该详细列出foo和*bar。
    • Returns: (或者 Yields: 用于生成器) 描述返回值的类型和语义。如果函数返回None, 这一部分可以省略。
    • Raises: 列出与接口有关的所有异常。
    • 外部不可见
    • 非常短小
    • 简单明了
  4. 类应该在其定义下有一个用于描述该类的文档字符串。 如果你的类有公共属性(Attributes),那么文档中应该有一个属性(Attributes)段。 并且应该遵守和函数参数相同的格式。
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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."""
  1. 块注释和行注释 最需要写注释的是代码中那些技巧性的部分。如果你在下次 代码审查的时候必须解释一下,那么你应该现在就给它写注释。 对于复杂的操作, 应该在其操作开始前写上若干行注释。对于不是一目了然的代码,应在其行尾添加注释。
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# We use a weighted dictionary search to find out where i is in
# the array.  We extrapolate position based on the largest num
# in the array and the array size and then do binary search to
# get the exact number.

if i & (i-1) == 0:        # true iff i is a power of 2

为了提高可读性,注释应该至少离开代码2个空格。

另一方面,绝不要描述代码。假设阅读代码的人比你更懂Python。 他只是不知道你的代码要做什么。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# BAD COMMENT: Now go through the b array and make sure whenever i occurs
# the next element is i+1

9 类

如果一个类不继承自其它类, 就显式的从object继承. 嵌套类也一样。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Yes: 
class SampleClass(object):
    passclass OuterClass(object):
    class InnerClass(object):
        passclass ChildClass(ParentClass):
    """Explicitly inherits from another class already."""No: 
class SampleClass:
    passclass OuterClass:
    class InnerClass:
        pass

继承自 object 是为了使属性(properties)正常工作, 并且这样可以保护你的代码, 使其不受 PEP-3000的一个特殊的潜在不兼容性影响。 这样做也定义了一些特殊的方法, 这些方法实现了对象的默认语义, 包括 __new__, __init__, __delattr__, __getattribute__, __setattr__, __hash__, __repr__, and __str__


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

本文分享自 C与Python实战 微信公众号,前往查看

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

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

评论
登录后参与评论
暂无评论
推荐阅读
C# 的事件,一般你不需要担心它的线程安全问题!
时不时会有小伙伴跟我提到在 C# 写事件 += -= 以及 Invoke 时可能遇到线程安全问题。然而实际上这些操作并不会有线程安全问题,所以我特别写一篇博客来说明一下,从原理层面说说为什么不会有线程安全问题。
walterlv
2023/10/22
4410
.Neter所应该彻底了解的委托
本文将通过引出几个问题来,并且通过例子来剖析C#中的委托以及用法,做抛砖引玉的作用
ryzenWzd
2020/04/29
4220
.NET基础拾遗(4)委托、事件、反射与特性
  委托这个概念对C++程序员来说并不陌生,因为它和C++中的函数指针非常类似,很多码农也喜欢称委托为安全的函数指针。无论这一说法是否正确,委托的的确确实现了和函数指针类似的功能,那就是提供了程序回调指定方法的机制。
Edison Zhou
2018/08/20
1K0
.NET基础拾遗(4)委托、事件、反射与特性
C#中的委托和事件 - Part.1
文中代码在VS2005下通过,由于VS2003(.Net Framework 1.1)不支持隐式的委托变量,所以如果在一个接受委托类型的位置直接赋予方法名,在VS2003下会报错,解决办法是显式的创建一个委托类型的实例(委托变量)。例如:委托类型 委托实例 = new 委托类型(方法名);
张子阳
2018/09/30
1.1K0
C#中的委托和事件 - Part.1
C#委托与事件
Delegate ----   C# 中的委托(Delegate)类似于 C 或 C++ 中函数的指针。委托 是存有对某个方法的引用的一种引用类型变量。引用可在运行时被改变。委托特别用于实现事件和回调方法。所有的委托(Delegate)都派生自 System.Delegate 类。 声明委托   声明委托的语法如下: delegate return-type delegate-name (parameter-list);   其中:return-type是返回类型,delegate-name是委托名字,pa
拾点阳光
2018/05/10
1K0
C# 事件
一、前言:前面的随笔中说完了委托,现在看看事件到底可以干什么,在前面的随笔中,使用委托的过程中,有一个很别扭,也很显然易见的问题,就是委托第一次必须初始化用"=",绑定二次事件用"+="这个是非常的,
郑小超.
2018/01/24
6420
C#  事件
C# 中的委托和事件
文中代码在VS2005下通过,由于VS2003(.Net Framework 1.1)不支持隐式的委托变量,所以如果在一个接受委托类型的位置直接赋予方法名,在VS2003下会报错,解决办法是显式的创建一个委托类型的实例(委托变量)。例如:委托类型 委托实例 = new 委托类型(方法名); 欢迎浏览本文的后续文章: C#中的委托和事件(续)
Edison Zhou
2018/08/20
9430
C# 中的委托和事件
.Net中观察者模式(Observer):C#事件
奥运会参加百米的田径运动员听到枪声,比赛立即进行。其中枪声是事件,而运动员比赛就是这个事件发生后的动作。不参加该项比赛的人对枪声没有反应。
小蜜蜂
2019/07/15
7690
.Net中观察者模式(Observer):C#事件
C# 学习笔记(7)—— 事件
事件是基于委托之上的新特性,自然也是 C# 独有的概念。事件理解起来不难,意思就是他的字面意思,就是我们日常理解的事件
Karl Du
2023/10/20
2350
[C#1] 9-委托
委托揭秘 编译器和CLR在后台做了很多工作来隐藏委托本身的复杂性,如下一句委托声明: //编译器为我们产生了一个同名的类 public delegate void MyDelegate(int i);
blackheart
2018/01/19
8070
[C#1] 9-委托
C#委托与事件学习笔记
      今天跟随视频学习了一下C#中最重要的一些概念之委托与事件。老杨的视频讲的还是挺深入浅出,不过刚接触C#.NET的人还是朦朦胧胧,就像张子阳先生说的“每次见到委托和事件就觉得心里别(biè)得慌,混身不自在”。跨过这道坎的人就有种一览众山小的感觉了。我又浏览了皱华栋老师JamesZou的博文《深入理解C#委托及原理》(地址:http://www.cnblogs.com/jameszou/archive/2011/07/21/2112497.html),以及张子阳Jimmy Zhang的博文《C# 中的委托和事件》(地址:http://www.cnblogs.com/jimmyzhang/archive/2007/09/23/903360.html)总算对委托有了一点理性的感觉了,在此谢谢ITCAST,JamesZou以及Jimmmy Zhang的博文,谢谢。
Edison Zhou
2018/08/20
4730
C#委托与事件学习笔记
庖丁解牛——深入解析委托和事件
这篇博文我不讲委托和事件的概念,因为大段的文字概念没有任何意义。 具体想了解,委托和事件的概念可以MSDN查阅。 我这篇文章的主题思路是委托如何一步步进化成事件: 何为委托--->委托来实现事件--->用方法对委托的封装--->Event的 add,remove方法的引入--->标准事件写法--->反编译探究竟。 用几个例子以及Reflector反编译探究委托和事件的关系。不足之处,还望多多指教... 何为委托: 首先,委托是一种类型,是一种定义了方法签名的类型。 委托可以理解为函数指针(安全
用户1161731
2018/03/12
1K0
庖丁解牛——深入解析委托和事件
.NET面试题系列[7] - 委托与事件
C#中的委托可以说俯拾即是,从LINQ中的lambda表达式到(包括但不限于)winform,wpf中的各种事件都有着委托的身影。C#中如果没有了事件,那绝对是一场灾难,令开发者寸步难行。而委托又是事件的基础,可以说是C#的精髓,个人认为,其地位如同指针之于C语言。
s055523
2018/09/14
1.2K0
.NET面试题系列[7] - 委托与事件
事件与委托的区别就是“+=”和“-="?
事件的作用与委托变量一样,只是功能上比委托变量有更多的限制。(比如:1.只能通过+=或 -= 来绑定方法(事件处理程序)2.只能在类内部调用(触发)事件。)
静心物语313
2020/03/24
1.3K0
事件与委托的区别就是“+=”和“-="?
理解委托类型
delegate 是一种可用于封装命名或匿名方法的引用类型。(并不是所有的引用类型都是class)
lulianqi
2018/09/07
7390
理解委托类型
C#基础篇 - 理解委托和事件
委托类似于C++中的函数指针(一个指向内存位置的指针)。委托是C#中类型安全的,可以订阅一个或多个具有相同签名方法的函数指针。简单理解,委托是一种可以把函数当做参数传递的类型。很多情况下,某个函数需要动态地去调用某一类函数,这时候我们就在参数列表放一个委托当做函数的占位符。在某些场景下,使用委托来调用方法能达到减少代码量,实现某种功能的用途。
Esofar
2018/09/05
5420
C#基础篇 - 理解委托和事件
深入分析委托与事件—C#综合揭秘——细说多线程
本篇文章将为你介绍一下 Delegate 的使用方式,逐渐揭开 C# 当中事件(Event)的由来,它能使处理委托类型的过程变得更加简单。 还将为您解释委托的协变与逆变,以及如何使用 Delegate 使 Observer(观察者)模式的使用变得更加简单。 在事件的介绍上,会讲述事件的使用方式,并以ASP.NET的用户控件为例子,介绍一下自定义事件的使用。 最后一节,将介绍Predicate<T>、Action<T>、Func<T,TResult>多种泛型委托的使用和Lambda的发展过程与其使用方式。 因为时间仓促,文中有错误的地方敬请点评。
vv彭
2021/01/06
2.2K0
[读书笔记]C#学习笔记二: 委托和事件的用法及不同.
前言:  C#委托是什么     c#中的委托可以理解为函数的一个包装, 它使得C#中的函数可以作为参数来被传递, 这在作用上相当于C++中的函数指针. C++用函数指针获取函数的入口地址, 然后通过这个指针来实现对函数的操作. 委托的定义和方法的定义类似, 只是在定义的前面多了一个delegate关键字. 正文: 委托可以被视为一个更高级的指针,它不仅仅能把地址传指向另一个函数,而且还能传递参数,返回值等多个信息。 系统还为委托对象自动生成了同步,异步的调用方式,开发人员使用BeginInvok
一枝花算不算浪漫
2018/05/18
6540
java多线程的常用方法(以及注意事项)
1 /* 2 * 线程的常用方法 3 * 1.start(); 4 * 2.run(); 5 * 3.sleep(int millsecond); 6 * 4.isAlive(); -->判断线程是否还在运行 7 * 5.currentThread(); -->返回当前正在使用CPU资源的线程 8 * 6.interrupt();
Gxjun
2018/03/22
7350
java多线程的常用方法(以及注意事项)
C#编写简单的聊天程序
这是一篇基于Socket进行网络编程的入门文章,我对于网络编程的学习并不够深入,这篇文章是对于自己知识的一个巩固,同时希望能为初学的朋友提供一点参考。文章大体分为四个部分:程序的分析与设计、C#网络编程基础(篇外篇)、聊天程序的实现模式、程序实现。
张子阳
2018/09/30
1.6K0
C#编写简单的聊天程序
相关推荐
C# 的事件,一般你不需要担心它的线程安全问题!
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档