百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术教程 > 正文

一文明白Python 的import如何工作

csdh11 2025-04-01 16:56 27 浏览

Python import系统的基础知识

Python 的import系统是该语言设计的关键部分,允许模块化编程和代码的轻松重用。了解这个系统对任何 Python 程序员都很重要,因为它决定了代码的结构、共享和执行方式。

什么是模块?

在 Python 中,模块只是一个包含 Python 定义和语句的文件。文件名是添加了后缀 .py 的模块名称。在模块中,可以定义函数、类和变量,还可以包含可运行的代码。下面是一个名为 mymodule.py

# mymodule.py

def greet(name):
    return f"Hello, {name}!"

class Greeter:
    def __init__(self, name):
        self.name = name

    def greet(self):
        return f"Hello, {self.name}!"

# Module-level variable
message = "Welcome to my module"

此模块可以导入并在另一个 Python 脚本中使用:

# main.py

import mymodule

print(mymodule.greet("Kaitlyn"))
greeter = mymodule.Greeter("Alex")
print(greeter.greet())
print(mymodule.message)

包是一种使用“带点的模块名称”来构建 Python 模块命名空间的方法。包是包含名为 __init__.py 的特殊文件的目录。 __init__.py 文件(可以是空的或包含初始化代码)的存在向 Python 发出信号,表明该目录应被视为一个包。

下面是包结构的示例:

mypackage/
    __init__.py
    module1.py
    module2.py

可以按如下方式从此包导入模块:

from mypackage import module1
from mypackage.module2 import some_function

Import 语句

Python 提供了多种导入模块和包的方法。最常见的 import 语句是:

  • import module_name:这将导入整个模块,需要在其成员前面加上模块名称。
import mymodule
print(mymodule.greet("Kaitlyn"))
  • from module_name import member_name:这会将模块中的特定成员直接导入到命名空间中。
from mymodule import greet
print(greet("Kaitlyn"))
  • 导入module_name作为别名:这将导入具有别名的模块,允许您使用较短的名称。
import mymodule as mm
print(mm.greet("Kaitlyn"))
  • from module_name import *:这会将模块中的所有公共成员导入到命名空间中。通常不鼓励这种做法,因为它可能导致命名空间混乱和潜在的名称冲突。
from mymodule import *
print(greet("Kaitlyn"))

Python 如何执行导入

导入模块时,Python 将执行以下步骤:

  1. 检查模块是否已导入:如果模块已导入, sys.modules Python 使用缓存版本。
  2. 查找模块:Python 在 中 sys.path 列出的目录中搜索该模块。
  3. 编译模块:如果模块是源文件,Python 会将其编译为字节码。
  4. 执行模块:Python 执行模块的代码以创建模块的命名空间。

从不同位置导入

Python 的灵活性允许您从不同位置导入模块。除了从标准库和已安装的软件包导入外,还可以从自定义目录导入。这通常是通过修改 sys.path 或设置 PYTHONPATH 环境变量来完成的。

相对进口

在具有多个模块的大型项目中,相对导入对于保持模块化和可读性很有用。相对导入使用点表示法来引用当前包和父包。

例如,请考虑以下包结构:

mypackage/
    __init__.py
    module1.py
    subpackage/
        __init__.py
        module2.py

可以执行相对导入 module2.py 以导入 module1

# module2.py

from .. import module1

在此示例中, .. 在包层次结构中向上一层到父包 ( mypackage ),然后 module1 从中导入。

搜索路径以及 Python 查找模块的方式

在 Python 中导入模块时,解释器会遵循系统方法来查找和加载模块。此过程涉及搜索搜索路径中定义的一系列目录。了解此搜索路径对于解决导入错误和有效管理代码库非常重要。

搜索路径

搜索路径由 sys.path 列表定义,该列表包含 Python 搜索模块的目录。目录的 sys.path 顺序决定了搜索顺序。搜索路径的典型组件包括:

  1. 包含输入脚本的目录:如果正在运行脚本,Python 首先会检查脚本所在的目录。这允许您在不修改搜索路径的情况下导入本地模块。
  2. PYTHONPATH 环境变量:这是一个环境变量,您可以设置它以将其他目录添加到搜索路径。它是用冒号(在 Unix 上)或分号(在 Windows 上)分隔的目录路径列表。
  3. 标准库目录:Python 的标准库目录包含内置模块和包。默认情况下,这些目录包含在当前脚本的目录和 PYTHONPATH 之后。
  4. 第三方站点包:此目录包含通过包管理器安装的包,例如 pip 。它通常位于 Python 安装目录中,包括从 Python 包索引 (PyPI) 安装的模块和包。

查看和修改搜索路径

要查看 Python 环境中的当前搜索路径,可以打印列表 sys.path

import sys
print(sys.path)

这将输出 Python 将搜索模块的目录列表。

示例:将目录添加到搜索路径

可以在运行时修改 sys.path 列表以包含其他目录。如果您将自定义模块存储在非标准位置,这可能很有用:

