М. Лутц - Изучаем Python (4-е издание)- 2011 (1126907), страница 88
Текст из файла (страница 88)
Мы уже немного познакомились с инструкцией for вовторой части книги и продолжим знакомство с ней в главе 13. В ���������������Python��������� 3.0 расширенная инструкция распаковывания может помещаться после слова for, гдеобычно указывается простое имя переменной:for (a, *b, c) in [(1, 2, 3, 4), (5, 6, 7, 8)]:...При таком использовании в каждой итерации интерпретатор будет просто присваивать очередной кортеж значений кортежу переменных. На первом проходе, например, будет выполнено присваивание, как если бы оно было реализовано в виде выражения:a, *b, c = (1, 2, 3, 4)# Переменная b получит значение [2, 3]352Глава 11.
Присваивание, выражения и printПеременные a, b и c можно использовать в теле цикла для ссылки на извлеченные компоненты. В действительности, в этом вообще нет ничего особенного –это лишь разновидность более общей операции присваивания. Как мы виделивыше в этой главе, того же эффекта можно добиться с помощью простой операции присваивания кортежей в обеих версиях Python 2.X и 3.X:for (a, b, c) in [(1, 2, 3), (4, 5, 6)]:# a, b, c = (1, 2, 3), ...И мы всегда сможем имитировать поведение расширенной инструкции распаковывания последовательностей в Python 2.6, применив операцию извлечениясреза:for all in [(1, 2, 3, 4), (5, 6, 7, 8)]:a, b, c = all[0], all[1:3], all[3]Мы пока недостаточно подготовлены к анализу подробностей синтаксиса цикла for, поэтому мы еще вернемся к этой теме в главе 13.Групповое присваиваниеПри групповом присваивании объект, расположенный справа, присваиваетсявсем указанным переменным.
В следующем примере трем переменным a, b и cприсваивается строка ‘spam’:>>> a = b = c = ‘spam’>>> a, b, c(‘spam’, ‘spam’, ‘spam’)Эта инструкция эквивалентна (но записывается компактнее) следующим треминструкциям присваивания:>>> c = ‘spam’>>> b = c>>> a = bГрупповое присваивание и разделяемые ссылкиИмейте в виду, что в этом случае существует всего один объект, разделяемыйвсеми тремя переменными (все они указывают на один и тот же объект в памяти). Такой способ присваивания хорошо подходит для неизменяемых объектов,например для инициализации нескольких счетчиков нулевым значением (незабывайте, что в языке Python переменная должна быть инициализирована,прежде чем к ней можно будет обратиться, поэтому вам всегда придется устанавливать начальные значения в счетчиках, прежде чем они смогут использоваться для счета):>>>>>>>>>(0,a = b = 0b = b + 1a, b1)Здесь изменение переменной b затронет только переменную b, потому что числане допускают возможность непосредственного изменения.
Если присваиваемый объект является неизменяемым, совершенно не важно, как много ссылокна него будет создано.Но, как обычно, следует проявлять осторожность, выполняя присваиваниепеременным изменяемых объектов, таких как списки или словари:353Инструкции присваивания>>> a = b = []>>> b.append(42)>>> a, b([42], [42])На этот раз, поскольку a и b ссылаются на один и тот же объект, непосредственное добавление значения к объекту через переменную b будет воздействоватьи на переменную a. В действительности это всего лишь другой пример взаимовлияния разделяемых ссылок, с которым мы впервые встретились в главе 6.Чтобы избежать этой проблемы, инициализацию изменяемыми объектамиследует производить в отдельных инструкциях, чтобы в каждой из них создавался новый пустой объект с помощью отдельных литеральных выражений:>>> a = []>>> b = []>>> b.append(42)>>> a, b([], [42])Комбинированные инструкции присваиванияНачиная с версии Python����������������������������������������������������������������������������������������������������2.0, в языке появился набор дополнительных инструкций присваивания, перечисленных в табл. 11.2.
Известные как комбинированные инструкции присваивания и заимствованные из языка C, они посуществу являются лишь более компактной формой записи. Они комбинируютв себе выражение и операцию присваивания. Например, следующие две формы записи практически эквивалентны:X = X + YX += Y# Традиционная форма записи# Новая, комбинированная форма записиТаблица 11.2. Комбинированные инструкции присваиванияX += YX &= YX -= YX |= YX *= YX ^= YX /= YX >>= YX %= YX <<= YX **= YX //= YКомбинированные операции присваивания существуют для любого поддерживаемого двухместного оператора. Например, ниже приводится два способа прибавления 1 к значению переменной:>>>>>>>>>2>>>>>>3x = 1x = x + 1xx += 1x# Традиционная форма записи# КомбинированнаяЕсли комбинированную инструкцию применить к строкам, будет выполненаоперация конкатенации. Таким образом, вторая строка ниже эквивалентна более длинной инструкции S = S + “SPAM”:>>> S = “spam”>>> S += “SPAM”# Выполняется операция конкатенации354Глава 11.
Присваивание, выражения и print>>> S‘spamSPAM’Как показано в табл. 11.2, для каждого двухместного оператора (то есть дляоператора, слева и справа от которого располагаются значения, участвующиев операции) в языке Python существует своя комбинированная инструкцияприсваивания. Например, инструкция X *= Y выполняет умножение и присваивание, X >>= Y – сдвиг вправо и присваивание, и так далее.
Инструкция X //= Y(деление с округлением вниз) была добавлена в версии Python 2.2.Комбинированные инструкции присваивания обладают следующими преимуществами:1•• Уменьшается объем ввода с клавиатуры. Нужно ли мне продолжать?•• Левая часть инструкции должна быть получена всего один раз. В инструкции «X += Y» X может оказаться сложным выражением, которое в комбинированной форме должно быть вычислено всего один раз. В более длиннойформе записи «X = X + Y» X появляется дважды, и поэтому данное выражениедолжно быть вычислено дважды. Вследствие этого комбинированные инструкции присваивания выполняются обычно быстрее.•• Автоматически выбирается оптимальный алгоритм выполнения. Для объектов, поддерживающих возможность непосредственного изменения, комбинированные инструкции присваивания автоматически выполняютсянепосредственно на самих объектах, вместо выполнения более медленнойоперации копирования.И последний момент, который требует дополнительных разъяснений.
Комбинированные инструкции присваивания, при применении к изменяемым объектам, могут служить для оптимизации. Вспомним, что списки могут расширяться разными способами. Чтобы добавить в список единственный элемент,мы можем выполнить операцию конкатенации или вызвать метод append:>>>>>>>>>[1,>>>>>>[1,L = [1, 2]L = L + [3] # Конкатенация: более медленнаяL2, 3]L.append(4) # Более быстрая, но изменяет сам объектL2, 3, 4]А чтобы добавить несколько элементов, мы можем либо снова выполнить операцию конкатенации, либо вызвать метод extend2:>>>>>>[1,>>>L = L + [5, 6] # Конкатенация: более медленнаяL2, 3, 4, 5, 6]L.extend([7, 8]) # Более быстрая, но изменяет сам объект1Программисты C/C++, конечно, заметят, что несмотря на появление в языке Pythonтаких инструкций, как X +=Y, в нем до сих пор отсутствуют операторы инкрементаи декремента (например, X++, --X).
Однако эти операторы вообще не соответствуютмодели языка Python, в котором отсутствует возможность непосредственно изменятьнеизменяемые объекты, такие как числа.2Как предлагалось в главе 6, для этого также можно было бы использовать операциюприсваивания срезу (например, L[len(L):] = [11,12,13]), но этот прием работает практически так же, как метод extend.355Инструкции присваивания>>> L[1, 2, 3, 4, 5, 6, 7, 8]В обоих случаях операция конкатенации несет в себе меньше побочных эффектов при работе с разделяемыми ссылками на объекты, но вообще она выполняется медленнее, чем эквивалентные операции, воздействующие на объект непосредственно.
Операция конкатенации должна создать новый объект, копиюсписка слева, и затем скопировать в него список справа. В противовес ей метод,воздействующий на объект непосредственно, просто добавляет новый элементв конец блока памяти.При использовании комбинированной инструкции присваивания для расширения списка мы можем не думать об этих деталях – интерпретатор Python��������������автоматически вызовет более быстрый метод extend вместо использования болеемедленной операции конкатенации, которую предполагает оператор +:>>> L += [9, 10] # Выполняется как L.extend([9, 10])>>> L[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]Комбинированные инструкции присваиванияи разделяемые ссылкиТакой порядок вещей чаще всего именно то, что нам требуется, но необходимоучитывать – он предполагает, что применительно к спискам операция += выполняет изменения непосредственно в объекте, а это далеко не то же самое, чтооперация конкатенации +, в результате которой всегда создается новый объект.