当前位置:网站首页>LeetCode_Nov_5th_Week

LeetCode_Nov_5th_Week

2022-08-04 05:30:00 KuoGavin

November 29th : 786. 第 K 个最小的素数分数
December 1st : 1446. 连续字符
December 2nd : 506. 相对名次
December 3rd : 1005. K 次取反后最大化的数组和
December 4th : 383. 赎金信


November 29th : 786. 第 K 个最小的素数分数

class Solution {
    
public:
    vector<int> kthSmallestPrimeFraction(vector<int>& arr, int k) {
    
        int n = arr.size();
        auto cmp = [&](const pair<int, int>& x, const pair<int, int>& y) {
    
        	//分式进行交叉相乘,就可以得到比较关系如下:
            return arr[x.first] * arr[y.second] > arr[x.second] * arr[y.first];
        };
        priority_queue<pair<int, int>, vector<pair<int, int>>, decltype(cmp)> q(cmp);
		//初始化优先队列中的元素,将n-1个列表中的元素加入到优先队列中去 
        for (int j = 1; j < n; ++j) q.emplace(0, j);
		//从优先队列中依次取出最小值,并加入相应列表中的次小值,循环k次,优先队列的头即为所求
        for (int _ = 1; _ < k; ++_) {
    
            auto [i, j] = q.top();
            q.pop();
            if (i + 1 < j) q.emplace(i + 1, j);
        }
        return {
    arr[q.top().first], arr[q.top().second]};
    }
};
/********************************************************************************************/
//brutal force. 超时
class Solution {
    
public:
    vector<int> kthSmallestPrimeFraction(vector<int>& arr, int k) {
    
        unordered_map<float, vector<int>> mp;
        priority_queue<float, vector<float>, greater<float>> pq;
        for(int i = 0; i < arr.size(); ++i) {
    
            for(int j = i+1; j < arr.size(); ++j) {
    
                float frac = (float)arr[i] / (float)arr[j];
                pq.push(frac);
                mp[frac] = {
    arr[i], arr[j]};
            }
        }
        while(k-- != 1) {
    cout << pq.top() << endl; pq.pop();};
        return mp[pq.top()];
    }
};

December 1st : 1446. 连续字符

典型的滑动窗口问题,新加入窗口的字符与窗口中的字符是否一致,一致则窗口扩大,不一致则重置窗口。

class Solution {
    
public:
    int maxPower(string s) {
    
        int ret = 0, l = 0, r = 0;
        while(r < s.size()) {
    
            if(s[l] == s[r]) ret = max(ret, r-l+1);
            else l = r;
            r++;
        }
        return ret;
    }
};

December 2nd : 506. 相对名次

哈希表+排序,时间复杂度 O ( n l o g n ) O(nlogn) O(nlogn)

class Solution {
    
public:
    vector<string> findRelativeRanks(vector<int>& score) {
    
        unordered_map<int, int> mp;
        vector<string> vecStr(score.size());
        for(int i = 0; i < score.size(); ++i) mp[score[i]] = i;
        sort(score.begin(), score.end(), greater<int>());
        vecStr[mp[score[0]]] = "Gold Medal";
        if(score.size() > 1) vecStr[mp[score[1]]] = "Silver Medal";
        if(score.size() > 2) vecStr[mp[score[2]]] = "Bronze Medal";
        for(int i = 3; i < score.size(); ++i) vecStr[mp[score[i]]] = to_string(i+1);
        return vecStr;
    }
};

另开辟一个O(n+k)的数组,进行一次遍历,得到各得分在score数组中的下标,然后从左到右遍历一下另开辟数组,并根据下表对结果字符串数组进行修改。

class Solution {
    
public:
    vector<string> findRelativeRanks(vector<int>& score) {
    
        int max_score = *max_element(score.begin(), score.end()); //获取分数的最大值
        vector<int> index(max_score + 1, -1);  // 申请一个包含max+1个全是-1的数组
        // 把该元素对应的下标处的值置为在原始数组中的索引下标
        for(int i = 0; i < score.size(); i++) index[score[i]] = i; 

        vector<string> vecStr(score.size());//存放输出结果
        int rank = score.size(), j = 0; //rank排名, j是index数组的下标。
        while(rank > 0){
    
            if(index[j] != -1){
    
                if(rank == 1)      vecStr[index[j]] = "Gold Medal"; 
                else if(rank == 2) vecStr[index[j]] = "Silver Medal"; 
                else if(rank == 3) vecStr[index[j]] = "Bronze Medal"; 
                else vecStr[index[j]] = to_string(rank); 
                rank--;
            }
            j++;
        }
        return vecStr;
    }
};

December 3rd : 1005. K 次取反后最大化的数组和

依据数组中数字的绝对值对数组进行排序,然后将其中的非正数进行反转,若是非正数个数大于 k k k 个,那么余下的反转次数若是偶数则直接返回数组的和即可,若是奇数则将数组第一个元素/绝对值最小的元素进行反转再返回数组累加和即可。

//思想:能全部为非负数则全部变为非负数,
//如果全部变为非负数后还有剩余奇数次反转次数,
//则反转经过反转操作后最小的非负数再返回累加和
class Solution {
    
public:
    int largestSumAfterKNegations(vector<int>& nums, int k) {
    
        sort(nums.begin(), nums.end());
        for(int i = 0; i < nums.size() && k > 0; ++i) {
    
            if(nums[i] > 0) break;
            nums[i] = -nums[i];
            k--;
        }
        sort(nums.begin(), nums.end()); //获得新的数值排序,先前把绝对值较大的非正数进行反转
        if(k & 0x1) return -nums[0] + accumulate(nums.begin()+1, nums.end(), 0);
        else return accumulate(nums.begin(), nums.end(), 0);
    }
};
/*********************************************************************************************************/
//更直观一点的
class Solution {
    
    static bool cmp(int a, int b) {
    return abs(a) > abs(b);}
public:
    int largestSumAfterKNegations(vector<int>& nums, int k) {
    
        sort(nums.begin(), nums.end(), cmp);  //按照绝对值大小排序,大的在前 
        for (int i = 0; i < nums.size() && k > 0; ++i) {
     //遍历数组,将非负数反转
            if (nums[i] < 0) {
    
                nums[i] = -nums[i];
                k--;
            }
        }
        if (k & 0x1) nums[nums.size() - 1] *= -1; //若还有奇数次反转操作剩余,则反转绝对值最小的数
		return accumulate(nums.begin(), nums.end(), 0); //返回数组累加和
    }
};

December 4th : 383. 赎金信

统计 m a g a z i n e magazine magazine当中的字符的频次,然后对应 r a n s o m − n o t e ransom-note ransomnote中的字符的频次。若是遍历完整个 r a n s o m − n o t e ransom-note ransomnote,所有 m a g a z i n e magazine magazine中的字符频次均大于等于则 r a n s o m − n o t e ransom-note ransomnote的字符频次则返回true,否则返回false

class Solution {
    
public:
    bool canConstruct(string ransomNote, string magazine) {
    
        int dict[26] = {
    0};
        //vector<int> dict(26, 0);
        for(int i = 0; i < magazine.size(); ++i) dict[magazine[i]-'a']++;
        for(int i = 0; i < ransomNote.size(); ++i) {
    
            if(dict[ransomNote[i]-'a'] == 0) return false;
            dict[ransomNote[i]-'a']--;
        }
        return true;
    }
};

原网站

版权声明
本文为[KuoGavin]所创,转载请带上原文链接,感谢
https://blog.csdn.net/yueguangmuyu/article/details/121614329