Learn Python The Hard Way学习(48) - 更高级的用户输入
你的游戏可能已经做得不错了,但是用户输入很麻烦,每次都输入完全正确的字符才能执行命令,我们需要一个东西让用户可以输入不同的短语就可以执行命令,比如,你想要下面这些词语都能执行:
open door
open the door
go THROUGH the door
punch bear
Punch The Bear in the FACE
你的程序要能识别出用户的输入,并且知道它是什么意思,为了达到目的,我们需要写一个模块来完成这个任务,这个模块有一些类用来处理用户的输入,然后执行对应的代码。
简单的英文包括下面这些元素:
用空格区分词语
句子由单词组成
语法结构决定句子的意思
所以,最好的办法是得到用户输入的词汇,然后确定词汇的意思。
我们的游戏词典
建立一个我们的游戏词典:
方向:north, south, east, west, down, up, left, right, back.
动作:go, stop, kill eat.
禁用词:the, in, of, from, at, it
名称:door, bear, princess, cabinet.
数字:0-9
说到名词,不同的房间可能名词不一样,不过我们先这样,以后慢慢改进。
分析一个句子
有了字典后,我们需要一个方法来断句,下面这个方法可以完成任务:
stuff = raw_input('> ')
words = stuff.split()
字典元组
有了词语,我们就需要知道这个词语的类型,下面我们使用python的一个结构叫元组,元组其实就是一个不能修改的列表,用括号包含,逗号区分:
first_word = ('dirction', 'north')
second_word = ('verb', 'go')
sentence = [first_word, second_word]
我们创建了一组(TYPE, WORD),让你识别单词。
这只是一个例子,不过基本就是这样,你接受用户输入,使用split拆分句子,区分单词类型,然后重新组合。
扫描输入
现在开始写你的扫描器吧,这个扫描器取得用户输入,然后返回一个包括元组(TOKEN, WORD)的列表。如果不存在这样一个单词,还是应该返回WORD,但是TOKEN应该返回错误信息。
我不会告诉你具体的代码,而是写一个测试,你要确保测试能通过。
异常和数字
我会先帮你一个小忙,关于数字转换,我们使用异常,比如:
Python 2.6.5 (r265:79063, Apr 16 2010, 13:09:56)
[GCC 4.4.3] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> int("hell")
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: invalid literal for int() with base 10: 'hell'
>>>
这个ValueError是int函数的异常,因为int需要一个数字参数。本来int()可以返回一个-1表示输入错误,但是-1也是整数,所以只能返回ValueError异常。
我们下面用try和except处理异常:
[python]
def convert_number(s):
try:
return int(s)
except ValueError:
return none
把执行代码放到try代码块中,错误返回放到except代码块中,这样发生错误的时候就可以返回none。
在扫描器中你要使用这个函数来测试输入的是不是一个数字。
测试文件 www.2cto.com
把下面的代码写入tests/lexicon_tests.py文件中。
[python]
from nose.tools import *
from ex48 import lexicon
def test_directions():
assert_equal(lexicon.scan("north"), [('direction', 'north')])
result = lexicon.scan("north south east")
assert_equal(result, [('direction', 'north'),
('direction', 'south'),
('direction', 'east')])
def test_verbs():
assert_equal(lexicon.scan("go"), [('verb', 'go')])
result = lexicon.scan("go kill eat")
assert_equal(result, [('verb', 'go'),
('verb', 'kill'),
('verb', 'eat')])
def test_stops():
assert_equal(lexicon.scan("the"), [('stop', 'the')])
result = lexicon.scan("the in of")
assert_equal(result, [('stop', 'the'),
('stop', 'in'),
('stop', 'of')])
def test_nouns():
assert_equal(lexicon.scan("bear"), [('noun', 'bear')])
result = lexicon.scan("bear princess")
assert_equal(result, [('noun', 'bear'),
('noun', 'princess')])
def test_numbers():
assert_equal(lexicon.scan("1234"), [('number', 1234)])
result = lexicon.scan("3 91234")
assert_equal(result, [('number', 3),
('number', 91234)])
def test_errors():
assert_equal(lexicon.scan("ASDFADFASDF"), [('error', 'ASDFADFASDF')])
result = lexicon.scan("bear IAS princess")
assert_equal(result, [('noun', 'bear'),
('error', 'IAS'),
('noun', 'princess')])
记得新建一个项目来完成这些代码。
设计提示
集中实现一个测试工作。尽量简单的把所有的单词放到lexicon列表中。不要修改输入列表,你要做的是创建你自己的元组。使用in来判断单词是否在字典中。
加分练习
1. 改进测试用例,让它可以判断更多单词。
2. 添加字典,更新测试用例。
3. 让你的扫描器能识别大写,更新测试用例。
4. 寻找其他数字转换的方法。
5. 我的代码只有37行,你的呢?
作者:lixiang0522