混乱的 Python:生命游戏

偶尔在 python.cn上看到一篇《谁说使用 Python 你就写不出混乱的代码?》,地址在这里。觉得非常有意思,记得使用Perl的时候,因为其灵活的语法,可以有很优美的代码书写方式,也可以写出非常混乱的代码。而Python,从一开始就致力于消除这种混乱,缩进的结构使得Python代码相对其他语言整齐了许多,刚开始用Python的时候虽然也想过能不能写出乱糟糟的代码,但是一开始就败在缩进上,也就慢慢忘记了。

分析

这次看到,自然是兴致盎然,稍微研究一下,发现它是把一些看似乱糟糟的字符串重新编码了,然后交给exec执行了。关键就在这一句:

exec(reduce(lambda x,i:x.replace(chr(i),"\n "[34-i:]), range(35), long_long_str)

用了Python的几个函数式编程的函数。lambda非常常用,reduce相对来说少用一些,但是需要使用的时候非常好用!

可能有人不是非常熟悉这个reduce函数,因为其他的语言里未必有这个东西,其实reduce的含义是折叠(我知道这么说还是很抽象),举一个小例子,让你计算0~100所有数字的和(OK,我知道你知道答案,但请编程实现),一般的话,就会这么写(这里不考虑等差数列的公式):

sum = 0
for i in range(101):
    sum += i
print sum

很简单,但是使用reduce会更简单:

print reduce(lambda x,y:x+y, range(101))

没有循环,没有中间变量,一句搞定!

它的运行过程如下图所示

reduce

reduce接受两个或三个参数,第一个为要执行的函数,第二个为一个序列,第三个这里先无视。

在我们这个例子的执行过程中,reduce从序列中取出两个值(0和1),相加计算出一个结果(1),然后再从序列中取出一个值(2),和刚刚得到结果(1)相加再得到一个结果(3),然后重复这个过程直到结束。

上面的混乱代码中,reduce对字符串不停的修正,把空白(空格和回车)删除,把!替换为空格,”替换为回车空格。然后得到一个格式完好的代码。换句话说,通过这个运算方法,我们把原先的代码中的空格和回车先用”或!替换以后,可以在代码中随意增加空格和换行了,图片的样子就是这么出来的。

生命游戏

知道了原理,我们就能很容易的造一个我们自己喜欢的混乱python代码了。不过我们可以看到,虽然有任意的回车和空白,我们还是能很容易的在字符串里看到python里的关键字,这样显得不够酷……

我把这个东西再升级了一下,字符全都混淆(ASCII值加1),基本就成乱码了,然后再用这个方法调整空白。

用pygame做了一个生命游戏的演示,代码写成一个骷髅头的样子(抱歉,非常的不像。。。。)。关于生命游戏,相信有些编程经验的应该会知道,可以在维基上看看说明,百度百科里好像更全一些,而这里,则有更学术性的研究。

代码完成后是这个样子的:

XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX=r'''
               jnqpsu!sboepn"m>\\
             sboepn.d      ipjdf((
           0,2)                 )!gp
          s!j                      !jo
         !sb                        ohf
         (59)                      ^!gp
         s!j!                      jo!s
         bohf  (               5   9)^"
         l>\\  0!g           ps!  j!jo!
         sbohf  (59)^!   gps!j!   jo!sb
         ohf(59  )^"jn   qpsu!   qzhbnf
         "qzhbn   f.jo   ju()"   t>qzhb
         nf.ejt                  qmbz.t
         fu`npe       f((5        90,59
         0),0,43)"    p>Usv  f"efg!{(y,
         z);"!n>0"    !gps!  j!jo!(y-2,
          y,y+2);"    !!gps   !k!jo!(z
           -2, z   ,        z +2);"!!
           !jg! j > > y ! b o e !z>>k
  ;"!!!!d   poujo v f"!!!jg!(j= 0!ps !j?58!ps!
k=0!ps!k? 58);"!!!!dp oujovf" !!!jg!(m\j ^\k^);
"!! !  !n+>2"!sfuv s o ! n " jnqpsu!dpqz"xijmf!
p;"   !t.gjmm((36              6,366,36   6))"!
gps    !f!jo!qzhb              nf.fwfo    u.hfu
();    "!!jg!f.uz              qf>>23    ;"!!!p
>Gbmtf" !gps!y!jo!y   s      bohf(59     );"!!g
ps!z!jo!ysbo   hf(59);"!!!n>{(y   ,!z)"!!!jg!n>
  >3;"!!!!l\      y^\z^>m\y^\     z^"!!!fmjg!
     n>>4;"!!!      !l\y^\z     ^>2"!!!f
         mtf;"!!!!l\y^\z     ^>0"!!!
      jg!l\y^\z^;"!!!     !t.gjmm((0,0,3
66),(z*20,y*20,20,     20))"!!!!qzhbnf.esbx.sfd
u(t ,(0,0,0),(       z*20,y*2    0,20,20),2)"!m
>d                pqz.effqdpqz    (l  )"     !q
zhb  nf   .ej t qmbz.vqebuf()"!q z h  b      nf
.ujn  f    .xbju(200)"''';      exec( ''.join(
map(      lambda x:chr(ord(x)-1) if        ord(
x) > 49 else x,   reduce(   lambda x, i :  x.
replace( chr(i), "\n "[34-i:35-i] ), range(35),
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX))))

为了“清晰”起见,解码的代码就写在了最后没有混杂在里面了,稍微有些明白的就能打出原始代码来。

执行需要安装pygame库,执行效果如下(示意图,非连续):

希望能有“玩心”的朋友继续研究,让Python再混沌一些吧~~

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。