Life is short, just use Python.
自從08年接觸Python,就有愛不釋手的感覺,逐漸地,有些不忍地疏遠了Perl 和Shell程式設計,因為python 的優雅麼? 不全是,主要是可以高效開發吧。
那一行程式碼可以幹什麼呢?
有趣
我孩子的英文名叫andy,也許當初教他寫程式的時候,如果先秀一下這行程式碼,可能就更能激起他對程式碼的興趣了。
1 |
>>> print'\n'.join([''.join([('AndyLove'[(x-y)%8]if((x*0.05)**2+(y*0.1)**2-1)**3-(x*0.05)**2*(y*0.1)**3<=0 else' ')for x in range(-30,30)])for y in range(15,-15,-1)]) |
再python 裡執行它,會輸出一個字元拼出的心形。
字元圖形還是很有趣的,有一個著名的影象叫mandelbrot。Mandelbrot影象中的每個位置都對應於公式N=x+y*i 中的一個複數,高中學過複數的都還應該有印象。每個位置用引數N來表示,它是x*x+y*y的平方根。如果這個值大於或等於2,則這個數字對應的位置值是0。如果引數N的值小於2,就把N的值改為N*N-
N(N=(x*x-y*y-x)+(2*x*y-y)*i)),並再次測試這個新N值。wiki百科給出的影象是這樣的:
讓我們用一行程式碼畫一個Mandelbrot:
1 |
>>> print'\n'.join([''.join(['*'if abs((lambda a:lambda z,c,n:a(a,z,c,n))(lambda s,z,c,n:z if n==0else s(s,z*z+c,c,n-1))(0,0.02*x+0.05j*y,40))2 else' 'for x in range(-80,20)])for y in range(-20,20)]) |
高效
對於隨手小工具而言,更是Python的拿手好戲。
一行程式碼列印九九乘法表:
1 |
print '\n'.join([' '.join(['%s*%s=%-2s' % (y,x,x*y) for y in range(1,x+1)]) for x in range(1,10)]) |
輸出:
一行程式碼計算出1-1000之間的素數
1 |
print(*(i for i in range(2, 1000) if all(tuple(i%j for j in range(2, int(i**.5)))))) |
一行程式碼可以輸出前100項斐波那契數列的值:
1 |
print [x[0] for x in [ (a[i][0], a.append((a[i][1], a[i][0]+a[i][1]))) for a in ([[1,1]], ) for i in xrange(100) ]] |
一行程式碼實現階乘,而且還帶互動:
1 2 3 |
>>> reduce ( lambda x,y:x*y, range(1,input()+1)) 10 3628800 |
一行程式碼實現攝氏度與華氏度之間的轉換器:
1 2 3 4 5 6 7 8 |
>>> print((lambda i:i not in [1,2] and "Invalid input!" or i==1 and (lambda f:f<-459.67 and "Invalid input!" or f)(float(input("Please input a Celsius temperature:"))*1.8+32) or i==2 and (lambda c:c<-273.15 and "Invalid input!" or c)((float(input("Please input a Fahrenheit temperature:"))-32)/1.8))(int(input("1,Celsius to Fahrenheit\n2,Fahrenheit to Celsius\nPlease input 1 or 2\n")))) 1,Celsius to Fahrenheit 2,Fahrenheit to Celsius Please input 1 or 2 1 Please input a Celsius temperature:28 82.4 >>> |
至於字串排序和快速排序更是手到擒來。
1 2 3 |
"".join((lambda x:(x.sort(),x)[1])(list(‘string’))) qsort = lambda arr: len(arr) > 1 and qsort(filter(lambda x: x<=arr[0], arr[1:] )) + arr[0:1] + qsort(filter(lambda x: x>arr[0], arr[1:] )) or arr |
內涵
看一看下面一行python程式碼,可能就要暈了:
這是原來為了激發孩子程式設計興趣,讓孩子練習的程式碼,它的真實面貌是大致這樣的:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
def guess_my_number(n): while True: user_input = raw_input("Enter a positive integer to guess: ") if len(user_input)==0 or not user_input.isdigit(): print "Not a positive integer!" else: user_input = int(user_input) if user_input > n: print "Too big ! Try again!" elif user_input < n: print "Too small ! Try again!" else: print "You win!" return True guess_my_number(42) |
實際上,只要理解了函數語言程式設計,使用神奇的Lambda,配合列表推導以及複雜一點的判斷語句,任何的python 程式碼都可以轉換成一行程式碼的。
例如,取一個列表裡的隨機數
1 2 |
import random as rnd print rnd.choice([2,3, 5,7, 11,13,17]) |
轉換成Lambda 可以是:
1 |
print (lambda rnd: rnd.choice([1, 2, 3, 10]))(__import__('random')) |
這些程式碼出了覺得好玩,主要是可以幫助我們瞭解某些Python的雕蟲小記,尤其是神奇的Lambda 用法。
延展
當然,還有其他好玩的地方,輸入下面這一行
1 |
import antigravity |
它開啟了瀏覽器,展示網站上的漫畫和相關內容:
我們可以把python的檔案打包,做成庫的形式,然後import進來,是一種偷換概念和前提的一行程式碼。例如,為了與windows 傳輸檔案,再Mac上臨時搭個ftp:
1 |
$ python -m pyftpdlib |
這當然要依賴pyftpdlib 這個庫了,機器上沒有,pip install pyftpdlib
就可以了。
如果一行程式碼中允許分號存在,那就只是犧牲可讀性而已了,那就基本上無所不能。
在連網的前提下,獲取公網IP地址
1 |
python -c "import socket; sock=socket.create_connection(('ns1.dnspod.net',6666)); print sock.recv(16); sock.close()" |
一行程式碼就可以輕易寫個小遊戲了,來模擬一下golf擊球。
1 2 |
python -c "import math as m;a,v=eval(input());[print('%03d'%x+' '*m.floor(0.5+x*m.tan(a)-x*x/(v*m.cos(a)))+'o') for x in range(102)]" |
輸入角度和力量大小如(0.8,80),就能得到一條字元描畫的拋物線了。
增加上while 等語句,畫一個沒完沒了的
1 |
python -c "while 1:import random;print(random.choice('╱╲'), end='')" |
最後,一行程式碼以python 的哲學結束吧。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
$ python -c "import this" The Zen of Python, by Tim Peters Beautiful is better than ugly. Explicit is better than implicit. Simple is better than complex. Complex is better than complicated. Flat is better than nested. Sparse is better than dense. Readability counts. Special cases aren't special enough to break the rules. Although practicality beats purity. Errors should never pass silently. Unless explicitly silenced. In the face of ambiguity, refuse the temptation to guess. There should be one-- and preferably only one --obvious way to do it. Although that way may not be obvious at first unless you're Dutch. Now is better than never. Although never is often better than *right* now. If the implementation is hard to explain, it's a bad idea. If the implementation is easy to explain, it may be a good idea. Namespaces are one honking great idea -- let's do more of those! |
參考資料:
http://www.zhihu.com/question/37046157
https://onelinepy.herokuapp.com/
https://github.com/csvoss/onelinerizer
打賞支援我寫出更多好文章,謝謝!
打賞作者
打賞支援我寫出更多好文章,謝謝!