Что такое импорт из файла
Порой бывает трудно правильно реализовать import с первого раза, особенно если мы хотим добиться правильной работы на плохо совместимых между собой версиях Python 2 и Python 3. Попытаемся разобраться, что из себя представляют импорты в Python и как написать решение, которое подойдёт под обе версии языка.
Содержание
Ключевые моменты
- Выражения import производят поиск по списку путей в sys.path .
- sys.path всегда включает в себя путь скрипта, запущенного из командной строки, и не зависит от текущей рабочей директории.
- Импортирование пакета по сути равноценно импортированию __init__.py этого пакета.
Основные определения
- Модуль: любой файл *.py . Имя модуля — имя этого файла.
- Встроенный модуль: «модуль», который был написан на Си, скомпилирован и встроен в интерпретатор Python, и потому не имеет файла *.py .
- Пакет: любая папка, которая содержит файл __init__.py . Имя пакета — имя папки.
- С версии Python 3.3 любая папка (даже без __init__.py ) считается пакетом.
Пример структуры директорий
Обратите внимание, что в корневой папке test/ нет файла __init__.py .
Что делает import
При импорте модуля Python выполняет весь код в нём. При импорте пакета Python выполняет код в файле пакета __init__.py , если такой имеется. Все объекты, определённые в модуле или __init__.py , становятся доступны импортирующему.
Встроенные функции Python: какие нужно знать и на какие не стоит тратить времяОсновы import и sys.path
Вот как оператор import производит поиск нужного модуля или пакета согласно документации Python:
- директории, содержащей исходный скрипт (или текущей директории, если файл не указан);
- директории по умолчанию, которая зависит от дистрибутива Python;
- PYTHONPATH (список имён директорий; имеет синтаксис, аналогичный переменной окружения PATH ).
Технически документация не совсем полна. Интерпретатор будет искать не только файл (модуль) spam.py , но и папку (пакет) spam .
Обратите внимание, что Python сначала производит поиск среди встроенных модулей — тех, которые встроены непосредственно в интерпретатор. Список встроенных модулей зависит от дистрибутива Python, а найти этот список можно в sys.builtin_module_names (Python 2 и Python 3). Обычно в дистрибутивах есть модули sys (всегда включён в дистрибутив), math , itertools , time и прочие.
В отличие от встроенных модулей, которые при поиске проверяются первыми, остальные (не встроенные) модули стандартной библиотеки проверяются после директории запущенного скрипта. Это приводит к сбивающему с толку поведению: возможно «заменить» некоторые, но не все модули стандартной библиотеки. Допустим, модуль math является встроенным модулем, а random — нет. Таким образом, import math в start.py импортирует модуль из стандартной библиотеки, а не наш файл math.py из той же директории. В то же время, import random в start.py импортирует наш файл random.py .
Sportmaster Lab , Санкт-Петербург, Москва, Краснодар, можно удалённо , От 100 000 до 350 000 ₽
Кроме того, импорты в Python регистрозависимы: import Spam и import spam — разные вещи.
Функцию pkgutil.iter_modules() (Python 2 и Python 3) можно использовать, чтобы получить список всех модулей, которые можно импортировать из заданного пути:
Чуть подробнее о sys.path
Чтобы увидеть содержимое sys.path , запустите этот код:
Документация Python описывает sys.path так:
Список строк, указывающих пути для поиска модулей. Инициализируется из переменной окружения PYTHONPATH и директории по умолчанию, которая зависит от дистрибутива Python.
При запуске программы после инициализации первым элементом этого списка, path[0] , будет директория, содержащая скрипт, который был использован для вызова интерпретатора Python. Если директория скрипта недоступна (например, если интерпретатор был вызван в интерактивном режиме или скрипт считывается из стандартного ввода), то path[0] является пустой строкой. Из-за этого Python сначала ищет модули в текущей директории. Обратите внимание, что директория скрипта вставляется перед путями, взятыми из PYTHONPATH .
Источник: Python 2 и Python 3
Документация к интерфейсу командной строки Python добавляет информацию о запуске скриптов из командной строки. В частности, при запуске python <script>.py .
Если имя скрипта ссылается непосредственно на Python-файл, то директория, содержащая этот файл, добавляется в начало sys.path , а файл выполняется как модуль main .
Источник: Python 2 и Python 3
Итак, повторим порядок, согласно которому Python ищет импортируемые модули:
- Модули стандартной библиотеки (например, math , os ).
- Модули или пакеты, указанные в sys.path :
- Если интерпретатор Python запущен в интерактивном режиме:
- sys.path[0] — пустая строка '' . Это значит, что Python будет искать в текущей рабочей директории, из которой вы запустили интерпретатор. В Unix-системах эту директорию можно узнать с помощью команды pwd .
Если мы запускаем скрипт командой python <script>.py :
Обратите внимание, что при запуске скрипта для sys.path важна не директория, в которой вы находитесь, а путь к самому скрипту. Например, если в командной строке мы находимся в test/folder и запускаем команду python ./packA/subA/subA1.py , то sys.path будет включать в себя test/packA/subA/ , но не test/ .
Кроме того, sys.path общий для всех импортируемых модулей. Допустим, мы вызвали python start.py . Пусть start.py импортирует packA.a1 , а a1.py выводит на экран sys.path . В таком случае sys.path будет включать test/ (путь к start.py ), но не test/packA (путь к a1.py ). Это значит, что a1.py может вызвать import other , так как other.py находится в test/ .
Всё о __init__.py
У файла __init__.py есть две функции:
- Превратить папку со скриптами в импортируемый пакет модулей (до Python 3.3).
- Выполнить код инициализации пакета.
Превращение папки со скриптами в импортируемый пакет модулей
Чтобы импортировать модуль (или пакет) из директории, которая находится не в директории нашего скрипта (или не в директории, из которой мы запускаем интерактивный интерпретатор), этот модуль должен быть в пакете.
Как было сказано ранее, любая директория, содержащая файл __init__.py , является пакетом. Например, при работе с Python 2.7 start.py может импортировать пакет packA , но не packB , так как в директории test/packB/ нет файла __init__.py .
Это не относится к Python 3.3 и выше благодаря появлению неявных пакетов пространств имён. Проще говоря, в Python 3.3+ все папки считаются пакетами, поэтому пустые файлы __init__.py больше не нужны.
Допустим, packB — пакет пространства имён, так как в нём нет __init__.py . Если запустить интерактивную оболочку Python 3.6 в директории test/ , то мы увидим следующее:
Выполнение кода инициализации пакета
В момент, когда пакет или один из его модулей импортируется в первый раз, Python выполняет __init__.py в корне пакета, если такой файл существует. Все объекты и функции, определённые в __init__.py , считаются частью пространства имён пакета.
Рассмотрим следующий пример:
Вывод после запуска python start.py :
Примечание Если a1.py вызовет import a2 , и мы запустим python a1.py , то test/packA/__init__.py не будет вызван, несмотря на то, что a2 вроде бы является частью пакета packA . Это связано с тем, что когда Python выполняет скрипт (в данном случае a1.py ), содержащая его папка не считается пакетом.
Использование объектов из импортированного модуля или пакета
Есть 4 разных вида импортов:
- import <пакет>
- import <модуль>
- from <пакет> import <модуль или подпакет или объект>
- from <модуль> import <объект>
Пусть X — имя того, что идёт после import :
- Если X — имя модуля или пакета, то для того, чтобы использовать объекты, определённые в X , придётся писать X.объект .
- Если X — имя переменной, то её можно использовать напрямую.
- Если X — имя функции, то её можно вызвать с помощью X() .
Опционально после любого выражения import X можно добавить as Y . Это переименует X в Y в пределах скрипта. Учтите, что имя X с этого момента становится недействительным. Частым примером такой конструкции является import numpy as np .
Аргументом для import может быть как одно имя, так и их список. Каждое из имён можно переименовать с помощью as . Например, следующее выражение будет действительно в start.py : import packA as pA, packA.a1, packA.subA.sa1 as sa1 .
Пример: нужно в start.py импортировать функцию helloWorld() из sa1.py .
- Решение 1: from packA.subA.sa1 import helloWorld . Мы можем вызвать функцию напрямую по имени: x = helloWorld() .
- Решение 2: from packA.subA import sa1 или то же самое import packA.subA.sa1 as sa1 . Для использования функции нам нужно добавить перед её именем имя модуля: x = sa1.helloWorld() . Иногда такой подход предпочтительнее первого, так как становится ясно, из какого модуля взялась та или иная функция.
- Решение 3: import packA.subA.sa1 . Для использования функции перед её именем нужно добавить полный путь: x = packA.subA.sa1.helloWorld() .
Прим. перев. После переименования с помощью as новое имя нельзя использовать в качестве имени пакета или модуля для последующих импортов. Иными словами, команда вроде следующей недействительна: import packA as pA, pA.a1 .
Используем dir() для исследования содержимого импортированного модуля
После импортирования модуля можно использовать функцию dir() для получения списка доступных в модуле имён. Допустим, мы импортируем sa1 . Если в sa1.py есть функция helloWorld() , то dir(sa1) будет включать helloWorld :
Импортирование пакетов
Импортирование пакета по сути равноценно импортированию его __init__.py . Вот как Python на самом деле видит пакет:
После импорта становятся доступны только те объекты, что определены в __init__.py пакета. Поскольку в packB нет такого файла, от import packB (в Python 3.3.+) будет мало толку, так как никакие объекты из этого пакета не становятся доступны. Последующий вызов модуля packB.b1 приведёт к ошибке, так как он ещё не был импортирован.
Абсолютный и относительный импорт
При абсолютном импорте используется полный путь (от начала корневой папки проекта) к желаемому модулю.
При относительном импорте используется относительный путь (начиная с пути текущего модуля) к желаемому модулю. Есть два типа относительных импортов:
- При явном импорте используется формат from .<модуль/пакет> import X , где символы точки . показывают, на сколько директорий «вверх» нужно подняться. Одна точка . показывает текущую директорию, две точки .. — на одну директорию выше и т. д.
- Неявный относительный импорт пишется так, как если бы текущая директория была частью sys.path . Такой тип импортов поддерживается только в Python 2.
В документации Python об относительных импортах в Python 3 написано следующее:
Единственный приемлемый синтаксис для относительных импортов — from .[модуль] import [имя] . Все импорты, которые начинаются не с точки . , считаются абсолютными.
Источник: What’s New in Python 3.0
В качестве примера допустим, что мы запускаем start.py , который импортирует a1 , который импортирует other , a2 и sa1 . Тогда импорты в a1.py будут выглядеть следующим образом:
Явные относительные импорты:
Неявные относительные импорты (не поддерживаются в Python 3):
Учтите, что в относительных импортах с помощью точек . можно дойти только до директории, содержащей запущенный из командной строки скрипт (не включительно). Таким образом, from .. import other не сработает в a1.py . В результате мы получим ошибку ValueError: attempted relative import beyond top-level package .
Как правило, абсолютные импорты предпочтительнее относительных. Они позволяют избежать путаницы между явными и неявными импортами. Кроме того, любой скрипт с явными относительными импортами нельзя запустить напрямую:
Имейте в виду, что относительные импорты основаны на имени текущего модуля. Так как имя главного модуля всегда "__main__" , модули, которые должны использоваться как главный модуль приложения, должны всегда использовать абсолютные импорты.
Источник: Python 2 и Python 3
Примеры
Пример 1: sys.path известен заранее
Если вы собираетесь вызывать только python start.py или python other.py , то прописать импорты всем модулям не составит труда. В данном случае sys.path всегда будет включать папку test/ . Таким образом, все импорты можно писать относительно этой папки.
Пример: файлу в проекте test нужно импортировать функцию helloWorld() из sa1.py .
Решение: from packA.subA.sa1 import helloWorld (или любой другой эквивалентный синтаксис импорта).
Пример 2: sys.path мог измениться
Зачастую нам требуется как запускать скрипт напрямую из командной строки, так и импортировать его как модуль в другом скрипте. Как вы увидите далее, здесь могут возникнуть проблемы, особенно в Python 3.
Пример: пусть start.py нужно импортировать a2 , которому нужно импортировать sa2 . Предположим, что start.py всегда запускается напрямую, а не импортируется. Также мы хотим иметь возможность запускать a2 напрямую.
Звучит просто, не так ли? Нам всего лишь нужно выполнить два импорта: один в start.py и другой в a2.py .
Проблема: это один из тех случаев, когда sys.path меняется. Когда мы выполняем start.py , sys.path содержит test/ , а при выполнении a2.py sys.path содержит test/packA/ .
С импортом в start.py нет никаких проблем. Так как этот модуль всегда запускается напрямую, мы знаем, что при его выполнении в sys.path всегда будет test/ . Тогда импортировать a2 можно просто с помощью import packA.a2 .
С импортом в a2.py немного сложнее. Когда мы запускаем start.py напрямую, sys.path содержит test/ , поэтому в a2.py импорт будет выглядеть как from packA.subA import sa2 . Однако если запустить a2.py напрямую, то в sys.path уже будет test/packA/ . Теперь импорт вызовет ошибку, так как packA не является папкой внутри test/packA/ .
Вместо этого мы могли бы попробовать from subA import sa2 . Это решает проблему при запуске a2.py напрямую, однако теперь создаёт проблему при запуске start.py . В Python 3 это приведёт к ошибке, потому что subA не находится в sys.path (в Python 2 это не вызовет проблемы из-за поддержки неявных относительных импортов).
Запускаем from packA.subA import sa2 from subA import sa2 start.py Нет проблем В Py2 нет проблем, в Py3 ошибка ( subA не в test/ ) a2.py Ошибка ( packA не в test/packA/ ) Нет проблем Использование относительного импорта from .subA import sa2 будет иметь тот же эффект, что и from packA.subA import sa2 .
Вряд ли для этой проблемы есть чистое решение, поэтому вот несколько обходных путей:
1. Использовать абсолютные импорты относительно директории test/ (т. е. средняя колонка в таблице выше). Это гарантирует, что запуск start.py напрямую всегда сработает. Чтобы запустить a2.py напрямую, запустите его как импортируемый модуль, а не как скрипт:
- В консоли смените директорию на test/ .
- Запустите python -m packA.a2 .
2. Использовать абсолютные импорты относительно директории test/ (средняя колонка в таблице). Это гарантирует, что запуск start.py напрямую всегда сработает. Чтобы запустить a2.py напрямую, можно изменить sys.path в a2.py , чтобы включить test/packA/ перед импортом sa2 .
Примечание Обычно этот метод работает, однако в некоторых случаях переменная __file__ может быть неправильной. В таком случае нужно использовать встроенный пакет inspect . Подробнее в этом ответе на StackOverflow.
3. Использовать только Python 2 и неявные относительные импорты (последняя колонка в таблице).
4. Использовать абсолютные импорты относительно директории test/ и добавить её в переменную среды PYTHONPATH . Это решение не переносимо, поэтому лучше не использовать его. О том, как добавить директорию в PYTHONPATH , читайте в этом ответе.
Пример 3: sys.path мог измениться (вариант 2)
А вот ещё одна проблема посложнее. Допустим, модуль a2.py никогда не надо запускать напрямую, но он импортируется start.py и a1.py , которые запускаются напрямую.
В этом случае первое решение из примера выше не сработает. Тем не менее, всё ещё можно использовать остальные решения.
Пример 4: импорт из родительской директории
Если мы не изменяем PYTHONPATH и стараемся не изменять sys.path программно, то сталкиваемся со следующим основным ограничением импортов в Python: при запуске скрипта напрямую невозможно импортировать что-либо из его родительской директории.
Например, если бы нам пришлось запустить python sa1.py , то этот модуль не смог бы ничего импортировать из a1.py без вмешательства в PYTHONPATH или sys.path .
На первый взгляд может показаться, что относительные импорты (например from .. import a1 ) помогут решить эту проблему. Однако запускаемый скрипт (в данном случае sa1.py ) считается «модулем верхнего уровня». Попытка импортировать что-либо из директории над этим скриптом приведёт к ошибке ValueError: attempted relative import beyond top-level package .
Для решения этой проблемы лучше её не создавать и избегать написания скриптов, которые импортируют из родительской директории. Если этого нельзя избежать, то предпочтительным обходным путём является изменение sys.path .
Python 2 vs Python 3
Мы разобрали основные отличия импортов в Python 2 и Python 3. Они ещё раз изложены здесь наряду с менее важными отличиями:
Люди часто сталкиваются с необходимостью переноса контактных данных между сотовыми. Чтобы не добавлять каждую запись заново и по отдельности предусмотрены функции экспорта и импорта. Иногда потребители путают эти понятия. Статья поможет разобраться, чем различаются два термина, и покажет, как совершаются такие процессы в современных смартфонах.
Что значит импорт и экспорт контактов в телефоне
Говоря простым языком, импорт — это добавление, а экспорт — изъятие.
Что такое импорт и зачем он нужен
Импорт — добавление контактных данных в приложение на девайсе с внутреннего хранилища, SIM- или MicroSD-карты.
Нужен он при смене пользователем мобильного устройства и необходимости перенести значения с SIM-карты. Этот процесс находит применение при восстановлении данных с резервной копии.
Экспорт и его функции
Экспортом называется перенос контактных данных на мобильном девайсе во внутренний или внешний накопитель либо на SIM-карту.
Основными его функциями являются перемещение текстовой информации между сотовыми либо на SIM-карту и создание Backup’а (резервной копии). Процесс бывает полезен при освобождении места на накопителе. Но это актуально лишь для старых сотовых, где максимальное количество имен в телефонной книге небольшое.
В чем ключевые отличия импорта от экспорта
Импорт и экспорт контактов в телефоне отличаются тем, что в первом происходит добавление информации в список номеров, а во втором сведения, наоборот, изымаются оттуда. Разница в том, что экспортировать данные обычно означает записать текст в память девайса, а импортировать контакты — это значит считать сведения с файла, находящегося на устройстве.
Как экспортировать и импортировать контакты
Экспортировать или импортировать контакты на Android-устройствах можно несколькими способами. Первый — через стандартное приложение « Контакты » на телефоне, второй — используя сторонний софт.
Используя Google-аккаунт
Перенести номера на аккаунт Google можно таким образом:
- Коснуться « Экспортировать в VCF-файл » или « Сохранить ».
Внимание! Добавить данные на девайсах с ОС Android можно с SIM-карты или виртуальной карточки, находящейся в памяти аппарата.
Для извлечения с SIM понадобится произвести следующие действия:
- Вставить СИМ-карту в смартфон или планшет.
- Открыть программу, содержащую номера на Android-устройстве.
- Нажать на пункт « SIM-карта » (в некоторых версиях ОС предлагает выбор оператора). Указать аккаунт для сохранения материала.
Если сведения находятся в специальном документе формата . vcf , импорт производится так:
- Произвести нажатие на виртуальную кнопку « Меню » в левом верхнем углу экрана.
- Выбрать « VCF-файл ». Из предложенных аккаунтов указать тот, где требуется сохранить контактные данные.
- Найти и открыть документ . vcf , откуда будут извлечены значения.
Через сторонние приложения
Перенос информации с мобильных устройств можно осуществить с помощью компьютерной программы MOBILedit .
Экспортировать контакты с помощью MOBILedit можно следующим образом:
- В самой программе перейти на вкладку « Телефон/Phones », далее указать « Соединение/Connection Wizars » (в последней версии утилиты кнопка доступна на стартовой вкладке).
- В появившемся окне выбрать PC Sync или Cable (в зависимости от версии программы).
- Перейти в настройки мобильного телефона и кликнуть на пункт « Система » или « Расширенные настройки ».
- Если этой надписи нет, открыть пункт « О телефоне » и тапнуть несколько раз на « Номер сборки » или « Версия MIUI » до появления соответствующего уведомления.
- После этого в настройках для разработчиков включить « Отладку по USB ».
- Теперь с ПК кликнуть на пункт « Phonebook », находящийся на панели MOBILedit’а.
- Выбрать, какие именно нужно экспортировать контакты.
- Выбрав контакты, нажать кнопку « Экспорт/Export ».
- Указать тип файла CSV и путь сохранения его на компьютере.
Импортировать контакты в данной утилите ещё проще. Требуется совершить следующие действия:
- Подключить смартфон к компьютеру.
- Открыть пункт « Импорт/Import » в программе MOBILedit .
- Выбрать файл, полученный при экспорте, и кликнуть на « Import ».
Достоинствами программы MOBILedit являются поддержка большого количества моделей телефонов (не только смартфонов с ОС Android) и возможность экспорта в CSV-файл, который беспрепятственно открывается в Microsoft Excel. Но у приложения есть и недостатки: отсутствие русского языка и платная лицензия.
Импорт данных означает перенос информации, которую вы сохранили с другим источником, в свое финансовое программное обеспечение. Это могут быть учетные записи с безопасного веб-сайта банка или данных, которые вы ввели в другую финансовую программу, и теперь вы хотите интегрировать их. Гораздо удобнее располагать всю вашу информацию в одном приложении, а не распространяться через другие приложения.
Импорт данных зависит от конкретного программного обеспечения, которое вы используете, но различия между приложениями обычно незначительны.
Примеры импорта данных
Типичный импорт данных будет поступать из транзакций учетной записи, которые вы загрузили из своего банка или другого финансового учреждения. Ваше финансовое программное обеспечение обычно может конвертировать данные, поэтому в учетном журнале появляются новые транзакции.
Другими примерами импорта данных будут импорт всей учетной записи или файла данных, который переносится из одного пакета программного обеспечения в другой, чтобы сэкономить время при вводе транзакций.
Данные учетной записи и файлы данных также могут быть экспортированы и импортированы между одним и тем же названием программного обеспечения и версией на двух разных компьютерах.
Вы должны экспортировать, прежде чем импортировать
Вы должны экспортировать и сохранить данные из онлайн или другого источника, прежде чем сможете импортировать его. Например, вы можете импортировать старые данные MS Money в выпуск Microsoft Money Plus Sunset, чтобы сравнить старый бюджет с новым, который вы создали.
Вот как вы это сделаете:
Следующий шаг
Теперь вам нужно импортировать данные в свое программное обеспечение.
У вашего браузера, вероятно, есть место по умолчанию, где он сохраняет файлы, которые вы загрузили.Обычно это папка, логически называемая «Загрузки», но проверьте параметры браузера, если вы не видите такой файл или файл экспорта, похоже, не существует. Вам нужно будет найти данные, прежде чем вы сможете импортировать их.
Теперь откройте программное обеспечение, для которого вы хотите импортировать данные. На главной странице должна быть опция «Выбрать файл». Это тот термин, который использует программное обеспечение Quicken, но может быть что-то другое в зависимости от вашего финансового программного приложения. Например, если вы импортировали в Microsoft Office, вкладка просто обозначается как «Открыть». В любом случае вы должны быть вознаграждены всплывающим окном, показывающим все ваши сохраненные файлы, когда вы нажимаете на него.
Перейдите в папку, в которой находится импортированный файл, и откройте или импортируйте его.
Файл экспорта должен быть удален после того, как загруженный файл будет импортирован в ваше финансовое программное обеспечение.
Упражнение в корзине: что это такое и как его использовать
Упражнение в корзине - это упражнение, данное кандидатам на работу, чтобы определить, насколько хорошо они выполняют задания, связанные с работой, в течение установленного промежутка времени.
Вращающийся лазерный уровень: как его использовать и его применения
Самовыравнивающийся роторный лазер: Ротационный лазерный уровень, преимущества и применение. Узнайте, сколько доступных типов вращающихся лазеров доступно и как вы можете извлечь максимальную пользу из этого.
Обзор социальных сетей: что это такое и как его использовать
Что такое социальных медиа и как вы можете его использовать? Обзор типов сайтов социальных сетей, включая Facebook, Twitter, Instagram, YouTube, Pinterest.
Общее / Ключевые моменты
- инструкция import осуществляет поиск модулей в списке путей, содержащемся в переменной sys.path ;
- sys.path всегда включает путь к скрипту, запущенному в настоящий момент времени в командной строке и однозначно задает рабочую директорию для текущего сеанса командной строки;
- концептуально импорт пакета происходит, как импорт файла пакета __init__.py .
Основные определения
- модуль: любой файл с расширением *.py . Именем модуля является имя файла.
- встроенный модуль: модуль (написанный на С ), который интегрирован непосредственно в интерпретатор Python и следовательно не имеет отдельного файла с расширением *.py .
- пакет: любая папка содержащая файл с именем __init\__.py . Имя пакета совпадает с именем папки. В Python 3.3 и выше любая папка (даже без файла __init\__.py ) считается пакетом.
- объект: в Python почти всё является объектом: функции, классы, переменные и т.д.
Пример структуры директории
Обратите внимание, что мы не размещаем __init\__.py в корневой папке test/ .
Что такое import ?
Когда вы импортируете модуль, Python запускает код из файла модуля. Когда вы импортируете пакет, Python сначала запускает код из файла пакета __init__.py , если такой файл существует. После этого все объекты, определенные в файле модуля или __init__.py пакета, становятся доступны.
Основы использования оператора import и переменной sys.path
В соответствии с документацией Python, оператор import ищет корректный модуль или пакет для импорта в соответствии со следующими правилами.
- директория, содержащая запущенный на исполнение скрипт (или текущий каталог, когда не указан файл).
- переменная окружения PYTHONPATH (список имен каталогов с тем же синтаксисом, что и PATH переменная оболочки shell ).
- значение по умолчанию, заданное при установке Python.
Технически документация Python является неполной. На самом деле интерпретатор ищет не только файл (то есть, модуль) с именем spam.py , также он будет искать папку (то есть, пакет) с именем spam .
Обратите внимание, что интерпретатор Python сначала ищет список встроенных модулей, то есть модулей, которые интегрированы непосредственно в интерпретатор Python. Состав списка встроенных модулей зависит от типа установки и может быть найден в файлах sys.builtin_module_names (для Python версии 2 и 3). Обычно встроенные модули, которые входят с состав начальной установки, включают в себя sys (всегда устанавливается), math , itertools и time и другие.В отличие от встроенных модулей (модулей стандартной библиотеки), пути к которым помещаются первыми в списке для поиска при импорте остальные модули в стандартной библиотеке Python появляются после пути каталога текущего скрипта. Это приводит к запутанному поведению: становится возможно «заменить» некоторые, но не все модули в стандартной библиотеке Python.
Например, на моем компьютере (Windows 10, Python 3.6) модуль math является встроенным модулем, тогда как модуль random нет. Таким образом, import math в start.py будет импортировать модуль math из стандартной библиотеки, а не мой собственный файл math.py , находящийся в тот же каталоге.
А оператор import random в start.py будет импортировать мой файл random.py , а не модуль random из стандартной библиотеки.Функция pkgutil.iter_modules (Python 2 и 3) может использоваться для получения списка всех импортируемых модулей по заданному пути:
Подробнее о sys.path
Чтобы узнать, что находится в переменной sys.path , запустите в командной строке следующие команды или скрипт содержащий инструкции:
Документация Python для sys.path описывает это следующим образом.
Документация для интерфейса командной строки Python добавляет следующее о запуске сценариев из командной строки. В частности, при запуске команды python script.py , происходит следующее.
Если имя сценария ссылается непосредственно на файл с расширением py , то каталог, содержащий этот файл, добавляется в начало sys.path и файл выполняется в качестве основного модуля.
Источник: Python 2 и 3Давайте рассмотрим порядок, в соответствии с которым интерпретатор Python ищет модули для импорта:
Обратите внимание, что при запуске скрипта Python в sys.path не учитывается, каков ваш текущий «рабочий каталог». Учитывается только о путь к сценарию. Например, если интерпретатор запущен из папки test/ и вы запускаете команду python ./packA/subA/subA1.py , то sys.path включает test/packA/subA/ , а не test/ .
Кроме того, значение sys.path будет использоваться для всех импортированных модулей. Например, предположим, что мы вводим команду python start.py . Пусть start.py импортирует packA.a1 , и пусть a1.py выведет содержимое переменной sys.path . Выводимое содержимое переменной sys.path будет включать test/ (путь к start.py ), а не test/packA/ (путь к a1.py ). Это означает, что скрипт a1.py может вызывать import other , поскольку файл other.py находится в test/ .
Все о __init__.py
Файл __init__.py выполняет 2 функции.
- Преобразование папки со скриптами в импортируемый пакет модулей (до Python 3.3).
- Запуск кода инициализации пакета.
Преобразование папки сценариев в импортируемый пакет модулей
Как уже было сказано выше, любой каталог, содержащий файл с именем __init__.py представляет собой пакет Python. Этот файл так же может быть пустым. Например, при запуске скрипта start.py в Python 2.7 можно импортировать пакет packA , но не packB , так как в каталоге test/packB/ нет файла __init__.py .
Это НЕ применимо к Python 3.3 и выше, благодаря принятию неявных пространств имен пакетов. В принципе, Python 3.3+ рассматривает все папки как пакеты, поэтому пустые файлы __init__.py больше не нужны и могут быть опущены.
Например, packB представляет собой пространство имен пакета, поскольку в папке нет файла __init__.py . Если мы запустим интерактивный интерпретатор Python версии 3.6 в каталоге test/ , то получим следующий результат:
Запуск кода инициализации пакета
В первый раз, когда вы импортируете пакет или один из его модулей, Python будет выполнять файл __init__.py в корневой папке пакета, если этот файл существует. Все объекты и функции, определенные в __init__.py , считаются частью пространства имен пакета.
Рассмотрим следующий пример.
Листинг файла test/packA/a1.py :
Листинг файла test/packA/__init__.py :
Листинг файла test/start.py :
Команда python start.py выведет следующее:
Примечание: если в файле a1.py вызывается import a2 , и вы запускаете в командной строке команду python a1.py , то test/packA/__ init__.py НЕ будет вызван, хотя на первый взгляд кажется, что a2 является частью пакета packA . Так происходит потому, что Python запускает скрипт (в нашем случае a1.py ), но содержащая его папка не является пакетом.
Использование объектов из импортированного модуля или пакета
Существует четыре различных вида синтаксиса для записи операторов импорта.
- import <package>
- import <module>
- from <package> import <module or subpackage or object>
- from <module> import <object>
Пусть X любое имя после ключевого слова import .
Необязательно, но после любого оператора import X может быть добавлена инструкция вида as Y , например, import X as Y . Эта инструкция переименовывает X в Y в пределах файла скрипта. Обратите внимание, что имя X далее не действительно и его использовать не имеет смысла. Например import numpy as np .
Аргументом при ключевом слове import может быть одно имя или список из нескольких имен. Каждое из этих имен может быть также переименовано с помощью ключевого слова as . Например, следующие инструкции импорта в файле в start.py будут работать правильно: import packA as pA , packA.a1, packA.subA.sa1 as sa1 .
Рассмотрим следующий пример: в файле start.py необходимо импортировать функцию helloWorld() из файла sa1.py .
Решение №1:
затем мы можем вызывать функцию непосредственно по имени:
Решение №2:
затем мы должны будем использовать в качестве префикса к имени функции имя модуля.
Иногда это решение предпочтительнее Решения №1 для того чтобы сделать явным вызов функции helloWorld из модуля sa1 .
Решение №3:
Далее необходимо использовать полный путь:
Использование dir() для проверки содержимого импортированного модуля
После импорта модуля, используйте функцию dir() для того чтобы получить список доступных имен модуля. Например, предположим, что вы импортируете sa1 . Если в sa1.py определена функция helloWorld() , то инструкция dir(sa1) в числе прочих выведет имя helloWorld .
Импорт пакетов
Импорт пакета концептуально эквивалентен импорту файла __init__.py из папки пакета в качестве модуля. И действительно это так. Вот как Python рассматривает импортируемый пакет:
Скриптам, импортирующим пакеты, доступны лишь те объекты, которые были объявлены в файле пакета __init__.py . Например, так как каталог packB не содержит файла __init__.py , то вызов инструкции import packB в Python 3.3+ будет мало полезен, поскольку никакие объекты при импорте пакета packB не будут доступны. Последующий вызов packB.b1 завершится неудачей, поскольку объект не будет импортирован.
Импорт с использованием абсолютных и относительных путей
Импорт по абсолютному пути использует полный путь (начинающийся с корневой папки проекта) до требуемого для импорта модуля .
Импорт по относительному пути использует относительный путь (начинающийся с пути к текущему модулю) до требуемого для импорта модуля. Существует два вида импорта с использованием относительного пути:
- явный импорт задается в следующей форме from .<module/package> import X , где <module/package> имеет префикс в виде последовательности имен каталогов, разделенных точками . , которые указывают, на сколько каталогов необходимо переместиться вверх или вниз. При этом одна точка . соответствует текущему каталогу, а две точки .. предписывают переместиться на одну папку вверх и т.д.
- неявный импорт записывается так, как будто текущий каталог является частью содержимого переменной sys.path . Неявный относительный импорт поддерживается только Python 2 и не поддерживается Python 3.
В документации Python говорится о том, как Python 3+ обрабатывает импорт по относительному пути:
Единственным верным синтаксисом для импорта по относительному пути является формат from .[module] import name . Все формы инструкции импорта не начинающиеся с точки . интерпретируются как импорт по абсолютному пути.
Источник: What’s New in Python 3.0.Например, предположим, что мы запускаем на исполнение скрипт start.py , который импортирует a1 , который, в свою очередь, импортирует other , a2 и sa1 . Инструкции import в файле a1.py будут выглядеть следующим образом:
Импорт по относительному пути может выполняться только для модулей в пакете; не допускается использовать эту возможность для ссылки на модули, которые находятся в другом каталоге файловой системы. Обратите внимание, что при импорте по относительному пути, точки .. помогут перемещаться при поиске файла для импорта только выше каталога (но не включая его), содержащего скрипт, запускаемый из командной строки. Таким образом, from .. import other не будет работать в a1.py . Это приводит к ошибке ValueError: attempted relative import beyond top-level package (попытка импорта по относительному пути за пределы пакета верхнего уровня).
В целом импорт по абсолютному пути предпочтительнее чем по относительному. Их использование позволяет избежать путаницы между явным или неявным импортом. Кроме того, любой сценарий, который использует явный импорт по относительному пути, не может быть запущен напрямую:
Обратите внимание, что импорт по относительному пути основан на имени текущего модуля. Поскольку имя основного модуля всегда «main», то модули, предназначенные для использования в качестве основного модуля приложения Python, должны всегда использовать импорт по абсолютному пути.
Источники: Python 2 и 3.Примеры
Пример 1: содержание sys.path заранее известно
Если вы запускаете скрипты командой python start.py или python other.py , то вам будет очень просто настроить импорт всех модулей. В этом случае переменная sys.path будет всегда включать директорию test/ в списке путей поиска при импорте. Поэтому все операторы импорта могут быть записаны относительно папки test/ .
Например: сценарий в папке test должен импортировать функцию helloWorld() из файла sa1.py .
Решение: from packA.subA.sa1 import helloWorldПример 2: содержание sys.path может изменяться
Часто мы хотим гибко использовать сценарии Python независимо от того выполняются ли они в командной строке или импортированы как модуль в другой скрипт. Как будет показано ниже, здесь мы столкнемся с проблемами, а в особенности в коде написанном на Python 3.
Пример: Предположим, что в сценарии start.py необходимо импортировать a2.py , который в свою очередь должен импортировать sa2.py . Предположим, что start.py всегда запускается непосредственно напрямую, и никогда не импортируется. Мы также хотим иметь возможность запускать a2.py самостоятельно. Кажется, все достаточно просто. В конце концов, нам нужно всего лишь два оператора импорта: первый в start.py и второй в a2.py .
Проблема: Это случай, когда sys.path будет изменяться. Когда мы запускаем start.py , переменная sys.path содержит директорию test/ . Когда мы запустим файл a2.py , то sys.path будет содержать test/ packA/ .
Инструкция import в start.py проста для понимания. Зная заранее, что start.py будет всегда запускаться напрямую и никогда не будет импортирован, мы полагаем, что при его запуске путь test/ всегда будет находиться в переменной sys.path . Тогда инструкция для импорта a2.py будет следующая import packA.a2 .
Инструкция import в a2.py более сложна. Когда мы запускаем start.py напрямую, переменная sys.path содержит test/ , поэтому в a2.py следует использовать инструкцию from packA.subA import sa2 . Однако, если после этого мы будем запускать a2.py напрямую, то переменная sys.path будет содержать путь test/packA/ . Теперь инструкция для импорта не будет корректной, поскольку packA не является папкой внутри test/packA/ .
Вместо этого мы могли бы попробовать применить инструкцию from subA import sa2 . Это решит нашу проблему, если мы запускаем a2.py напрямую. Но теперь у нас возникает проблема, когда мы напрямую запускаем start.py . В Python 3 это не будет выполняться, потому что subA не находится в переменной sys.path . Но будет выполняться без ошибок в Python 2, благодаря поддержке неявного импорта с относительными путями.
Таким образом, по использованию инструкции import в файле a2.py имеем:
Инструкция from packA.subA import sa2 from subA import sa2 start.py Работает в Py2 работает , в Py3 не работает (subA нет в test/) a2.py Не работает (packA нет в test/packA/) Работает Для полноты картины можно попытался использовать импорт по относительному пути: from .subA import sa2 . И это будет соответствовать результату выполнения инструкции from packA.subA import sa2 .
Решение (обходной путь решения проблемы): Мне ничего не известно о простом и наглядном способе решения этой проблемы. Вот некоторые обходные пути:
- Используйте импорт по абсолютному пути и test/ как корневую директорию (средний столбец в таблице выше). Это гарантирует запуск скрипта start.py напрямую. Чтобы запустить a2.py напрямую, необходимо запустить его как импортированный модуль, а не как скрипт. Для этого надо изменить текущую директорию на test/ в консоли, то есть выполнить команду python -m packA.a2 .
- Используйте импорт по абсолютному пути и test/ как корневую директорию (средний столбец в таблице выше). Это гарантирует, что будет работать напрямую запуск start.py . Чтобы напрямую запустить a2.py , мы можем изменить значение переменной sys.path в a2.py , для того чтобы включить в список путей для импорта путь test/packA/ , прежде чем sa2.py будет импортирован.
ПРИМЕЧАНИЕ. Обычно этот метод работает. Однако в некоторых установках Python переменная __file__ может быть некорректной. В этом случае вам нужно будет использовать встроенный в стандартную библиотеку Python пакет inspect. Ознакомьтесь с вопросом на StackOverflow для правильного использования приведенных выше инструкций.
- это решение не переносимо в другую среду выполнения, поэтому не рекомендуется его применять.
- инструкции по его использованию здесь: Как постоянно добавлять директорию в переменную среды PYTHONPATH
Пример 3: содержание sys.path может изменяться (вариант 2)
Более сложная проблема заключается в следующем. Предположим, что a2.py никогда не нужно запускать напрямую, но он импортируется такими файлами, как start.py и a1.py , которые будут запускаться напрямую из консоли.
В этом случае Решение №1 не будет работать. Но другие будут.
Пример 4: Импорт из родительской директории
Если мы не собираемся модифицировать переменные PYTHONPATH и sys.path в коде, то в этом случае основным ограничением импорта является следующее: При непосредственном запуске скрипта через консоль невозможно импортировать что-либо из его родительского каталога.
Например, если вы хотите запустить следующую команду python sa1.py , то невозможно в файле sa1.py что-либо импортировать из a1.py , не прибегая к обходным путям, описанным выше: изменения значений переменных PYTHONPATH или sys.path .
Во-первых, может показаться, что импорт по относительному пути (например, from .. import a1 ) может обойти это ограничение. Однако скрипт, который выполняется (в данном случае sa1.py ) при этом считается «модулем верхнего уровня». Попытка импортировать что-либо из каталога на уровень выше каталога этого скрипта приводит к появлению следующей ошибки: ValueError: attempted relative import beyond top-level package (попытка импорта по относительному пути за пределы пакета верхнего уровня).
Можно не писать сценарии, в которых необходимо импортировать что-либо из родительского каталога. В тех случаях когда это все таки необходимо сделать предпочтительным обходным путем является изменение переменной sys.path .
Python 2 VS. Python 3
Наиболее важные различия между тем, как Python 2 и Python 3 рассматривают инструкции import , были описаны выше. Они снова рассматриваются ниже, наряду с некоторыми другими менее важными отличиями.
Читайте также:
- Если интерпретатор Python запущен в интерактивном режиме: