После ответа на вопрос в SO о поиске города в пользовательском вопросе я начал думать о best способе поиска строки в тексте, когда у вас ограниченный набор данныхкак этот.
in
и find
совпадают с подстрокой, которая не нужна.Регулярные выражения, использующие «границы слов», работают, но работают довольно медленно.Подход «пунктуации» кажется подходящим, но есть много знаков препинания символов , которые могут появляться как в вопросе, так и в названии города (то есть точка в «Сент-Луисе»").
Регулярные выражения, вероятно, являются лучшим решением общего назначения, но мне любопытно, может ли это быть решено с использованием некоторой другой техники.
Задача состоит в том, чтобы:
Найти город в США по тексту, предоставленному пользователем на английском языке, независимо от регистра.
Мой код в значительной степени вдохновлен http://www.python.org/doc/essays/list2str/
#!/usr/bin/env python
import time
import re
def timing(f, n):
print f.__name__,
r = range(n)
t1 = time.clock()
for i in r:
f(); f(); f(); f(); f(); f(); f(); f(); f(); f()
t2 = time.clock()
print round(t2-t1, 6)
def f0():
'''broken since it finds sub-strings, e.g.
city "Erie" is found in "series"'''
Q = question.upper()
for c in cities:
c = c.upper()
if c in Q:
pass
def f1():
'''slow, but working'''
for c in cities:
re.search('\\b%s\\b' % c, question, re.IGNORECASE)
def f2():
'''broken, same problem as f0()'''
Q = question.upper()
for c in cities:
c = c.upper()
if Q.find(c) > 0:
pass
def f3():
'''remove all punctuation, and then search for " str " '''
Q = question.upper()
punct = ['.', ',', '(', ')', '"', '\n', ' ', ' ', ' ']
for p in punct:
Q = Q.replace(p, ' ')
for c in cities:
c = ' ' + c.upper() + ' '
for p in punct:
c = c.replace(p, ' ')
if c in Q:
pass
with open('cities') as fd:
cities = [line.strip() for line in fd]
with open('question') as fd:
question = fd.readlines()[0]
testfuncs = f0, f1, f2, f3
for f in testfuncs:
print f
timing(f, 20)
Вкл.мой старый изворотливый ноутбук, я получаю следующие результаты
<function f0 at 0xb7730bc4>
f0 0.14
<function f1 at 0xb7730f7c>
f1 10.4
<function f2 at 0xb7730f44>
f2 0.15
<function f3 at 0xb7738684>
f3 0.61
Если кто-то захочет проверить мои тестовые данные, его можно найти здесь