import sys
sys.path.append('/path/to/your/modules')

import your_custom_module

sys.path 以这种方式添加目录是临时的,仅影响当前脚本。如果需要永久添加目录,请考虑设置 PYTHONPATH 环境变量或修改 sitecustomize or usercustomize 模块。

模块解析过程

当您尝试导入模块时,Python 会遵循特定的步骤序列来查找和加载该模块:

  1. 检查模块是否已导入:Python 首先检查 sys.modules 字典以查看模块是否已导入。如果存在,Python 将使用缓存 sys.modules 的版本,并且不会重新加载模块。
  2. 搜索内置模块:Python 检查模块是否为内置模块(例如, sysos )。内置模块是预先编译的,并包含在 Python 解释器中。
  3. 搜索 中的 sys.path 目录: Python 按顺序遍历列出的每个 sys.path 目录。它查找与模块名称匹配的文件 .py ,该文件具有 、 .pyc.pyo.pyw.pyd 扩展名(在 Windows 上)或相应的包目录。

示例:了解搜索路径顺序

请考虑以下示例,其中具有以下目录结构:

/project
    /scripts
        main.py
    /modules
        mymodule.py

如果 main.py 包含:

import sys
sys.path.append('../modules')
import mymodule

Python 将首先检查 main.py/project/scripts 的目录),然后是 (如果设置了)中的 PYTHONPATH 目录,然后是标准库目录,最后是附加到 sys.path../modules 目录。

处理导入错误

导入错误可能由于各种原因而发生,例如缺少模块、不正确的模块名称或模块代码中的问题。Python 在找不到模块或模块内出现错误时引发 。 ImportError

示例:处理导入错误

处理 ImportError 允许您提供回退机制或信息性错误消息:

try:
    import non_existent_module
except ImportError as e:
    print(f"Error importing module: {e}")
    print("Please make sure the module is installed and the name is correct.")
    print("You can try installing it using pip: pip install non_existent_module")
    # Optionally, provide fallback code here if the module is not essential.

自定义搜索路径

除了 sys.path 在脚本中进行修改外,还可以通过设置 PYTHONPATH 环境变量或使用 .pth site-packages 目录中的文件来自定义搜索路径。

示例:设置 PYTHONPATH 环境变量

要将目录永久添加到搜索路径,您可以设置 PYTHONPATH 环境变量。例如,在 Unix 系统上,您可以将以下行添加到 或 .bashrc .profile

export PYTHONPATH="/path/to/your/modules:$PYTHONPATH"

在 Windows 上,可以通过“系统属性”或使用命令提示符中 set 的命令来设置 PYTHONPATH 环境变量:

set PYTHONPATH=C:\path\to\your\modules;%PYTHONPATH%

模块缓存和重新导入模块

当 Python 导入模块时,它会执行多个操作以确保该模块可供使用。此过程的一个关键方面是模块缓存。了解 Python 如何缓存模块以及如何重新导入模块可以帮助您更有效地管理代码,尤其是在开发和调试期间。

模块缓存

导入模块时,Python 会将其存储在名为 sys.modules .此字典跟踪所有导入的模块,将模块名称映射到模块对象。此缓存机制可确保每个模块在每个会话中仅加载一次,从而通过避免冗余加载和初始化来提高性能。

示例:查看缓存的模块

可以通过检查 sys.modules 字典来查看缓存的模块:

import sys

# List all cached modules
print(sys.modules.keys())

# Check if a specific module is cached
if 'mymodule' in sys.modules:
    print('mymodule is already imported and cached.')
else:
    print('mymodule is not imported yet.')

模块缓存的工作原理

导入模块时,Python 将执行以下步骤:

  1. 检查 sys.modules :Python 首先检查该模块是否已存在于字典中 sys.modules
  2. 使用缓存模块:如果在 sys.modules 中找到该模块,Python 使用缓存版本。
  3. 加载和缓存模块:如果找不到该模块,Python 会从文件系统加载该模块,执行其代码,并将其添加到 sys.modules .

此过程可确保同一模块的任何后续导入语句都使用缓存版本,从而避免重复执行模块的代码。

重新导入模块

在开发过程中,您可能需要重新导入模块以反映模块代码中的更改,而无需重新启动 Python 解释器。这在交互式会话和长时间运行的应用程序中特别有用。

示例:使用importlib

Python 提供了该 importlib 模块,其中包括一个 reload 用于重新加载以前导入的模块的函数:

import importlib
import mymodule

# Modify the mymodule.py file externally (e.g., add new functions or update code)

# Reload the module to reflect changes
importlib.reload(mymodule)

通过使用 importlib.reload ,Python 重新执行模块的代码并更新 中的 sys.modules 模块对象。这样可以确保在不重新启动解释器的情况下提供最新的更改。

