Python中 if name == ‘__main__‘的详细讲解及Python中的命名规则

1、if name == ‘main‘: 是干啥的?

先给出结论,对该语句用法简单的解释就是: 如果if __name__ == '__main__' 所在模块是被直接运行的,则该语句下代码块被运行,如果所在模块是被导入到其他的python脚本中运行的,则该语句下代码块不被运行。


2、程序入口的理解

对于很多编程语言来说,程序都必须要有一个入口,比如 C,C++,以及完全面向对象的编程语言 Java,C# 等。其中C 和 C++ 都需要有一个 main 函数来作为程序的入口,也就是程序的运行会从 main 函数开始。同样的,Java 和 C# 则必须要有一个包含 Main 方法的主类来作为程序入口。

但和C,C++、Java 以及 C# 等有所不同的是,Python属于脚本语言,不像编译型语言那样先将程序编译成二进制再运行,python是动态的逐行解释运行,也就是从脚本第一行开始运行,没有统一的入口。

另外我们知道,python文件有两种使用方法:

  • 第一种:直接作为脚本执行
  • 第二种:import到其他的python脚本中被调用(模块重用)执行,也就是作为模块(库)被导入执行

不管是直接运行还是导入,最顶层的代码都会被运行(Python 用缩进来区分代码层次)。但实际使用过程中,python文件作为模块被其它python脚本调用执行的时候,有的代码我们是不希望被运行的。

if __name__ == '__main__' 的作用就是定义这两种情况执行代码的执行方式,在该语句下的代码只有在文件作为脚本直接执行情况下才会被执行,而import到其他脚本中是不会被执行的。


3、程序案例

第一步:首先创建一个 param.py 的文件,其中代码如下:

1
2
3
4
5
pi = 3.1415926

def main():
print("pi:", pi)
main()

在param.py文件里定义了一个参数 pi,直接执行该文件输出:

1
pi: 3.1415926

第二步:然后创建 calculate.py 文件,用于计算圆的面积,该文件里边需要用到 param.py 文件中的 pi 参数,需要我们从 param.py 中把 pi 变量导入到 calculate.py 中:

1
2
3
4
5
6
7
8
9
10
from param import pi

def area_calculate(r):
s = pi * (r ** 2)
return s

def main():
print("The area of the circle: ", area_calculate(2))

main()

运行 calculate.py(python calculate.py),输出结果:

1
2
pi: 3.1415926
The area of the circle: 12.5663704

从输出结果可以看出,param.py 中的 main 函数也被运行了,实际上我们不希望它被运行,这时我们就可以用if __name__ == '__main__' 对 param.py 做修改:

1
2
3
4
5
6
7
pi = 3.1415926

def main():
print("pi:", pi)

if __name__ == "__main__":
main()

再次运行param.py,输出如下:

1
pi: 3.1415926

再次运行 calculate.py,输出如下:

1
The area of the circle:  12.5663704

对比可以看出,再次运行 param.py ,if __name__ == '__main__'语句之前和之后的代码都被执行,仍然输出 pi: 3.1415926 ,但再次运行 calculate.py,导入的 param.py 里if __name__ == '__main__'语句之后的代码已经不被执行,输出已经不包含 pi: 3.1415926。

if __name__ == '__main__'就相当于是 Python 模拟的程序入口。Python 本身并没有规定这么写,这只是一种编码习惯。由于模块之间相互引用,不同模块可能都有这样的定义,而入口程序只能有一个。到底哪个入口程序被选中,这取决于__name__的值。而__name__是内置变量,用于表示当前模块的名字。


4、运行原理

每个python模块(python文件,也就是此处的 param.py 和 calculate.py)都包含内置的变量__name__,当运行模块被执行的时候__name__等于文件名(包含了后缀.py);如果import到其他模块中,则__name__等于模块名称(不包含后缀.py)。而'__main__'等于当前执行文件的名称(包含了后缀.py)。进而当模块被直接执行时,__name__ == '__main__'结果为真,所以后续代码可以继续执行。

举例说明。
我们在 param.py 脚本的if __name__ == '__main__'之前加入print(__name__),即将__name__打印出来。文件内容如下:

1
2
3
4
5
6
7
8
9
pi = 3.1415926

def main():
print("pi:", pi)

print(__name__)

if __name__ == "__main__":
main()

运行结果输出:

1
2
__main__
pi: 3.1415926

可以看出,此时变量__name__的值为'__main__'

再执行 calculate.py,执行结果如下:

1
2
param
The area of the circle: 12.5663704

此时,param.py 中的__name__变量值为 param,__name__ == '__main__'条件为假,所以无法执行其后的代码。


5、Python中的命名规则

【1】Python项目组成

1、顶层的包(类似文件夹)。
2、各个模块(Python file)。
3、类(class 定义类)。
4、变量和方法(函数)。其中,方法中也可以定义变量(方法中的变量)。


【2】命名规范

1、项目:首字母大写+大写式驼峰, 如:ProjectName;
2、包:使用小写字母命名,不推荐使用下划线。
3、模块:使用小写字母命名。多个单词之间用下划线分隔。
4、类/异常(驼峰命名法):首字母大写+大写式驼峰。Python中一个模块可以包含多个类。私有类名称需要以下划线开头。如:HelloWorld 或 _HelloWorld;
5、函数:使用小写字母命名。多个单词之间用下划线分隔。私有函数名称需要以双下划线开头。
6、变量:使用小写字母命名。多个单词之间用下划线分隔。私有变量名称需要以双下划线开头。
7、常量/全局变量:使用大写字母命名。多个单词之间用下划线分隔。私有常量名称需要以双下划线开头。

8、使用单下划线“_”开头的模块变量或者函数是受保护的,在使用from xxx import * 语句从模块中导入时这些变量或函数不能被导入。


【3】Python中的特殊模块_init_.py

1、 包含此模块的文件夹才能成为包。(需要确保包的命名符合规范,才能被导入)
2、 init.py当包被导入的时候会自动运行。


【4】导包路径

1、绝对导入(from只能从根目录导入) :
from my_package import *
from my_package import my_test
import my_package.my_test as p
p.func
2、相对导入(“.”代表当前目录,“..”代表上一层目录):
from .my_package import my_test