赎金信规则:为了不暴露赎金信字迹,要从杂志上搜索各个需要的字母,组成单词来表达意思。杂志字符串中的每个字符只能在赎金信字符串中使用一次。
一、题目描述
给你两个字符串:ransomNote 和 magazine ,判断 ransomNote 能不能由 magazine 里面的字符构成。
如果可以,返回 true ;否则返回 false 。
magazine 中的每个字符只能在 ransomNote 中使用一次。
二、示例与提示
示例 1:
输入: ransomNote = “a”, magazine = “b”
输出: false
示例 2:
输入: ransomNote = “aa”, magazine = “ab”
输出: false
示例 3:
输入: ransomNote = “aa”, magazine = “aab”
输出: true
提示
- 1 <= ransomNote.length, magazine.length <= 105
- ransomNote 和 magazine 由小写英文字母组成
三、思路
本题可理解为字符串a能否由另一字符串b组成,理解了题目,首先可能会想到本题需要用到哈希的思想,因为题目提示中说到本题两字符串皆由小写英文字母组成,且字符串b中的每一字母只可使用一次,所以我们可以采用空间换取时间的方法,定义一个长度为26的数组(对应26个英文字母),来记录b字符串中字母出现的次数,之后再用a字符串去对应消除在数组中记录b的各元素值,若数组中存在元素数值小于0,则表示a字符串无法用b字符串来组成。
四、代码
class Solution {
public:
bool canConstruct(string ransomNote, string magazine) {
int record[26] = {0};
// 因各元素只能使用一次,若ransomNote中的元素个数大于magazine中的,则ransomNote一定不能由magazine组成
if (ransomNote.size() > magazine.size()) {
return false;
}
for (int i = 0; i < magazine.length(); i++) {
// 首先通过record数组来记录magazine里各个字符出现次数
record[magazine[i]-'a'] ++;
}
for (int j = 0; j < ransomNote.length(); j++) {
// 之后遍历ransomNote,在record数组里使对应的字符个数做--操作
record[ransomNote[j]-'a']--;
// 如果record数组中存在元素值小于零则说明ransomNote里出现的字符,magazine中已没有
if(record[ransomNote[j]-'a'] < 0) {
return false;
}
}
return true;
}
};
复杂度分析
- 时间复杂度:O(n)
- 空间复杂度:O(1)