当前位置:网站首页>LeetCode_Nov_3rd_Week

LeetCode_Nov_3rd_Week

2022-08-04 06:29:00 KuoGavin

November 15th : 319. 灯泡开关
November 16th : 391. 完美矩形
November 17th : 318. 最大单词长度乘积
November 18th : 563. 二叉树的坡度
November 19th : 397. 整数替换
November 20th : 594. 最长和谐子序列
November 21st : 559. N 叉树的最大深度


November 15th : 319. 灯泡开关

From the official solution of Likou,Go to sleep!

//version1 brutal force 根据1billion的数据量,The violent solution is destined to be just a pleasure,哈哈
//The brute force solution is to simulate the whole process of opening and closing the light bulb
class Solution {
    
public:
    int bulbSwitch(int n) {
    
        if(n < 2) return n;
        if(n == 2) return 1;
        vector<int> bulbs = vector<int>(n, 1);
        for(int i = n & 0x1 ? 1 : 0; i < n; i+=2) bulbs[i] = 0;
        for(int i = 3; i <= n; i++) 
            for(int j = 0; j < n; j+=i) 
                bulbs[j] = bulbs[j] == 1 ? 0 : 1;
        return accumulate(bulbs.begin(), bulbs.end(), 0);
    }
};
/********************************************************************************************/
//version2 math method
class Solution {
    
public:
    int bulbSwitch(int n) {
    
        //return sqrt(n);
        return sqrt(n+0.5);
    }
};

November 16th : 391. 完美矩形

两种方式,The first and more intuitive idea is as follows:

  • ① Count the area of ​​a small rectangle,Update the bottom left and top right four values;
  • ② Put the coordinates of the four points of the rectangle traversed each time into itset中,Duplicates are removed;
  • ③ 观察set是否只剩4个,And it is a combination of the four values ​​of the lower left and upper right;
  • ④ ③Satisfy and the sum of the areas of the small rectangles is equal to the four values ​​that make up the area of ​​the rectangle,Returns true otherwise false.

注意:对于pair,unordered_set There is no dedicated hash function,若是定义unordered_set<pair<,>>编译器会报错.

class Solution {
    
public:
    bool isRectangleCover(vector<vector<int>>& rectangles) {
    
        int sumArea = 0;
        int left = INT_MAX, up = INT_MIN;
        int right = INT_MIN, down = INT_MAX;
        set<pair<int, int>> s;
        for(auto& rectangle : rectangles) {
    
            left = min(left, rectangle[0]);
            right = max(right, rectangle[2]);
            up = max(up, rectangle[3]);
            down = min(down, rectangle[1]);

            sumArea += (rectangle[2]-rectangle[0]) * (rectangle[3]-rectangle[1]);

            pair<int, int> lu = make_pair(rectangle[0], rectangle[3]);
            if(!s.count(lu)) s.insert(lu); else s.erase(lu);
            pair<int, int> ld = make_pair(rectangle[0], rectangle[1]);
            if(!s.count(ld)) s.insert(ld); else s.erase(ld);
            pair<int, int> ru = make_pair(rectangle[2], rectangle[3]);
            if(!s.count(ru)) s.insert(ru); else s.erase(ru);
            pair<int, int> rd = make_pair(rectangle[2], rectangle[1]);
            if(!s.count(rd)) s.insert(rd); else s.erase(rd);
        }

        pair<int, int> lu = make_pair(left, up);
        pair<int, int> ld = make_pair(left, down);
        pair<int, int> ru = make_pair(right, up);
        pair<int, int> rd = make_pair(right, down);

        if(s.size() == 4 && s.count(lu) && s.count(ld) && s.count(ru) && s.count(rd))
            return sumArea == (up-down)*(right-left);

        return false;
    }
};

第二种思路,is in the solution,Use heap for edge matching of matrices,Somewhat like a puzzle,从左至右,Find matrix blocks that can be stitched from bottom to top,If there are no matrix blocks in the final heap, then it is a perfect matrix,否则就不是.The idea can be seen to solve the problem:图解 完美矩形 (扫描线).

