当前位置:网站首页>LeetCode_Nov_3rd_Week
LeetCode_Nov_3rd_Week
2022-08-04 05:30: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. 灯泡开关
源自力扣官方题解,完事睡觉!
//version1 brutal force 根据1billion的数据量,暴力解法注定只是图一乐,哈哈
//暴力解法也就是把整个灯泡开闭的过程模拟一遍
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. 完美矩形
两种方式,第一种也是比较直观的思路如下:
- ① 统计小矩形面积,更新左下右上四个值;
- ② 把每次遍历到的矩形四点坐标放入set中,重复的就去除;
- ③ 观察set是否只剩4个,且是左下右上四个值的组合;
- ④ ③满足且小矩形面积之和等于四个值组成矩形面积,返回真否则假。
注意:对于pair,unordered_set 没有专门的哈希函数,若是定义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;
}
};
第二种思路,是题解当中的,使用堆进行矩阵的边匹配,有些类似拼图的过程,从左至右,从下至上寻找可以拼接的矩阵块,若是最终堆中没有矩阵块的话那么就是完美矩阵,否则就不是。思路可见力扣题解:图解 完美矩形 (扫描线)。
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;
// 记录扫描线的堆。扫描线包含矩形的右边界的 x、下边 y 和上边 y
sort(rectangles.begin(), rectangles.end());//由左往右,自下而上排序
int i = 0;// i 记录已经扫描的矩形
int len = rectangles.size();
int nextX = rectangles[0][0];
int nextY = INT_MIN;
// 先将最左边一列的矩形加到堆中,作为匹配的开始
while(i<len && rectangles[i][0]==nextX){
// 只有当上一个矩形的上边和下一个矩形的下边相邻时,才符合完美矩形
if(nextY==INT_MIN || nextY==rectangles[i][1]){
nextY = rectangles[i][3];
}else{
return false;
}
// 添加矩形的扫描线,也就是右边界
tuple<int,int,int> tp = {
rectangles[i][2], rectangles[i][1], rectangles[i][3]};
pq.push(tp);
i++;
}
// 从左到右,从下到上,从已经匹配的矩形开始往右推进,不断进行匹配
// 因为都是矩形,所以当匹配成功时也一定是矩形
while(!pq.empty()){
auto cur = pq.top();
pq.pop();
// 从堆顶找一条 x 相同、连续的最长的线段,也就是上下边相接
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:] 开始找到与上面线段完美匹配的连续矩形,
// 也就是和上面线段 x 相接并且高度相等的矩形
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(yStart!=yEnd && (i<len || !pq.empty())){
return false;
}
}
return true;
}
};
November 17th : 318. 最大单词长度乘积
最开始我觉得 O ( n 2 ) O(n^2) O(n2) 是暴力解以至于花了半个小时去想有什么精妙的能做到 O ( n l o g n ) O(nlogn) O(nlogn) 的解法。于是想着使用位来判定两个字符串是否有重复字母,这样的话,整体的时间复杂度是 O ( n 2 ) + L O(n^2)+L O(n2)+L , L L L 是 w o r d s words words 的长度。当然不过不使用位判定的话,时间复杂度就是 O ( m ∗ n 2 ) O(m*n^2) O(m∗n2),其中 m m m 是单词长度值,时间复杂度更高。当时没注意看数据规模是 1 0 3 10^3 103,想想 O ( n 2 ) O(n^2) O(n2) 的时间复杂度是可以接受的。
class Solution {
public:
int maxProduct(vector<string>& words) {
int n = words.size();
//由于小写字母只有26位,那么使用32位的int做字典即可
vector<int> dict = vector<int>(n, 0);
for(int i = 0; i < n; ++i) {
//创建每个word的32位整数字典
for(auto ch : words[i]) {
//左移ch-'a'位,将某位赋为1,使用或运算
dict[i] |= 1 << (int)(ch - 'a');
}
}
int ret = 0, rnd = 0;
//O(n^2)将word字典两两匹配,如果与结果是0,则满足长度乘积不为0,进行长度乘积的更新
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. 二叉树的坡度
思路就是求二叉树的节点和,并在合适的计算坡度的时机进行坡度的计算,有两种方式:
① 在求得二叉树节点的左右子树/子节点的节点和时,计算出该二叉树节点的坡度,递归累加得到整颗二叉树的坡度;
② 预先求出所有二叉树节点的节点和(使用哈希表存储,记忆化递归),然后按照递归方式求二叉树的坡度和(当前节点左子节点的坡度,当前节点右子节点的坡度,当前节点的坡度);
//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. 整数替换
三种方法,简单的无优化和dfs,bfs,记忆化递归dfs,记忆化迭代bfs,再者就是贪心。
对于递归,就是简单枚举从 n n n 到 1 1 1 的过程中的所有数字变化的可能性:奇数 n = n + 1 n = n+1 n=n+1 或 n = n − 1 n = n-1 n=n−1,偶数 n = n / 2 n = n/2 n=n/2;
对于贪心,对于奇数的需要两次操作( n n n 加减 1 1 1,再除于 2 2 2,也即 n n n 值变化慢),所以把选择将 n n n 的位中的 1 1 1 减少多的为贪心的思想(/2的次数若多,则单次操作 n n n 减小的平均数值更大)。对于次高位为 1 1 1 的奇数,则 + 1 +1 +1,减少 1 1 1 的个数,如果 3 3 3 或是次高位非 1 1 1,则 − 1 -1 −1,也为了减少 1 1 1 的个数,将数往减小的方向去。三叶的题解:【宫水三叶】一题三解 :「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为奇数
//当次高位为1且n此时不为3时,增加更能减少更多的二进制数中的1
if(_n != 3 && (_n >> 1 & 0x1) == 1) _n++;
//否则减少能消除最低位上的1
else _n--;
} else _n >>= 1; //偶数就只有/2的选择
ret++; //本轮操作结束
}
return ret;
}
};
November 20th : 594. 最长和谐子序列
① 先排序,再使用双指针对问题进行求解。双指针维护的是同一个数字的窗口,若是右指针滑过窗口,则判断窗口原先数字和新的数字的差值是否是 1 1 1,是则更新序列长度,然后和不是的情况一致,平移窗口直至遍历整个数组;
② 利用 m a p map map 底层是红黑树的实现,将数组中的每个数进行频数统计,再先序遍历 m a p map map ,得到相邻两数的 < k e y , v a l u e > <key,value> <key,value> 对,若是相邻的 k e y key key 值差 1 1 1,则更新序列长度;
//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 叉树的最大深度
写题的时候不要被示例迷惑,好好读一下题目的节点定义:多叉树节点中包含一个 v e c t o r vector vector 数组,用以存放它的子节点,和二叉树相比,也就是将 l e f t left left 和 r i g h t right right 放到了子节点集合中去,求二叉树最大深度时是取左右较大深度加 1 1 1 返回,这里类比就是所有子节点最大深度加 1 1 1 返回。套用二叉树深度模板,稍作调整即可。
当然,也可以使用 B F S BFS BFS,也即是求多叉树总共有多少层节点, B F S BFS BFS标配使用队列,这里就不再实现了。
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;
}
};
边栏推荐
- Data reading in yolov3 (1)
- 语音驱动嘴型与面部动画生成的现状和趋势
- 深度学习,“粮草”先行--浅谈数据集获取之道
- "A minute" Copy siege lion log 】 【 run MindSpore LeNet model
- 空洞卷积
- 第三章 标准单元库(下)
- MOOSE平台使用入门攻略——如何运行官方教程的例子
- Deep Learning Theory - Overfitting, Underfitting, Regularization, Optimizers
- Image-Adaptive YOLO for Object Detection in Adverse Weather Conditions
- ConnectionRefusedError: [Errno 111] Connection refused问题解决
猜你喜欢
Amazon Cloud Technology Build On 2022 - AIot Season 2 IoT Special Experiment Experience
在AWS-EC2中安装Minikube集群
DRA821 环境搭建
Deep Adversarial Decomposition: A Unified Framework for Separating Superimposed Images
AWS使用EC2降低DeepRacer的训练成本:DeepRacer-for-cloud的实践操作
BatchNorm&&LayerNorm
Windows10重置MySQL用户密码
Cut the hit pro subtitles export of essays
深度确定性策略梯度(DDPG)
MNIST手写数字识别 —— ResNet-经典卷积神经网络
随机推荐
arm-3-中断体系结构
Rules.make-适合在编辑模式下看
JPEG2jpg
【论文阅读】Multi-View Spectral Clustering with Optimal Neighborhood Laplacian Matrix
Linear Regression 02---Boston Housing Price Prediction
PP-LiteSeg
深度学习理论——过拟合、欠拟合、正则化、优化器
Pytest常用插件
Amazon Cloud Technology Build On-Amazon Neptune's Knowledge Graph-Based Recommendation Model Building Experience
迅雷关闭自动更新
yolov3 data reading (2)
【论文阅读】Mining Cross-Image Semantics for Weakly Supervised Semantic Segmentation
[Deep Learning 21-Day Learning Challenge] 3. Use a self-made dataset - Convolutional Neural Network (CNN) Weather Recognition
学习资料re-id
Halcon缺陷检测
arm-2-基础阶段
腾讯、网易纷纷出手,火到出圈的元宇宙到底是个啥?
典型CCN网络——efficientNet(2019-Google-已开源)
【CV-Learning】Convolutional Neural Network
CSDN大礼包--高校圆桌派大礼包