正式开始第1关,这一关的URL的特殊部分是map。
这关的图片上有一个本子,上面写着K→M,O→Q,E→G,稍微思索就能发现这几个字母都是按照字母表的顺序向后移动了两位,那么最投机取巧的方法就是把map的3个字母按照这个规则改动就行了。虽然这个是我想要的结果,但却并不是我想要的过程,还是得按照正规的编程方法来,而且下面还有其他的提示,姑且先全部看完。
图片下方有一行提示:三思而后解题。再下方是几行毫无规则的字母组合,很显然是要通过上述规则转化的。
对于这样的转换,很自然的想到的就是ASCII码。查阅资料后,发现python中ASCII码和字符的相互转换使用的是ord()函数和chr()函数。然后要注意的是y和z应该是要转换成a和b,那么这两个就是向前移动24位。写好程序之后发现一个问题,由于我用来记录转换后内容的是一个list表,如果直接print的话不利于阅读,继续查阅资料后发现join()函数可以实现list到string的转换,于是正式的代码如下:
code = 'g fmnc wms bgblr rpylqjyrc gr zw fylb. rfyrq ufyr amknsrcpq ypc dmp.\
bmgle gr gl zw fylb gq glcddgagclr ylb rfyrq ufw rfgq rcvr gq qm jmle.\
sqgle qrpgle.kyicrpylq() gq pcamkkclbcb. lmu ynnjw ml rfc spj.' decode = []
for i in code:
if ord('a') <= ord(i) <= ord('x'):
decode.append(chr(ord(i) + 2))
elif ord('y') <= ord(i) <= ord('z'):
decode.append(chr(ord(i) - 24))
else:
decode.append(i)
print(''.join(decode))
运行后输出了转换后的提示:i hope you didnt translate it by hand. thats what computers are for.doing it in by hand is inefficient and thats why this text is so long.using string.maketrans() is recommended. now apply on the url.
果然按部就班的做完还是有好处的,提示里提到了string.maketrans()这个函数。根据python的官方文档说明,string.maketrans(from,to)函数给translate()函数提供一个从from映射到to的翻译表,而translate(s,table,[deletechars])函数则是从s中删除出现在deletechars里(如果有的话)的字符,再根据table的的规则进行转换。
于是使用string下的maketrans()和translate()的代码如下:
import string code = 'g fmnc wms bgblr rpylqjyrc gr zw fylb. rfyrq ufyr amknsrcpq ypc dmp.\
bmgle gr gl zw fylb gq glcddgagclr ylb rfyrq ufw rfgq rcvr gq qm jmle.\
sqgle qrpgle.kyicrpylq() gq pcamkkclbcb. lmu ynnjw ml rfc spj.' print(string.translate(code, string.maketrans('abcdefghijklmnopqrstuvwxyz', 'cdefghijklmnopqrstuvwxyzab')))
在第2关的左下角提示到,把URL里的pc改成pcc可以看到前一关的各种解法。点开一看果然很多,还包括了其他语言的解法。值得一提的是,一个解法中使用string.ascii_lowercase和string.ascii_lowercase[2:] + string.ascii_lowercase[:2]的写法代替了我的代码中‘abcdefghijklmnopqrstuvwxyz'和’cdefghijklmnopqrstuvwxyzab‘这样看起来比较乱也容易出错的写法。首先,这里的string.ascii_lowercase就是从a到z的所有小写字母组合成的字符串,当然这里也可以写成string.lowercase,大部分情况下两者一样,但是有些特殊情况后者会有变化,同理还有string.ascii_uppercase和string.uppercase。其次,这里还用到了python中的切片,切片这个概念在看python的基本概念的时候有接触,但并没有机会用到,正好借此机会巩固一下。比如string.ascii_lowercase[2:]就是从string.ascii_lowercase[2]开始切片到字符串末尾,即从'c'到'z',而string.ascii_lowercase[:2]就是从字符串开头切片到第2位,即'ab',如果有个string.ascii_lowercase[3:6],那么就应该是'def'。
最后将map的翻译写进代码:
import string
'map'.translate(string.maketrans(string.ascii_lowercase, string.ascii_lowercase[2:] + string.ascii_lowercase[:2]))
得到了进入下一关的关键字:ocr。