最近在網路上看到一個非常有趣的 Python 遊戲通關網站,一共有 33 關,每一關都需要利用 Python 知識解題找到答案,然後進入下一關。很考驗 Python 的綜合掌握能力,例如有的闖關需要用到正規表示式,有的要用到爬蟲。
我們平常學 Python 都是依照章節順序、套件或模組來學,容易前學後忘。剛好可以拿這個網站來綜合測試一下對 Python 的掌握情況,以便查缺補漏。
來說說這個網站怎麼玩。
這是網站主頁,很有歷史感對吧,已經誕生了十幾年了。但千萬不要因為看著像老古董而小瞧它。
我們來玩玩看,點擊「get challenged」開始挑戰。
第0 關是Warming up 熱身環節:
這一關要求是修改URL 鏈接,給的提示是電腦上的數學表達式:2 的38 次方,所以大概就是要算出數值,然後修改url 進入下一關。
所以這關就是考 Python 的基本數值運算,你知道怎麼算麼?
打開Python 自帶終端,一行程式碼就能計算出結果:
把原連結中的 0替換為 274877906944回車就會進入下一關:
遊戲這就正式開始了。圖片中的筆記本給了三組字母,很容易發現規律:前面的字母往後移動兩位是後面的字母。
那麼需要做的就是根據這個規律把下面的提示字串,做位移解密得到真正的句子含義:
這題考察字符串編碼和for 循環相關知識,代碼實作如下:
text = '''g fmnc wms bgblr rpylqjyrc gr zw fylb. rfyrq ufyr amknsrcpq ypc dmp. bmgle gr gl zw fylb gq glcddgagclr ylb rfyr'q ufw rfgq rcvr gq qm jmle. sqgle qrpgle.kyicrpylq() gq pcamkkclbcb. lmu ynnjw ml rfc spj.''' text_translate = '' for i in text: if str.isalpha(i): n = ord(i) if i >= 'y': n = ord(i) + 2 - 26 else: n = ord(i) + 2 text_translate += chr(n) else: text_translate += i print(text_translate)
得到結果:
i hope you didnt translate it by hand. thats what computers are for. doing it in by hand is inefficient and that's why this text is so long. using string.maketrans()is recommended. now apply on the url.
作者很風趣,當然不能手動去一個推算了,推薦用string.maketrans() 這個方法解決,我們上面採取的是比較直接的方法,官方給了更精簡的方法:
import string l = string.lowercase t = string.maketrans(l, l[2:] + l[:2]) print (text.translate(t))
然後把url 中的 map 改為ocr回車就來到了第2 關:
作者接著說過關的提示可能在書裡(當然不可能了)也可能在網頁原始碼裡。那就右鍵查看原始程式碼往下拉看到綠色區域,果然找到了問題:
#意思是:要在下面這一大串字元裡找到出現次數最少的幾字元
考察了這麼多知識點:
正規表示式擷取字串
#list 計數
##條件語句如果是你,你會怎麼做? 來看下,十行程式碼快速實作:
import requests url = 'http://www.pythonchallenge.com/pc/def/ocr.html' res = requests.get(url).text text = re.findall('.*?<!--.*-->.*<!--(.*)-->',res,re.S) # list转为str便于遍历字符 str = ''.join(text) lst = [] key=[] #遍历字符 for i in str: #将字符存到list中 lst.append(i) #如果字符是唯一的,则添加进key if i not in key: key.append(i) # 将list列表中的字符出现字数统计出来 for items in key: print(items,lst.count(items))首先,用 Requests 請求網頁接著用正規提取字串,接著 for 迴圈計算每個字元出現的次數。
% 6104 $ 6046 @ 6157 _ 6112 ^ 6030 # 6115 ) 6186 & 6043 ! 6079 + 6066 ] 6152 * 6034 } 6105 [ 6108 ( 6154 { 6046 e 1 q 1 u 1 a 1 l 1 i 1 t 1 y 1可以看到出現次數最少的就是最後幾個字符,合起來是“equality”,替換 url 字符就闖過過了第 2 關進入下一關繼續挑戰。是不是有點意思? 後面每一關都需要用到相關的Python 技巧解決,例如第4 關: 這一關作者弄了個小惡作劇,要手動輸入數值到url 中然後回車,你以為這樣就結束了?並沒有它有會不斷重複彈出新的數值讓你輸入,看起來無窮無盡。
所以,这一关肯定不能采取手动输入的方法闯关,自然要用到 Python 了。要实现自动填充修改 url 回车跳转到新 url,循环直到网页再也无法跳转为止这一功能。
如果是你,你会怎么做?
其实,一段简单的爬虫加正则就能搞定。思路很简单,把每次网页中的数值提取出来替换成新的 url 再请求网页,循环下去,代码实现如下:
import requests import re import os # 首页url resp = requests.get( 'http://www.pythonchallenge.com/pc/def/linkedlist.php?nothing=12345').text url = 'http://www.pythonchallenge.com/pc/def/linkedlist.php?nothing=' # 计数器 count = 0 while True: try: # 提取下一页动态数值 nextid = re.search('\d+', resp).group() count = count + 1 nextid = int(nextid) except: print('最后一个url为:%s' % nexturl) break # 获取下一页url nexturl = url + str(nextid) print('url %s:%s' % (count, nexturl)) # 重复请求 resp = requests.get(nexturl).text
输出结果如下:
可以看到,最终循环了 85 次找到了最后一个数字16044,输入到 url 中就闯关成功。
如果遇到不会做的题,可以在这里找到参考答案:
中参考文教程:
https://www.cnblogs.com/jimnox/archive/2009/12/08/tips-to-python-challenge.html
官方参考教程:
http://garethrees.org/2007/05/07/python-challenge/
以上是透過遊戲闖關來測試自己的python掌握能力的詳細內容。更多資訊請關注PHP中文網其他相關文章!