class Solution {
    
public:
    bool isRectangleCover(vector<vector<int>>& rectangles) {
    
        priority_queue<tuple<int,int,int>, vector<tuple<int,int,int>>, greater<tuple<int,int,int>>> pq;
        // A stack of scanlines is recorded.The scanline contains the right border of the rectangle x、下边 y 和上边 y
        sort(rectangles.begin(), rectangles.end());//由左往右,自下而上排序
        int i = 0;// i Record the rectangle that has been scanned
        int len = rectangles.size();
        int nextX = rectangles[0][0];
        int nextY = INT_MIN;
        // Add the leftmost column of rectangles to the heap first,作为匹配的开始
        while(i<len && rectangles[i][0]==nextX){
    
            // Only if the upper edge of the previous rectangle is adjacent to the lower edge of the next rectangle,to fit a perfect rectangle
            if(nextY==INT_MIN || nextY==rectangles[i][1]){
    
                nextY = rectangles[i][3];
            }else{
    
                return false;
            }
            // Add rectangular scanlines,That is, the right border
            tuple<int,int,int> tp = {
    rectangles[i][2], rectangles[i][1], rectangles[i][3]};
            pq.push(tp);
            i++;
        }
        // 从左到右,从下到上,Start from the matched rectangle and advance to the right,Keep matching
        // Because they are all rectangles,So when the match is successful, it must also be a rectangle
        while(!pq.empty()){
    
            auto cur = pq.top();
            pq.pop();
            // Find one from the top of the pile x 相同、The longest continuous line segment,That is, the upper and lower sides meet
            int xStart = get<0>(cur);
            int yStart = get<1>(cur);
            int yEnd = get<2>(cur);
            while(!pq.empty() && xStart==get<0>(pq.top()) && yEnd==get<1>(pq.top())){
    
                yEnd = get<2>(pq.top());
                pq.pop();
            }
            // 从 rectangles[i:] Start finding continuous rectangles that perfectly match the line segments above,
            // That is, the line segment above x Rectangles that meet and are of equal height
            while(i<len && rectangles[i][0]==xStart && rectangles[i][1]==yStart){
    
                yStart = rectangles[i][3];
                pq.push({
    rectangles[i][2], rectangles[i][1], rectangles[i][3]});
                i++;
            }
            // If there is no perfect match,Either the match is wrong,Either the rightmost column has been matched
            if(yStart!=yEnd && (i<len || !pq.empty())){
    
                return false;
            }
        }
        return true;
    }
};

November 17th : 318. 最大单词长度乘积

At first I thought O ( n 2 ) O(n^2) O(n2) It was so violent that it took half an hour to figure out what ingenious could do it O ( n l o g n ) O(nlogn) O(nlogn) 的解法.So I thought about using bits to determine whether two strings have repeated letters,这样的话,整体的时间复杂度是 O ( n 2 ) + L O(n^2)+L O(n2)+L , L L L w o r d s words words 的长度.Of course, if you don't use bit judgment,时间复杂度就是 O ( m ∗ n 2 ) O(m*n^2) O(mn2),其中 m m m is the word length value,时间复杂度更高.I didn't pay attention to the size of the data at the time 1 0 3 10^3 103,想想 O ( n 2 ) O(n^2) O(n2) The time complexity is acceptable.

class Solution {
    
public:
    int maxProduct(vector<string>& words) {
    
        int n = words.size();
        //Since lowercase letters only26位,那么使用32位的intJust make a dictionary
        vector<int> dict = vector<int>(n, 0);
        for(int i = 0; i < n; ++i) {
     //创建每个word的32A dictionary of bit integers
            for(auto ch : words[i]) {
    
            	//左移ch-'a'位,Assign someone to1,使用或运算
                dict[i] |= 1 << (int)(ch - 'a');
            }
        }
        int ret = 0, rnd = 0;
        //O(n^2)将wordDictionary matches pairwise,If and result are0,Then the length product is not satisfied0,Update the length product
        for(int i = 0; i < n; ++i) {
     
            for(int j = i+1; j < n; ++j) {
    
                rnd = !(dict[i] & dict[j]) ? words[i].size()*words[j].size() : 0;
                ret = max(ret, rnd);
            }
        }           
        return ret;
    }
};

November 18th : 563. 二叉树的坡度

The idea is to find the node sum of the binary tree,And calculate the gradient at the appropriate time to calculate the gradient,有两种方式:
① Find the left and right subtrees of a binary tree node/Node and time of child nodes,Calculate the slope of the binary tree node,Recursively accumulate the slope of the entire binary tree;
② Find the node sum of all binary tree nodes in advance(使用哈希表存储,记忆化递归),Then recursively find the slope sum of the binary tree(The slope of the left child of the current node,The slope of the right child of the current node,The slope of the current node);

//version 1
class Solution {
    
public:
    int findTilt(TreeNode* root) {
    
        process(root);
        return ans;
    }
private:
    int ans;
    int process(TreeNode* root) {
    
        if(!root) return 0;
        int left = process(root->left);
        int right = process(root->right);
        ans += abs(left - right);
        return left + right + root->val;
    }
};
/********************************************************************************************/
//version 2
class Solution {
    
public:
    int findTilt(TreeNode* root) {
    
        if(!root) return 0;
        return abs(getSum(root->left)-getSum(root->right)) +
               findTilt(root->left) + findTilt(root->right);
    }

private:
    unordered_map<TreeNode*, int> rcd;
    int getSum(TreeNode* root) {
    
        if(!root) return 0;
        if(rcd.find(root) != rcd.end()) return rcd[root];
        int rootSum = 0;
        rootSum += getSum(root->left);
        rootSum += getSum(root->right);
        rootSum += root->val;
        rcd[root] = rootSum;
        return rootSum;
    }
};

November 19th : 397. 整数替换

三种方法,Simple no optimization and dfs,bfs,记忆化递归dfs,Memory iterationbfs,Then there is greed.

