Лутц М. - Изучаем Python (1077325), страница 117
Текст из файла (страница 117)
паве, рт11е.", аоео1е. Гт1е ргтпт "-"*30 сопит = О Как мы видели в главе 16, функция может получить доступ к вмещающему модулю с помощью таблицы вув, аоео1ев, что позволяет имитировать действие инструкции ц1оЬа1. Например, эффект действия инструкций д!оЬа1 Х, Х=О внутри функции можно реализовать (хотя для втого придется ввести с клавиатуры значительно больше символов!) так: )арогт вув; ц1оо=вув. аосо1ев( паве ]; ц1оЬ.
Х=О. Не забывайте, что каждый модуль имеет атрибут паае; внутри функции, принадлежащей модулю, он выглядит как глобальное нмя. Этот прием обеспечивает еще один способ изменения одноименных локальных и глобальных переменных внутри функции. 54б Глава 21. Дополнительные возможности модулей гог аттг тп ассо1е. стет .кеуэ(): рг1пт "%020) вв" % (ссопт, вттг), 11 аттг(0:2] == " рг1пт "<Ьот11-тп пэае>" е1ве. рыпт детэттг(асоо1е, аттг) своп! = своп!+1 № Сквнирпввть пространство имен № Пропустить Г1]в и др. № тп ив, нтв и . П1ст Гвггг] тг оэгьсэе: рг!пт "-"*30 ргтп1 асоо1е.
паве , "Ьвв %0 пваев" % саоп! рг1пт "-" ° 30 !Г пвае == " автп тарсгт ау01г 1тэт1пд(аувтг) № Код свиопровврки: виввсти свое № првстрвнствв иивн В модуле, в самом конце, реализована логика самопроверки, которая за- ставляет модуль импортировать самого себя и вывести содержимое сво- его пространства имен. Ниже показан результат работы этого модуля: С:<,рутпсп> рутьсп ауе1г.ру пвае: ауотг 111е', ауотг.ру 00) ГТ1е <Ьо111-1п пвае> 01) свае <Ьо111-тп паве> 02) 1>вт!пд <гопс(1сп 11вт1пд вт 885450> 03) Ссс <Ьо111-1п пвае> 04) Ью11тпэ <Ьо>11-1п свае> 05) оегвове 1 аузтг Ьвэ 8 пваеэ Инструменты, такие как ауе1г. 1тв!!пд, могут быть предварительно загружены в пространство имен интерактивной оболочки импортированием нх в файле, указанном з переменной окружения РУТН0%5ТАВТОР.
Так как программный код этого файла выполняется в интерактивном пространстве имен (модуль аэтп ), такой способ импортирования часто используемых инструментов позволит вам сэкономить время на вводе инструкций вручную. Дополнительная информация приводится в приложении А. С функцией детастг и родственными ей мы встретимся еще раз позднее. Самое важное здесь, что ауд! г — это программа, которая позволяет исследовать другие программы. Так как интерпретатор не скрывает внутреннее устройство модулей, вы можете реализовать обработку любых объектов единообразно.' Типичные проблемы при работе с модулями Типичные проблемы при работе с модулями В этом разделе мы рассмотрим обычный набор пограничных ситуаций, которые делают жизнь интересной для тех, кто только начинает осваивать язык Ру1)топ.
Некоторые нз них настолько неочевидны, что трудно привести к ним примеры, но в большинстве своем они иллюстрируют важные сведения о языке. Порядок следования инструкций на верхнем уровне имеет значение Когда модуль впервые импортируется (или загружается повторно), интерпретатор выполняет инструкции в нем одну за другой, сверху вниз.
Из этого следует несколько замечаний, которые касаются опережающих ссылок на переменные, которые следует подчеркнуть особо: ° Инструкции программного кода на верхнем уровне в файле модуля (не вложенные в функцию) выполняются, как только интерпретатор достигает их в процессе импортирования. По этой причине он не может ссылаться на имена, присваивание которым производится ниже. ° Программный код внутри функций не выполняется, пока функция не будет вызвана, — разрешение имен внутри функций не производится до момента их вызова, поэтому они обычно могут ссылаться на имена, расположенные в любой части файла. Вообще опережающие ссылки доставляют беспокойство только в программном коде верхнего уровня, который выполняется немедленно; функции могут ссылаться на любые имена.
Ниже приводится пример, демонстрирующий опережающие ссылки: типс1() и Ошибка: иил "гипс)" еше не сушествует бег твист(); ргтпт твпс2() и Оуе поиск ииени "Гвпс2" будет выполнен позднее и Ошибка: иип "гвпс2" еие не существуЕт твист() Еет тспс2(): гетогп "Нв11о" твпс1() а ОК: "таис)" и "Гипс2" определены Когда этот файл будет импортироваться (или запускаться как самостоятельная программа), интерпретатор Ру()топ будет выполнять его инструкции сверху вниз.
Первый вызов (опс1 потерпит неудачу, потому что инструкция Оет для имени Топс1 еще не была выполнена. Вызов Твпс2 внутри т'ппс1 будет работать без ошибок при условии, что к моменту вызова гппс1 инструкция Оег' гопс2 уже будет выполнена (этого еще не произошло к моменту второго вызова Тппс1 на верхнем уровне).
Последний вызов топот в конце файла будет выполнен успешно, потому что оба имени, ( по с1 и Т о пс2, уже определены. 548 Глава 21. Дополнительные возможности модулей Импортирование модулей по имени в виде строки Имя модуля в инструкции!арогт или Ггоа является именем переменной. Тем не менее иногда ваша программа будет получать имя модуля, который следует импортировать, в виде строки во время выполнения (например, в случае, когда пользователь выбирает нмя модуля внутри графического интерфейса). К сожалению, невозможно напрямую использовать инструкции импорта для загрузки модуля, имя которого задано в виде строки, — в этих инструкциях интерпретатор ожидает получить имя переменной, а не строку.
Например: »> 1арогт втг1пЯ Е>1е "<атс1п>", 1> пе 1 аарогт "атг1пЯ БуптахЕггог: тпна1Ш пумах Точно так же невозможно импортировать модуль, если просто присво- ить строку переменной: х = "жмпя" ппрогт х Чтобы решить эту проблему, необходим специальный инструмент, выполняющий динамически загрузку модулей, имена которых создаются в виде строк во время выполнения. Обычно для этого конструируется строка программного кода, содержащая инструкцию тарогт, которая затем передается инструкции ехес для исполнения: »> аобпаае = "а1г1пя" »> ехес "1арогт " + аобпаае »> атг1пя <аосоте 'атгапя'> а Ввполнявтся как строка лрограииногс кода а Модуль бил иипортироввн в пространство иивн Инструкция ехес (и родственная ей функция еуа1, используемая для вычисления значений выражений) скомпилирует строку в код и передаст его интерпретатору для исполнения. В языке Ру1)топ компилятор байт-кода доступен непосредственно во время выполнения, поэтому можно писать программы, которые конструируют и выполняют другие программы, как в этом случае.
По умолчанию инструкция ехес выполняет программный код в текущей области видимости, но существует возможность передавать ей необязательные словари пространств имен. Смешивание инструкций беб с программным кодом верхнего уровня не только осложняет его чтение, но еще и ставит его работоспособность в зависимость от порядка следования инструкций. Если вам необходимо объединять в модуле программный код, выполняемый непосредственно, с инструкциями бет, возьмите за правило помещать инструкции бет в начало файла, а программный код верхнего уровня — в конец файла. При таком подходе ваши функции гарантированно будут определены к моменту выполнения программного кода, который их использует.
549 Типичные проблемы при работе с модулями Единственный настоящий недостаток инструкции ехес состоит в том, что она должна компилировать инструкцию сарогс всякий раз, когда она запускается, — если импортировать приходится достаточно часто, программный код может работать немного быстрее при использовании встроенной функции 1арогт, которая выполняет загрузку модуля, получая его имя в виде строки. Результат получается тот же самый, но функция 1арсгт возвращает объект модуля, поэтому его надо присвоить переменной, чтобы сохранить: »> аобпаае "зтг1по" »> зтг1по = 1арогС (аобпвее) »> зсг1по <аобо1в зтг!по > Инструкция вагою создает копии, а не ссылки Несмотря на то что инструкция Тгоа широко применяется, она часто становится источником самых разных проблем. При выполнении присваивания именам в области видимости импортирующего модуля она не создает синонимы, а копирует имена. Результат будет тем же самым, что и для любых других инструкций присваивания в языке РуС)соп, но есть одно тонкое отличие, особенно когда программный код, использующий объекты совместно, находится в разных файлах.
Например, предположим, что у нас имеется следующий модуль (пеасег(1.ру): Х = 99 бе1 рг1птвг(); рг1пт Х Если импортировать эти два имени с помощью инструкции сгоа в другом модуле (левсес(ю.ру), будут получены копии этих имен, а не ссылки на них. Изменение имени в импортирующем модуле приведет к изменениям только локальной версии этого имени, а имя в модуле певвес)1.ру будет иметь прежнее значение: ггоа певтеб1 сарогС Х, рг1птег В Копировать имена Х = 88 а изивнит только локальную версию "Х"' ргсптег() а х в пезсеб1 по-лреянвиу будет равно 99 % русвоп певтеб2.ру 99 Однако, если выполнить импорт всего модуля с помощью инструкции сарогс и затем изменить значение с использованием полного имени, зто приведет к изменению имени в файле пеасег(1.ру.