模块重新导入的用例

  1. 交互式开发:在交互式环境(如 Jupyter Notebooks 或 IPython shell)中工作时,可用于 importlib.reload 测试模块中的更改,而无需重新启动会话。
  2. 动态应用程序:在需要动态加载模块的应用程序(如插件或扩展)中,您可以重新加载模块以在运行时更新其功能。
  3. 调试:调试时,可能需要频繁更改模块的代码。重新加载模块可帮助您立即测试这些更改。

示例:在交互式会话中重新导入模块

下面是在交互式会话期间重新导入模块的分步示例:

  • 创建和导入模块
# mymodule.py
def greet(name):
    return f"Hello, {name}!"
# Interactive session
import mymodule
print(mymodule.greet("Kaitlyn"))  # Output: Hello, Kaitlyn
  • 修改模块

更新 mymodule.py 以更改 greet 功能:

# mymodule.py
def greet(name):
    return f"Hi, {name}!"
  • 重新加载模块
import importlib
importlib.reload(mymodule)
print(mymodule.greet("Kaitlyn"))  # Output: Hi, Kaitlyn!

相关推荐

NUS邵林团队发布DexSinGrasp基于强化学习实现物体分离与抓取统一

本文的作者均来自新加坡国立大学LinSLab。本文的共同第一作者为新加坡国立大学实习生许立昕和博士生刘子轩,主要研究方向为机器人学习和灵巧操纵,其余作者分别为硕士生桂哲玮、实习生郭京翔、江泽宇以及...

「PLC进阶」如何通过编写SCL语言程序实现物料分拣?

01、前言SCL作为IEC61131-3编程语言的一种,由于其高级语言的特性,特别适合复杂运算、复杂数学函数应用的场合。本文以FactoryIO软件中的物料分拣案例作为硬件基础,介绍如何通过SCL来实...

zk源码—5.请求的处理过程一(http1.1请求方法)

大纲1.服务器的请求处理链...

自己动手从0开始实现一个分布式 RPC 框架

前言为什么要自己写一个RPC框架,我觉得从个人成长上说,如果一个程序员能清楚的了解RPC框架所具备的要素,掌握RPC框架中涉及的服务注册发现、负载均衡、序列化协议、RPC通信协议、Socket通信、异...

MLSys’25 | 极低内存消耗:用SGD的内存成本实现AdamW的优化性能

AIxiv专栏是机器之心发布学术、技术内容的栏目。过去数年,机器之心AIxiv专栏接收报道了2000多篇内容,覆盖全球各大高校与企业的顶级实验室,有效促进了学术交流与传播。如果您有优秀的工作想要分享,...

线程池误用导致系统假死(线程池会自动销毁吗)

背景介绍在项目中,为了提高系统性能使用了RxJava实现异步方案,其中异步线程池是自建的。但是当QPS稍微增大之后却发现系统假死、无响应和返回,调用方出现大量超时现象。但是通过监控发现,系统线程数正常...

大型乘用车工厂布局规划(六大乘用车基地)

乘用车工厂的布局规划直接影响生产效率、物流成本、安全性和未来扩展能力。合理的布局应确保生产流程顺畅、物流高效、资源优化,并符合现代化智能制造和绿色工厂的要求。以下是详细的工厂布局规划要点:1.工厂布...

西门子 S7-200 SMART PLC 连接Factory IO的方法

有很多同学不清楚如何西门子200smart如何连接FactoryIO,本教程为您提供了如何使用西门子S7-200SMARTPLC连接FactoryIO的说明。设置PC和PLC之间的...

西门子博图高级仿真软件的应用(西门子博途软件仿真)

1.博图高级仿真软件(S7-PLCSIMAdvancedV2.0)S7-PLCSIMAdvancedV2.0包含大量仿真功能,通过创建虚拟控制器对S7-1500和ET200SP控制器进行仿真...

PLC编程必踩的6大坑——请对号入座,评论区见

一、缺乏整体规划:面条式代码问题实例:某快递分拣线项目初期未做流程图设计,工程师直接开始编写传送带控制程序。后期增加质检模块时发现I/O地址冲突,电机启停逻辑与传感器信号出现3处死循环,导致项目延期2...

统信UOS无需开发者模式安装软件包
统信UOS无需开发者模式安装软件包

原文链接:统信UOS无需开发者模式安装软件包...

2025-05-05 14:55 csdh11

100个Java工具类之76:数据指纹DigestUtils

为了提高数据安全性,保证数据的完整性和真实性,DigestUtils应运而生。正确恰当地使用DigestUtils的加密算法,可以实现数据的脱敏,防止数据泄露或篡改。...

麒麟KYLINIOS软件仓库搭建02-软件仓库添加新的软件包

#秋日生活打卡季#原文链接:...

Java常用工具类技术文档(java中工具类的作用)

一、概述Java工具类(UtilityClasses)是封装了通用功能的静态方法集合,能够简化代码、提高开发效率。本文整理Java原生及常用第三方库(如ApacheCommons、GoogleG...

软路由的用法(自动追剧配置)(软路由教学)

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:值友98958248861环境和需求...