0843:猜猜这个单词(2077 分)
目录
题目
给你一个由 不同 字符串组成的单词列表 words
,其中 words[i]
长度均为 6
。words
中的一个单词将被选作秘密单词 secret
。
另给你一个辅助对象 Master
,你可以调用 Master.guess(word)
来猜单词,其中参数 word
长度为 6 且必须是 words
中的字符串。
Master.guess(word)
将会返回如下结果:
- 如果
word
不是words
中的字符串,返回-1
,或者 - 一个整数,表示你所猜测的单词
word
与 秘密单词secret
的准确匹配(值和位置同时匹配)的数目。
每组测试用例都会包含一个参数 allowedGuesses
,其中 allowedGuesses
是你可以调用 Master.guess(word)
的最大次数。
对于每组测试用例,在不超过允许猜测的次数的前提下,你应该调用 Master.guess
来猜出秘密单词。最终,你将会得到以下结果:
- 如果你调用
Master.guess
的次数大于allowedGuesses
所限定的次数或者你没有用Master.guess
猜到秘密单词,则得到"Either you took too many guesses, or you did not find the secret word."
。 - 如果你调用
Master.guess
猜到秘密单词,且调用Master.guess
的次数小于或等于allowedGuesses
,则得到"You guessed the secret word correctly."
。
生成的测试用例保证你可以利用某种合理的策略(而不是暴力)猜到秘密单词。
示例 1:
输入:secret = "acckzz", words = ["acckzz","ccbazz","eiowzz","abcczz"], allowedGuesses = 10 输出:You guessed the secret word correctly. 解释: master.guess("aaaaaa") 返回 -1 ,因为 "aaaaaa" 不在 words 中。 master.guess("acckzz") 返回 6 ,因为 "acckzz" 是秘密单词 secret ,共有 6 个字母匹配。 master.guess("ccbazz") 返回 3 ,因为 "ccbazz" 共有 3 个字母匹配。 master.guess("eiowzz") 返回 2 ,因为 "eiowzz" 共有 2 个字母匹配。 master.guess("abcczz") 返回 4 ,因为 "abcczz" 共有 4 个字母匹配。 一共调用 5 次 master.guess ,其中一个为秘密单词,所以通过测试用例。
示例 2:
输入:secret = "hamada", words = ["hamada","khaled"], allowedGuesses = 10 输出:You guessed the secret word correctly. 解释:共有 2 个单词,且其中一个为秘密单词,可以通过测试用例。
提示:
1 <= words.length <= 100
words[i].length == 6
words[i]
仅由小写英文字母组成words
中所有字符串 互不相同secret
存在于words
中10 <= allowedGuesses <= 30
分析
- 可以发现任何方法都不能保证在 10 次内一定猜出来
- 比如 [‘aaaaaa’, ‘bbbbbb’, … ‘zzzzzz’] 26 个单词的列表,必须要猜 26 次才能保证猜中
- 因此只考虑 x=guess(word) 是有用信息的情况,可以通过 x 缩小查找范围
- 比如示例 1 中,若选择了单词 word=‘ccbazz’,那么可以根据 guess(word)=3,更新 words 为与该单词匹配 3 项的单词列表 [‘acckzz’]
- 显然希望每一步得到的查找范围 cands 更小
- 先预处理与单词 u 匹配 x 的列表 f[u]
- 假如选择单词 u,guess 得到结果 x,查找范围变为 cands&f[u]
- 选择单词 u 的最坏结果即是 g=max(len(cands&f[u] ) for x in range(6))
- 遍历单词,选择 g 最小的单词即可
解答
|
|
71 ms