对于递归,It's just a simple enumeration from n n n 1 1 1 The possibility of all digital changes in the process:奇数 n = n + 1 n = n+1 n=n+1 n = n − 1 n = n-1 n=n1,偶数 n = n / 2 n = n/2 n=n/2;

对于贪心,Two operations are required for odd numbers( n n n 加减 1 1 1,再除于 2 2 2,也即 n n n The value changes slowly),所以把选择将 n n n in the bit 1 1 1 Reduce more greedy thoughts(/2How many times,a single operation n n n The reduced average value is larger).For the next highest order is 1 1 1 的奇数,则 + 1 +1 +1,减少 1 1 1 的个数,如果 3 3 3 Or the next highest non 1 1 1,则 − 1 -1 1,Also to reduce 1 1 1 的个数,Move the numbers in a decreasing direction.Three Leaf's Problem Solving:【宫水三叶】一题三解 :「DFS/BFS」&「贪心(位运算)」

//version 1 brutal force
class Solution {
    
public:
    int integerReplacement(int n) {
    
        minCnt = INT_MAX;
        dfs(n, 0);
        return minCnt;
    }
private:
    int minCnt;
    void dfs(long n, int cnt) {
    
        if(n == 1) {
    
            cout << cnt << endl;
            minCnt = min(cnt, minCnt);
            return;
        }
        if(n & 0x1) {
    
            dfs(n+1, cnt+1);
            dfs(n-1, cnt+1);
        } else {
    
            dfs(n/2, cnt+1);
        }
    }
};
/********************************************************************************************/
//version 2 Memorization recursion
class Solution {
    
public:
    int integerReplacement(int n) {
    
        if(n == 1) return 0;
        if(rcd.count(n)) return rcd[n];
        if(n & 0x1) rcd[n] = 2 + min(integerReplacement(n/2), integerReplacement(n/2+1));
        else rcd[n] = 1 + integerReplacement(n/2);
        return rcd[n];
    }
private:
    unordered_map<int, int> rcd;
};
/********************************************************************************************/
//version 3 greedy algorithm
class Solution {
    
public:
    int integerReplacement(int n) {
    
        if(n == 1) return 0;
        long _n = n;
        int ret = 0;
        while(_n != 1) {
    
            if(_n & 0x1) {
     //n为奇数
            	//The next highest is1且n此时不为3时,Adding more can reduce more in binary numbers1
                if(_n != 3 && (_n >> 1 & 0x1) == 1) _n++; 	
                //Otherwise the reduction can eliminate the least significant bit1
                else _n--;
            } else _n >>= 1; //Even numbers only/2的选择
            ret++; //This round of operation ends
        }
        return ret;
    }
};

November 20th : 594. 最长和谐子序列

① 先排序,Then use two pointers to solve the problem.The double pointer maintains a window of the same number,If the right pointer slides over the window,Then judge whether the difference between the original number of the window and the new number is yes 1 1 1,If yes, update the sequence length,And then consistent with the case of not,Pan the window until the entire array is traversed;

② 利用 m a p map map The bottom layer is the implementation of the red-black tree,Count the frequency of each number in the array,再先序遍历 m a p map map ,Get two adjacent numbers < k e y , v a l u e > <key,value> <key,value> 对,if adjacent k e y key key 值差 1 1 1,then update the sequence length;

//version 2 利用红黑树
class Solution {
    
public:
    int findLHS(vector<int>& nums) {
    
        map<int, int> mp;
        for(int num : nums) mp[num]++;
        auto iter = mp.begin(); 
        pair<int,int> pre = make_pair(iter->first, iter->second), cur;
        int ret = 0;
        ++iter;
        for(; iter != mp.end(); ++iter) {
    
            cur = make_pair(iter->first, iter->second);
            if(cur.first - pre.first == 1) ret = max(ret, cur.second + pre.second);
            pre = cur;
        }
        return ret;
    }
};

November 21st : 559. N 叉树的最大深度

Don't be fooled by examples when writing questions,Take a good look at the node definition of the topic:A polytree node contains one v e c t o r vector vector 数组,to store its child nodes,和二叉树相比,也就是将 l e f t left left r i g h t right right Put it in the set of child nodes,When finding the maximum depth of a binary tree, take the left and right larger depths and add them 1 1 1 返回,The analogy here is to add the maximum depth of all child nodes 1 1 1 返回.Apply the binary tree depth template,稍作调整即可.

当然,也可以使用 B F S BFS BFS,That is, how many levels of nodes are there in the multi-fork tree, B F S BFS BFSQueues are used as standard,这里就不再实现了.

class Solution {
    
public:
    int maxDepth(Node* root) {
    
        //if(!root) return 0;
        if(root == nullptr) return 0;
        int maxSubDepth = 0;
        for(auto& child : root->children) 
            maxSubDepth = max(maxSubDepth, maxDepth(child));
        return 1 + maxSubDepth;
    }
};
原网站

版权声明
本文为[KuoGavin]所创,转载请带上原文链接,感谢
https://yzsam.com/2022/216/202208040526103640.html