LeetCode 题解工作台
加密解密字符串
给你一个字符数组 keys ,由若干 互不相同 的字符组成。还有一个字符串数组 values ,内含若干长度为 2 的字符串。另给你一个字符串数组 dictionary ,包含解密后所有允许的原字符串。请你设计并实现一个支持加密及解密下标从 0 开始字符串的数据结构。 字符串 加密 按下述步骤进行:…
5
题型
5
代码语言
3
相关题
当前训练重点
困难 · 数组·哈希·扫描
答案摘要
我们用一个哈希表 记录每个字符的加密结果,用另一个哈希表 记录每个加密结果出现的次数。 在构造函数中,我们遍历 和 ,将每个字符和其对应的加密结果存入 中。然后遍历 ,统计每个加密结果出现的次数。时间复杂度 $(n + m)$,其中 和 分别是 和 的长度。
Interview AiBoxInterview AiBox 实时 AI 助手,陪你讲清 数组·哈希·扫描 题型思路
题目描述
给你一个字符数组 keys ,由若干 互不相同 的字符组成。还有一个字符串数组 values ,内含若干长度为 2 的字符串。另给你一个字符串数组 dictionary ,包含解密后所有允许的原字符串。请你设计并实现一个支持加密及解密下标从 0 开始字符串的数据结构。
字符串 加密 按下述步骤进行:
- 对字符串中的每个字符
c,先从keys中找出满足keys[i] == c的下标i。 - 在字符串中,用
values[i]替换字符c。
请注意,如果 keys 中不存在字符串中的字符,则无法执行加密过程,返回空字符串 ""。
字符串 解密 按下述步骤进行:
- 将字符串每相邻 2 个字符划分为一个子字符串,对于每个子字符串
s,找出满足values[i] == s的一个下标i。如果存在多个有效的i,从中选择 任意 一个。这意味着一个字符串解密可能得到多个解密字符串。 - 在字符串中,用
keys[i]替换s。
实现 Encrypter 类:
Encrypter(char[] keys, String[] values, String[] dictionary)用keys、values和dictionary初始化Encrypter类。String encrypt(String word1)按上述加密过程完成对word1的加密,并返回加密后的字符串。int decrypt(String word2)统计并返回可以由word2解密得到且出现在dictionary中的字符串数目。
示例:
输入:
["Encrypter", "encrypt", "decrypt"]
[[['a', 'b', 'c', 'd'], ["ei", "zf", "ei", "am"], ["abcd", "acbd", "adbc", "badc", "dacb", "cadb", "cbda", "abad"]], ["abcd"], ["eizfeiam"]]
输出:
[null, "eizfeiam", 2]
解释:
Encrypter encrypter = new Encrypter([['a', 'b', 'c', 'd'], ["ei", "zf", "ei", "am"], ["abcd", "acbd", "adbc", "badc", "dacb", "cadb", "cbda", "abad"]);
encrypter.encrypt("abcd"); // 返回 "eizfeiam"。
// 'a' 映射为 "ei",'b' 映射为 "zf",'c' 映射为 "ei",'d' 映射为 "am"。
encrypter.decrypt("eizfeiam"); // return 2.
// "ei" 可以映射为 'a' 或 'c',"zf" 映射为 'b',"am" 映射为 'd'。
// 因此,解密后可以得到的字符串是 "abad","cbad","abcd" 和 "cbcd"。
// 其中 2 个字符串,"abad" 和 "abcd",在 dictionary 中出现,所以答案是 2 。
提示:
1 <= keys.length == values.length <= 26values[i].length == 21 <= dictionary.length <= 1001 <= dictionary[i].length <= 100- 所有
keys[i]和dictionary[i]互不相同 1 <= word1.length <= 20001 <= word2.length <= 200- 所有
word1[i]都出现在keys中 word2.length是偶数keys、values[i]、dictionary[i]、word1和word2只含小写英文字母- 至多调用
encrypt和decrypt总计200次
解题思路
方法一:哈希表
我们用一个哈希表 记录每个字符的加密结果,用另一个哈希表 记录每个加密结果出现的次数。
在构造函数中,我们遍历 和 ,将每个字符和其对应的加密结果存入 中。然后遍历 ,统计每个加密结果出现的次数。时间复杂度 ,其中 和 分别是 和 的长度。
在加密函数中,我们遍历输入字符串 的每个字符,查找其加密结果并拼接起来。如果某个字符没有对应的加密结果,说明无法加密,返回空字符串。时间复杂度 ,其中 是 的长度。
在解密函数中,我们直接返回 中 对应的次数。时间复杂度 。
空间复杂度 。
class Encrypter:
def __init__(self, keys: List[str], values: List[str], dictionary: List[str]):
self.mp = dict(zip(keys, values))
self.cnt = Counter(self.encrypt(v) for v in dictionary)
def encrypt(self, word1: str) -> str:
res = []
for c in word1:
if c not in self.mp:
return ''
res.append(self.mp[c])
return ''.join(res)
def decrypt(self, word2: str) -> int:
return self.cnt[word2]
# Your Encrypter object will be instantiated and called as such:
# obj = Encrypter(keys, values, dictionary)
# param_1 = obj.encrypt(word1)
# param_2 = obj.decrypt(word2)
复杂度分析
| 指标 | 值 |
|---|---|
| 时间 | Depends on the final approach |
| 空间 | Depends on the final approach |
面试官常问的追问
外企场景- question_mark
Assess the candidate's ability to handle encryption and decryption using hashmaps.
- question_mark
Test if the candidate understands the importance of efficient lookups for decryption and encryption.
- question_mark
Evaluate how well the candidate can manage complex string manipulations and check for dictionary matches.
常见陷阱
外企场景- error
Failing to handle cases where a character in the string doesn't exist in the 'keys' array, returning an empty string.
- error
Inefficient decryption algorithms that fail to quickly find valid strings in the 'dictionary'.
- error
Not properly handling the case where a string has multiple possible decrypted matches.
进阶变体
外企场景- arrow_right_alt
Increasing the length of the 'values' array beyond 2 characters, which requires adjustments to the segment length for decryption.
- arrow_right_alt
Handling encrypted strings that are extremely large or require optimizations in the lookup process.
- arrow_right_alt
Adding multiple dictionaries and handling scenarios with several potential valid decrypted strings for a single encrypted input.