[英]Python: how to determine if a list of words exist in a string
给定一个列表["one", "two", "three"]
,如何确定每个单词是否存在于指定的字符串中?
单词列表很短(在我的例子中不到 20 个单词),但要搜索的字符串非常大(每次运行 400,000 个字符串)
我当前的实现使用re
来查找匹配项,但我不确定这是否是最好的方法。
import re
word_list = ["one", "two", "three"]
regex_string = "(?<=\W)(%s)(?=\W)" % "|".join(word_list)
finder = re.compile(regex_string)
string_to_be_searched = "one two three"
results = finder.findall(" %s " % string_to_be_searched)
result_set = set(results)
for word in word_list:
if word in result_set:
print("%s in string" % word)
我的解决方案中的问题:
可能的更简单的实现:
if word in string_to_be_searched
执行if word in string_to_be_searched
。 但是如果你要找“三人行”,它就不能处理“三人行”更新:
我已经接受了 Aaron Hall 的回答https://stackoverflow.com/a/21718896/683321,因为根据 Peter Gibson 的基准https://stackoverflow.com/a/21742190/683321,这个简单版本的性能最好。 如果您对这个问题感兴趣,可以阅读所有答案并获得更好的视图。
实际上我忘了在我原来的问题中提到另一个约束。 单词可以是短语,例如: word_list = ["one day", "second day"]
。 也许我应该问另一个问题。
Peter Gibson(下文)发现此函数是此处答案中性能最高的。 这对于可能保存在内存中的数据集很有用(因为它从要搜索的字符串中创建了一个单词列表,然后是一组这些单词):
def words_in_string(word_list, a_string):
return set(word_list).intersection(a_string.split())
用法:
my_word_list = ['one', 'two', 'three']
a_string = 'one two three'
if words_in_string(my_word_list, a_string):
print('One or more words found!')
其中打印One or words found!
到标准输出。
它确实返回找到的实际单词:
for word in words_in_string(my_word_list, a_string):
print(word)
打印出来:
three
two
one
为了满足我自己的好奇心,我对发布的解决方案进行了计时。 结果如下:
TESTING: words_in_str_peter_gibson 0.207071995735
TESTING: words_in_str_devnull 0.55300579071
TESTING: words_in_str_perreal 0.159866499901
TESTING: words_in_str_mie Test #1 invalid result: None
TESTING: words_in_str_adsmith 0.11831510067
TESTING: words_in_str_gnibbler 0.175446796417
TESTING: words_in_string_aaron_hall 0.0834425926208
TESTING: words_in_string_aaron_hall2 0.0266295194626
TESTING: words_in_str_john_pirie <does not complete>
有趣的是@AaronHall 的解决方案
def words_in_string(word_list, a_string):
return set(a_list).intersection(a_string.split())
这是最快的,也是最短的之一! 请注意,它不处理单词旁边的标点符号,但从问题中不清楚这是否是一项要求。 @MIE 和@user3 也建议了此解决方案。
我没有看很长时间为什么两个解决方案不起作用。 如果这是我的错误,请道歉。 这是测试的代码,欢迎评论和更正
from __future__ import print_function
import re
import string
import random
words = ['one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'ten']
def random_words(length):
letters = ''.join(set(string.ascii_lowercase) - set(''.join(words))) + ' '
return ''.join(random.choice(letters) for i in range(int(length)))
LENGTH = 400000
RANDOM_STR = random_words(LENGTH/100) * 100
TESTS = (
(RANDOM_STR + ' one two three', (
['one', 'two', 'three'],
set(['one', 'two', 'three']),
False,
[True] * 3 + [False] * 7,
{'one': True, 'two': True, 'three': True, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
(RANDOM_STR + ' one two three four five six seven eight nine ten', (
['one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'ten'],
set(['one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine', 'ten']),
True,
[True] * 10,
{'one': True, 'two': True, 'three': True, 'four': True, 'five': True, 'six': True,
'seven': True, 'eight': True, 'nine': True, 'ten':True}
)),
('one two three ' + RANDOM_STR, (
['one', 'two', 'three'],
set(['one', 'two', 'three']),
False,
[True] * 3 + [False] * 7,
{'one': True, 'two': True, 'three': True, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
(RANDOM_STR, (
[],
set(),
False,
[False] * 10,
{'one': False, 'two': False, 'three': False, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
(RANDOM_STR + ' one two three ' + RANDOM_STR, (
['one', 'two', 'three'],
set(['one', 'two', 'three']),
False,
[True] * 3 + [False] * 7,
{'one': True, 'two': True, 'three': True, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
('one ' + RANDOM_STR + ' two ' + RANDOM_STR + ' three', (
['one', 'two', 'three'],
set(['one', 'two', 'three']),
False,
[True] * 3 + [False] * 7,
{'one': True, 'two': True, 'three': True, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
('one ' + RANDOM_STR + ' two ' + RANDOM_STR + ' threesome', (
['one', 'two'],
set(['one', 'two']),
False,
[True] * 2 + [False] * 8,
{'one': True, 'two': True, 'three': False, 'four': False, 'five': False, 'six': False,
'seven': False, 'eight': False, 'nine': False, 'ten':False}
)),
)
def words_in_str_peter_gibson(words, s):
words = words[:]
found = []
for match in re.finditer('\w+', s):
word = match.group()
if word in words:
found.append(word)
words.remove(word)
if len(words) == 0: break
return found
def words_in_str_devnull(word_list, inp_str1):
return dict((word, bool(re.search(r'\b{}\b'.format(re.escape(word)), inp_str1))) for word in word_list)
def words_in_str_perreal(wl, s):
i, swl, strwords = 0, sorted(wl), sorted(s.split())
for w in swl:
while strwords[i] < w:
i += 1
if i >= len(strwords): return False
if w != strwords[i]: return False
return True
def words_in_str_mie(search_list, string):
lower_string=string.lower()
if ' ' in lower_string:
result=filter(lambda x:' '+x.lower()+' ' in lower_string,search_list)
substr=lower_string[:lower_string.find(' ')]
if substr in search_list and substr not in result:
result+=substr
substr=lower_string[lower_string.rfind(' ')+1:]
if substr in search_list and substr not in result:
result+=substr
else:
if lower_string in search_list:
result=[lower_string]
def words_in_str_john_pirie(word_list, to_be_searched):
for word in word_list:
found = False
while not found:
offset = 0
# Regex is expensive; use find
index = to_be_searched.find(word, offset)
if index < 0:
# Not found
break
if index > 0 and to_be_searched[index - 1] != " ":
# Found, but substring of a larger word; search rest of string beyond
offset = index + len(word)
continue
if index + len(word) < len(to_be_searched) \
and to_be_searched[index + len(word)] != " ":
# Found, but substring of larger word; search rest of string beyond
offset = index + len(word)
continue
# Found exact word match
found = True
return found
def words_in_str_gnibbler(words, string_to_be_searched):
word_set = set(words)
found = []
for match in re.finditer(r"\w+", string_to_be_searched):
w = match.group()
if w in word_set:
word_set.remove(w)
found.append(w)
return found
def words_in_str_adsmith(search_list, big_long_string):
counter = 0
for word in big_long_string.split(" "):
if word in search_list: counter += 1
if counter == len(search_list): return True
return False
def words_in_string_aaron_hall(word_list, a_string):
def words_in_string(word_list, a_string):
'''return iterator of words in string as they are found'''
word_set = set(word_list)
pattern = r'\b({0})\b'.format('|'.join(word_list))
for found_word in re.finditer(pattern, a_string):
word = found_word.group(0)
if word in word_set:
word_set.discard(word)
yield word
if not word_set:
raise StopIteration
return list(words_in_string(word_list, a_string))
def words_in_string_aaron_hall2(word_list, a_string):
return set(word_list).intersection(a_string.split())
ALGORITHMS = (
words_in_str_peter_gibson,
words_in_str_devnull,
words_in_str_perreal,
words_in_str_mie,
words_in_str_adsmith,
words_in_str_gnibbler,
words_in_string_aaron_hall,
words_in_string_aaron_hall2,
words_in_str_john_pirie,
)
def test(alg):
for i, (s, possible_results) in enumerate(TESTS):
result = alg(words, s)
assert result in possible_results, \
'Test #%d invalid result: %s ' % (i+1, repr(result))
COUNT = 10
if __name__ == '__main__':
import timeit
for alg in ALGORITHMS:
print('TESTING:', alg.__name__, end='\t\t')
try:
print(timeit.timeit(lambda: test(alg), number=COUNT)/COUNT)
except Exception as e:
print(e)
简单的方法:
filter(lambda x:x in string,search_list)
如果您希望搜索忽略字符的大小写,您可以这样做:
lower_string=string.lower()
filter(lambda x:x.lower() in lower_string,search_list)
如果您想忽略属于较大单词的单词,例如三合一:
lower_string=string.lower()
result=[]
if ' ' in lower_string:
result=filter(lambda x:' '+x.lower()+' ' in lower_string,search_list)
substr=lower_string[:lower_string.find(' ')]
if substr in search_list and substr not in result:
result+=[substr]
substr=lower_string[lower_string.rfind(' ')+1:]
if substr in search_list and substr not in result:
result+=[substr]
else:
if lower_string in search_list:
result=[lower_string]
arr=string.split(' ') result=list(set(arr).intersection(set(search_list)))
编辑:在一个包含 400,000 个单词的字符串中搜索 1,000 个单词的示例中,此方法是最快的,但如果我们将字符串增加到 4,000,000,则前一种方法更快。
def safe_remove(arr,elem): try: arr.remove(elem) except: pass not_found=search_list[:] i=string.find(' ') j=string.find(' ',i+1) safe_remove(not_found,string[:i]) while j!=-1: safe_remove(not_found,string[i+1:j]) i,j=j,string.find(' ',j+1) safe_remove(not_found,string[i+1:])
not_found
列表包含未找到的单词,您可以轻松获取找到的列表,一种方法是list(set(search_list)-set(not_found))
编辑:最后一种方法似乎是最慢的。
def words_in_str(s, wl):
i, swl, strwords = 0, sorted(wl), sorted(s.split())
for w in swl:
while strwords[i] < w:
i += 1
if i >= len(strwords): return False
if w != strwords[i]: return False
return True
你可以试试这个:
list(set(s.split()).intersection(set(w)))
它仅从您的单词列表中返回匹配的单词。 如果没有匹配的单词,它将返回空列表。
如果您的字符串很长而您的搜索列表很短,请执行以下操作:
def search_string(big_long_string,search_list)
counter = 0
for word in big_long_string.split(" "):
if word in search_list: counter += 1
if counter == len(search_list): return True
return False
您可以使用单词边界:
>>> import re
>>> word_list = ["one", "two", "three"]
>>> inp_str = "This line not only contains one and two, but also three"
>>> if all(re.search(r'\b{}\b'.format(re.escape(word)), inp_str) for word in word_list):
... print "Found all words in the list"
...
Found all words in the list
>>> inp_str = "This line not only contains one and two, but also threesome"
>>> if all(re.search(r'\b{}\b'.format(re.escape(word)), inp_str) for word in word_list):
... print "Found all words in the list"
...
>>> inp_str = "This line not only contains one and two, but also four"
>>> if all(re.search(r'\b{}\b'.format(re.escape(word)), inp_str) for word in word_list):
... print "Found all words in the list"
...
>>>
编辑:如您的评论所示,您似乎正在寻找字典:
>>> dict((word, bool(re.search(r'\b{}\b'.format(re.escape(word)), inp_str1))) for word in word_list)
{'three': True, 'two': True, 'one': True}
>>> dict((word, bool(re.search(r'\b{}\b'.format(re.escape(word)), inp_str2))) for word in word_list)
{'three': False, 'two': True, 'one': True}
>>> dict((word, bool(re.search(r'\b{}\b'.format(re.escape(word)), inp_str3))) for word in word_list)
{'three': False, 'two': True, 'one': True}
如果顺序不太重要,可以使用这种方法
word_set = {"one", "two", "three"}
string_to_be_searched = "one two three"
for w in string_to_be_searched.split():
if w in word_set:
print("%s in string" % w)
word_set.remove(w)
.split()
创建一个列表,这对于您的 400k 字串可能是一个问题。 但是如果你有足够的内存,你就完成了。
当然可以修改 for 循环以避免创建整个列表。 re.finditer
或使用str.find
的生成器是显而易见的选择
import re
word_set = {"one", "two", "three"}
string_to_be_searched = "one two three"
for match in re.finditer(r"\w+", string_to_be_searched):
w = match.group()
if w in word_set:
print("%s in string" % w)
word_set.remove(w)
鉴于你的评论
我实际上并不是在寻找单个 bool 值,而是在寻找将单词映射到 bool 的字典。 此外,我可能需要运行一些测试并查看多次运行 re.search 并运行一次 re.findall 的性能。 – 耶格尔
我会提出以下建议
import re
words = ['one', 'two', 'three']
def words_in_str(words, s):
words = words[:]
found = []
for match in re.finditer('\w+', s):
word = match.group()
if word in words:
found.append(word)
words.remove(word)
if len(words) == 0: break
return found
assert words_in_str(words, 'three two one') == ['three', 'two', 'one']
assert words_in_str(words, 'one two. threesome') == ['one', 'two']
assert words_in_str(words, 'nothing of interest here one1') == []
这将返回按顺序找到的单词列表,但您可以轻松修改它以根据需要返回dict{word:bool}
。
好处:
这是一个简单的生成器,它更适合大字符串或文件,因为我在下面的部分对其进行了调整。
请注意,这应该非常快,但只要字符串继续而不击中所有单词,它就会继续。 这在 Peter Gibson 的基准测试中排名第二: Python:如何确定字符串中是否存在单词列表
有关较短字符串的更快解决方案,请参阅我的其他答案: Python:如何确定字符串中是否存在单词列表
import re
def words_in_string(word_list, a_string):
'''return iterator of words in string as they are found'''
word_set = set(word_list)
pattern = r'\b({0})\b'.format('|'.join(word_list))
for found_word in re.finditer(pattern, a_string):
word = found_word.group(0)
if word in word_set:
word_set.discard(word)
yield word
if not word_set: # then we've found all words
# break out of generator, closing file
raise StopIteration
它在找到单词时遍历字符串,在找到所有单词后或到达字符串末尾时放弃搜索。
用法:
word_list = ['word', 'foo', 'bar']
a_string = 'A very pleasant word to you.'
for word in words_in_string(word_list, a_string):
print word
word
感谢 Peter Gibson 发现这是第二快的方法。 我为解决方案感到非常自豪。 由于最好的用例是通过一个巨大的文本流,让我在这里调整上面的函数来处理一个文件。 请注意,如果换行符上的单词被破坏,这不会捕获它们,但这里的任何其他方法也不会。
import re
def words_in_file(word_list, a_file_path):
'''
return a memory friendly iterator of words as they are found
in a file.
'''
word_set = set(word_list)
pattern = r'\b({0})\b'.format('|'.join(word_list))
with open(a_file_path, 'rU') as a_file:
for line in a_file:
for found_word in re.finditer(pattern, line):
word = found_word.group(0)
if word in word_set:
word_set.discard(word)
yield word
if not word_set: # then we've found all words
# break out of generator, closing file
raise StopIteration
为了演示,让我们写一些数据:
file_path = '/temp/temp/foo.txt'
with open(file_path, 'w') as f:
f.write('this\nis\nimportant\ndata')
和用法:
word_list = ['this', 'is', 'important']
iterator = words_in_file(word_list, file_path)
我们现在有一个迭代器,如果我们用一个列表来消费它:
list(iterator)
它返回:
['this', 'is', 'important']
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.