飞翔飞翔
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
  • 学习路径
  • 第1章 认识Python

    • Python 历史与特点
    • Python 2 与 Python 3 的核心差异
    • 安装与运行 Python 2.7.18
    • 编码规范 PEP 8
  • 第2章 基础语法

    • 变量与对象
    • 数字类型
    • 字符串 str
    • Unicode 字符串
    • 运算符
    • 空值 None
  • 第3章 流程控制

    • if 条件语句
    • if-else 条件语句
    • if-elif-else 多分支
    • 条件表达式(三元运算符)
    • while 循环
    • for 循环
    • range 与 xrange
    • 循环控制:break、continue、pass
    • 循环 else 子句
  • 第4章 数据结构

    • 列表基础
    • 列表方法
    • 列表推导式
    • 元组
    • 字典基础
    • 字典方法
    • 字典循环技巧
    • 集合
    • 序列解包
    • 序列比较
  • 第5章 函数

    • 定义函数
    • 参数传递机制
    • 默认参数
    • 关键字参数
    • 可变参数
    • Lambda 表达式
    • 文档字符串
    • 函数对象
  • 第6章 模块与包

    • import 导入
    • 模块搜索路径
    • name 与主程序
    • 编译文件 .pyc 与 .pyo
    • 包结构
    • dir() 函数
  • 第7章 文件与IO

    • 打开与关闭文件
    • 文件读写方法
    • with 上下文管理器
    • 格式化输出:% 操作符
    • 格式化输出:str.format()
    • JSON 序列化
  • 第8章 面向对象

    • 类定义与实例化
    • init 构造方法
    • 类变量与实例变量
    • 方法调用与 self
    • 继承基础
    • 多重继承
    • 新式类与旧式类
    • 私有变量与名称改写
    • 属性装饰器 property
    • 类方法与静态方法
    • 魔术方法
    • 空类与数据记录
  • 第9章 异常处理

    • 异常类型
    • try-except
    • try-except-else-finally
    • 抛出异常 raise
    • 自定义异常
    • with 语句与上下文管理器
  • 第10章 迭代器与生成器

    • 迭代器协议
    • 生成器函数
    • 生成器表达式
    • itertools模块
  • 第11章 标准库精要

    • os模块
    • sys模块
    • datetime模块
    • re模块
    • json模块
    • collections模块
    • math与random模块
    • urllib2与网络请求
    • subprocess与命令执行
    • threading与并发
    • unittest与测试
    • 虚拟环境与包管理
  • 第12章 工程实践

    • 调试技巧
    • 性能分析
    • 文档与注释
    • 下一步学习

模块搜索路径

当执行 import foo 时,Python 需要找到名为 foo.py 的文件。这个查找过程遵循一套明确的规则,理解这些规则对解决 ImportError 至关重要。

sys.path

Python 维护一个名为 sys.path 的列表,按顺序存储模块搜索路径:

import sys
for p in sys.path:
    print p

# 典型输出:
# ''                          (空字符串,表示当前目录)
# '/usr/lib/python2.7'
# '/usr/lib/python2.7/plat-linux2'
# '/usr/lib/python2.7/lib-tk'
# '/usr/lib/python2.7/lib-old'
# '/usr/lib/python2.7/lib-dynload'
# '/usr/local/lib/python2.7/dist-packages'
# '/usr/lib/python2.7/dist-packages'

Python 按 sys.path 的顺序逐个目录查找,第一个匹配的模块被加载,后续目录被忽略。

搜索路径的来源

sys.path 由以下部分组成:

  1. 当前脚本所在目录(或空字符串表示当前工作目录)
  2. PYTHONPATH 环境变量中的目录
  3. 标准库目录
  4. site-packages 目录(第三方包)
# Linux/macOS:设置 PYTHONPATH
export PYTHONPATH=/home/user/mylibs:/home/user/morelibs
python myscript.py

# Windows:设置 PYTHONPATH
set PYTHONPATH=C:\Users\User\mylibs;C:\Users\User\morelibs
python myscript.py

修改搜索路径

可以在运行时修改 sys.path:

import sys
sys.path.insert(0, "/path/to/modules")     # 插入到开头,优先查找
sys.path.append("/path/to/more")            # 追加到末尾

临时修改常用于:

  • 加载不在标准位置的模块
  • 测试新版本的库
  • 插件系统动态加载

同名模块的冲突

如果多个目录中有同名模块,先找到的优先:

# 当前目录下有 random.py
import random
print random.__file__       # 可能加载了当前目录的 random.py,而非标准库!

这是 Python 2 中常见的 bug:给脚本起名为 random.py、string.py、test.py 等,会覆盖标准库模块。

解决方法:

  1. 不要用标准库模块名作为脚本名
  2. 检查 module.__file__ 确认加载的是哪个文件
import random
print random.__file__       # /usr/lib/python2.7/random.pyc

内置模块

有些模块(如 sys、__builtin__)是内置的,没有对应的 .py 文件:

import sys
print sys.__file__          # AttributeError: 'module' object has no attribute '__file__'

import math
print math.__file__         # /usr/lib/python2.7/lib-dynload/math.so(C 扩展)

相对导入与绝对导入

Python 2 默认允许相对导入:如果当前包中有 foo.py,import foo 会优先加载它而非标准库的 foo。这在大型项目中容易出错。

Python 2.7 支持 from __future__ import absolute_import 来启用绝对导入:

from __future__ import absolute_import
import string         # 总是导入标准库的 string
from . import string # 显式相对导入,当前包的 string

这是 Python 3 的默认行为,建议 Python 2.7 项目也启用。

查找模块文件

可以用 imp 模块查找模块位置:

import imp
print imp.find_module("os")         # (None, '/usr/lib/python2.7/os.py', ('.py', 'U', 1))
print imp.find_module("sys")        # (None, 'sys', ('', '', 6)) —— 内置模块
上一页
import 导入
下一页
name 与主程序