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处理异常:
def convert_number(s):
    try:
        return int(s)
    except ValueError:
        return none


把执行代码放到try代码块中,错误返回放到except代码块中,这样发生错误的时候就可以返回none。

在扫描器中你要使用这个函数来测试输入的是不是一个数字。

测试文件
把下面的代码写入tests/lexicon_tests.py文件中。
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行,你的呢?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值