【不背八股】1.if __name__ == "__main__" 有什么作用?

起名动机

最近开始准备秋招,技术圈惯例会把常见的面试问答题戏称为"八股文",网上流传不少这样的材料,写得晦涩难懂,以至于让好多求职者去背诵

在我看来,用文科思维学工科毫无意义,八股实际就是一些基础计算机科学知识,如何用清晰地方式去理解,实践,应用才更有价值。

因此,打算开个新的系列文章[不背八股],用一种全新的表述方式,重新理解八股面试题。

从一个报错开始

在项目实践中,遇到报错:

vbnet 复制代码
RuntimeError:
        An attempt has been made to start a new process before the
        current process has finished its bootstrapping phase.

        This probably means that you are not using fork to start your
        child processes and you have forgotten to use the proper idiom
        in the main module:

            if __name__ == '__main__':
                freeze_support()
                ...

        The "freeze_support()" line can be omitted if the program
        is not going to be frozen to produce an executable.

        To fix this issue, refer to the "Safe importing of main module"
        section in https://docs.python.org/3/library/multiprocessing.htm

DeepSeek**给出的解释是:

若主模块未保护入口点(即缺少 if name == 'main':),会导致子进程重复执行主模块代码。

将主代码移至if __name__ == '__main__':下之后,果然问题解决了。

这不由让我进一步思考:多数情况下,把这个操作作为一种惯例,没有思考过为什么要这么做,这么做为什么有效。

正好看到一道面试题:if __name__ == "__main__"有什么作用?借此把这个问题再探究一下。

__name__是什么?

首先要理解__name__的作用。

__name__ 是 Python 在加载模块(文件)时,解释器自动设置的一个全局变量,用来标识这个模块的"名称"。

举个简单的例子,创建一个hello.py

bash 复制代码
print("我是:", __name__)

运行输出结果:

markdown 复制代码
我是: __main__

再建立一个新的文件test.py,去导入这个文件

arduino 复制代码
import hello

运行输出结果:

复制代码
我是: hello

因此可以得到结论:

  • __name__在直接运行时值为"main"
  • __name__被导入时值为模块名

当运行一个 Python 文件,或者通过 import 导入一个模块,解释器隐式去做这件事:

ini 复制代码
# 如果是直接运行
import types

module = types.ModuleType("__main__")  # 创建模块对象
module.__name__ = "__main__"
exec(open("foo.py").read(), module.__dict__)  # 执行代码

# 如果是模块导入
import importlib

spec = importlib.util.find_spec("模块名称")
module = importlib.util.module_from_spec(spec)
spec.loader.exec_module(module)

因此,在一些模块中,往往会看到用if __name__ == "__main__"包裹的代码块,用来执行单元测试(cell test)。

多进程启动问题

下面回到开篇遇到的问题,为什么会出现这个error。

可以用以下代码,最小程度复现出该问题:

scss 复制代码
from multiprocessing import Process


def worker():
    print("子进程")


p = Process(target=worker)
p.start()

实际上,该问题只会在WindowsmacOS平台上发生。

对于Python 多进程(multiprocessing),Windows 和 macOS 默认采用spawn的方式进行启动。

它会新开一个空白的 Python 解释器进程,去重新导入主模块,因为主模块没放置在if __name__ == "__main__"之中,它被导入时,又会被再次执行一下,这就导致无限递归执行,出现此问题。

Linux 采用 fork 的方式,操作系统会直接复制当前进程的内存和运行状态,不会重新执行主模块,因此不会出现该问题。

项目 Linux Windows/macOS
启动方式 fork spawn
主模块会不会重新执行 ❌ 否 ✅ 是
是否必须加 if __name__ == '__main__' 推荐 ✅ 必须
报错可能性 较低 较高(会崩)
相关推荐
YGGP14 小时前
【Golang】LeetCode 70. 爬楼梯
算法·leetcode
小熳芋14 小时前
组合总和- python-回溯哦&剪枝
算法·机器学习·剪枝
HyperAI超神经14 小时前
入选NeurIPS 2025,智源/北大/北邮提出多流控制视频生成框架,基于音频解混实现精确音画同步
人工智能·ai·音视频·视频生成·neurlps 2025
草莓熊Lotso14 小时前
C++ 异常完全指南:从语法到实战,优雅处理程序错误
android·java·开发语言·c++·人工智能·经验分享·后端
yi个名字14 小时前
智能编码新时代:Vibe Coding与MCP驱动的工作流IDE革命
ide·人工智能
IT_陈寒14 小时前
Python性能优化实战:7个让代码提速300%的冷门技巧(附基准测试)
前端·人工智能·后端
熊猫钓鱼>_>14 小时前
多智能体协作:构建下一代高智能应用的技术范式
人工智能·ai·去中心化·wpf·agent·多智能体·multiagent
likeshop 好像科技14 小时前
AI知识库架构深度解析:智能体记忆与学习的智慧核心
人工智能·学习·架构
lxh011314 小时前
缺失的第一个正数
数据结构·算法
啊阿狸不会拉杆14 小时前
《数字图像处理》第 12 章 - 图像模式分类
图像处理·人工智能·算法·机器学习·计算机视觉·分类·数据挖掘