day14

[0001.两数之和]

  • 不会用 map。。。后面再看 先放着

C++代码:

class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        std::unordered_map <int, int> map;
        for(int i = 0; i < nums.size(); i++) {
            // 遍历当前元素,并在map中寻找是否有匹配的key
            auto iter = map.find(target - nums[i]); 
            if(iter != map.end()) {
                return {iter->second, i};
            }
            // 如果没找到匹配对,就把访问过的元素和下标加入到map中
            map.insert(pair<int, int>(nums[i], i)); 
        }
        return {};
    }
};

[0454.四数相加II]

C++代码:

class Solution {
public:
    int fourSumCount(vector<int>& A, vector<int>& B, vector<int>& C, vector<int>& D) {
        unordered_map<int, int> umap; //key:a+b的数值,value:a+b数值出现的次数
        // 遍历大A和大B数组,统计两个数组元素之和,和出现的次数,放到map中
        for (int a : A) {
            for (int b : B) {
                umap[a + b]++;
            }
        }
        int count = 0; // 统计a+b+c+d = 0 出现的次数
        // 在遍历大C和大D数组,找到如果 0-(c+d) 在map中出现过的话,就把map中key对应的value也就是出现次数统计出来。
        for (int c : C) {
            for (int d : D) {
                if (umap.find(0 - (c + d)) != umap.end()) {
                    count += umap[0 - (c + d)];
                }
            }
        }
        return count;
    }
};

  • 没思路 不会用 以后再做

[0383.赎金信]

class Solution {
public:
    bool canConstruct(string ransomNote, string magazine) {
        vector<int> hash (26 , 0);
        for (int i = 0; i < magazine.size(); i++) {
            hash[magazine[i] - 'a']++;
        }
        for (int i = 0; i <  ransomNote.size(); i++) {
            hash[ransomNote[i] - 'a']--;
        }
        for (int i = 0; i <  26; i++) {
            if (hash[i] < 0) {
                return false;
            }
        }
        return true;
    }
};
  • 本来以为得用 map set 想着抄一遍答案 下次再做算了 看到解答思路 说是用数组 并且本题和 有效的字母异位词很像 虽然不能一下反应上来那道题是怎么做得 貌似是 一个数组先去映射成hash表 ++操作 另一个数组在hash表里查找--操作

  • 本题有暴力解法 说实话 里面用到erase函数我还没太理解

  • 本题哈希法采用数组结构 最关键的思想是 ”因为题目所只有小写字母,那可以采用空间换取时间的哈希策略, 用一个长度为26的数组还记录magazine里字母出现的次数。

    然后再用ransomNote去验证这个数组是否包含了ransomNote所需要的所有字母。“

// 时间复杂度: O(n)
// 空间复杂度:O(1)
class Solution {
public:
    bool canConstruct(string ransomNote, string magazine) {
        int record[26] = {0};
        //add
        if (ransomNote.size() > magazine.size()) {
            return false;
        }
        for (int i = 0; i < magazine.length(); i++) {
            // 通过recode数据记录 magazine里各个字符出现次数
            record[magazine[i]-'a'] ++;
        }
        for (int j = 0; j < ransomNote.length(); j++) {
            // 遍历ransomNote,在record里对应的字符个数做--操作
            record[ransomNote[j]-'a']--;
            // 如果小于零说明ransomNote里出现的字符,magazine没有
            if(record[ransomNote[j]-'a'] < 0) {
                return false;
            }
        }
        return true;
    }
};
  • 首先 标准答案比我好在最开始的判断语句 如果救赎金比杂志长度大 那么明显就是false
  • 其次 标准答案 把验证--操作 和验证是否<0 操做同时在一个for循环里进行 确实简洁 因为 反正不会再++ 那么只要 一旦<0 就可以返回false 而不需要等到所有--操作完成之后再判断每一个元素值是否小于0
posted @ 2022-11-08 20:03  跬步瑶  阅读(25)  评论(0)    收藏  举报