Что означает "для x в y или z:" в Python?
Я пытаюсь разобрать и де-обфускать этот код-пинг-код, генерирующий mandlebrot:
_ = (
255,
lambda
V ,B,c
:c and Y(V*V+B,B, c
-1)if(abs(V)<6)else
( 2+c-4*abs(V)**-0.4)/i
) ;v, x=1500,1000;C=range(v*x
);import struct;P=struct.pack;M,\
j ='<QIIHHHH',open('M.bmp','wb').write
for X in j('BM'+P(M,v*x*3+26,26,12,v,x,1,24))or C:
i ,Y=_;j(P('BBB',*(lambda T:(T*80+T**9
*i-950*T **99,T*70-880*T**18+701*
T **9 ,T*i**(1-T**45*2)))(sum(
[ Y(0,(A%3/3.+X%v+(X/v+
A/3/3.-x/2)/1j)*2.5
/x -2.7,i)**2 for \
A in C
[:9]])
/9)
) )
источник: http://preshing.com/20110926/high-resolution-mandelbrot-in-obfuscated-python/
Я нашел цикл for с интересным завихрением, и я не уверен, что это просто часть обфускации, или если на самом деле есть функция. Средняя линия:
for X in j('BM'+P(M,v*x*3+26,26,12,v,x,1,24))or C:
Что делает или делает в конце?
Я снова написал код для ясности:
import struct
image_name = "M.bmp"
mandlebrot = lambda V,B,c :c and Y(V*V+B,B,c-1) if(abs(V)<6) else (2+c-4*abs(V)**-0.4)/i
y,x = 600,800
pixelrange = range(y*x)
package = struct.pack
image = open(image_name,'wb').write
for X in image('BM'+package('<QIIHHHH',y*x*3+26,26,12,y,x,1,24)) or pixelrange:
i = 255
Y = mandlebrot
image(package('BBB',*(lambda T:(T*80+T**9*i-950*T**99,T*70-880*T**18+701*T**9,T*i**(1-T**45*2)))(sum([Y(0,(A%3/3.+X%y+(X/y+A/3/3.-x/2)/1j)*2.5/x-2.7,i)**2 for A in pixelrange [:9]])/9)))
Ответы
Ответ 1
Doing:
for x in y or z:
совпадает с:
for x in (y or z):
Если y
оценивается как True
, цикл for будет проходить через него. В противном случае он будет проходить через z
.
Ниже приведена демонстрация:
>>> y = [1, 2, 3]
>>> z = [4, 5, 6]
>>> for x in y or z:
... print x
...
1
2
3
>>> y = [] # Empty lists evaluate to False in Python
>>> for x in y or z:
... print x
...
4
5
6
>>>
Ответ 2
or
возвращает первый операнд, если он имеет значение boolean true, в противном случае второе. Boolean false в основном означает 0, False, None или "пустой объект" (пустая строка, список и т.д.); boolean true - это все остальное. Поэтому j('BM'+P(M,v*x*3+26,26,12,v,x,1,24)) or C
будет оценивать результат вызова j(...)
, если этот результат непусто, в противном случае - C
. Таким образом, он будет перебирать результат j(...)
, если этот результат не пуст, иначе он будет перебирать более C
.