給定一個模式和一個字符串str,找到str是否遵循相同的模式.
以下是完全匹配,使得在模式中的字母和str中的非空字之間存在雙射.
例子:
pattern =“abba”,str =“dog cat cat dog”應該返回true;狗是一只貓,貓是b,而這些詞形成了abba模式.
pattern =“abba”,str =“dog cat cat fish”應該返回false;字符串遵循abbc模式.
我的解決方案適用于Python 2:
def wordPattern(self, pattern, str):
s = pattern
t = str.split()
return map(s.find, s) == map(t.index, t)
但我只是想知道為什么這個解決方案不適用于Python 3.在嘗試測試上面的例子時,該函數將始終返回False.有人可以請一些建議嗎?
最佳答案 在Python 3中,map()返回一個迭代器對象,而不是一個列表.這些對象之間的等式測試將不起作用(相等性測試身份,而不是內存中的完全相同的對象).
明確轉換為列表:
def wordPattern(self, pattern, str):
s = pattern
t = str.split()
return list(map(s.find, s)) == list(map(t.index, t))
或使用列表推導:
def wordPattern(self, pattern, str):
s = pattern
t = str.split()
return [s.find(c) for c in s] == [t.index(w) for w in t]
或者通過將壓縮結果與all() function進行比較來避免完全創建列表:
from operator import eq
from itertools import starmap, zip_longest
def wordPattern(self, pattern, str):
s = pattern
t = str.split()
return all(starmap(eq, zip_longest(map(s.find, s), map(t.index, t))))
如果沒有匹配,后者短路而不必進行所有比較.本著保持功能風格的精神,我使用itertools.starmap()來測試與operator.eq() function的相等性.通過使用itertools.zip_longest(),我們確保我們可以檢測到模式長度和字數不匹配的情況.