LeetCode周赛#204 题解
1566. 重复至少 K 次且长度为 M 的模式 #模拟
题目链接
题意
给定正整数数组 arr,请你找出一个长度为 m 且在数组中至少重复 k 次的模式。
模式 是由一个或多个值组成的子数组(连续的子序列),连续 重复多次但 不重叠 。 模式由其长度和重复次数定义。如果数组中存在至少重复 k 次且长度为 m 的模式,则返回 true ,否则返回 false 。
分析
题目初看有点麻烦,但实际上隔着特定周期(即重复长度)去检查相应字符是否相等,模拟一下题意即可。
class Solution {
public:
bool containsPattern(vector<int>& arr, int m, int k) {
int cnt = 0; //统计重复度(次数)
int len = arr.size();
int tmp = 0; //计算当前重复长度
for (int i = 0; i < len; i++){
if(tmp == m){ //达到重复长度,答案+1
cnt++;
tmp = 0; //重复长度归零
}
if(i + m >= len) break; //越界
if(arr[i] == arr[i + m]) tmp++;
else{ //字符不匹配
if(cnt >= k - 1) return true;
cnt = 0; tmp = 0;
}
}
return cnt >= k - 1;
}
};
1567. 乘积为正数的最长子数组长度 #贪心
题目链接
题意
给定整数数组 nums ,请你求出乘积为正数的最长子数组的长度。其中,一个数组的子数组是由原数组中零个或者更多个连续数字组成的数组。
分析
我的版本(代码略长)
class Solution {
public:
int getMaxLen(vector<int>& nums) {
int len = nums.size();
int cnt = 0, mymax = -0x3f3f3f3f;
bool last = false; //记录前面遍历过的序列中是否存在负数
/*从左至右遍历*/
for (int i = 0; i < len; i++){
cnt++;
if(nums[i] == 0){
cnt = 0; last = false;
}
else if(nums[i] < 0){
if(last == false) last = true; //如果前面遍历过的序列中并没有过剩的负数
else last = false;
}
if(last == false) mymax = max(mymax, cnt); //只有当前面的负号全消掉的时候方能更新答案
}
/*从右至左遍历*/
cnt = 0; last = false;
for(int i = len - 1; i >= 0; i--){
cnt++;
if(nums[i] == 0){
cnt = 0; last = false;
}
else if(nums[i] < 0){
if(last == false) last = true;
else last = false;
}
if(last == false) mymax = max(mymax, cnt);
}
return (mymax == -0x3f3f3f3f) ? cnt - 1 : mymax; //避免全为负数
}
};
\(@xu012\)的队列版本,易写,不过复杂度有点高
class Solution {
public:
int getMaxLen(vector<int>& nums) {
int len = nums.size();
int last = -1, ans = -1;
queue<int> myque; //队列存放负数位置
for (int i = 0; i < len; i++){
if(nums[i] == 0){ //遇到0即隔断
last = i;
queue<int> tmp;
myque.swap(tmp); //清空队列
}
else if(nums[i] < 0) myque.push(i);
if(myque.size() % 2 == 0)
ans = max(ans, i - last); //当前数字下标-上一个0的下标
else
ans = max(ans, i - myque.front());
}
return ans;
}
};
1569. 将子数组重新排序得到同一个二叉查找树的方案数 #分治 #组合数
题目链接
题意
给定数组 nums表示 1 到 n 的一个排列。按照元素在nums中的顺序依次插入一个初始为空的二叉查找树。请你统计将 nums 重新排序后,统计满足如下条件的方案数:重排后得到的二叉查找树与 nums 原本数字顺序得到的二叉查找树相同,将结果对 10^9 + 7 取余数。
样例

分析
显然第一个插入的元素必为根节点,设为rt,通过rt的大小,我们就能决定数组后面的元素如何划分至左右子树。我们设lo表示小于rt的元素构成的集合,hi表示大于rt元素集合,显然lo、hi分别代表左、右子树节点集。
观察到,无论我先从lo集合中取元素插入rt左子树,还是我先从hi集合取元素插入右子树,树的拓扑结构并不影响,简单来说,两个大集合谁先谁后插入搜索树中,都不会发生相互影响。
那么,集合内部的元素插入顺序是否影响呢?我们回想第一句话,第一个插入元素必为根节点。改变了第一个插入的元素,就会使得新树的结构不再于给定的原树结构相同了。
我们只要保证左子树的元素内部相对顺序,以及右子树的元素内部相对顺序,调整两种集合的插入顺序,能够获得新插入序列,但不影响原树的结构。
如何统计?从n个元素中将k个元素(均属于某一集合)挑选出来,同时保证这k个元素相对顺序相同(需要去序),共有\(C^{k}_n\)(或\(C^{lo.size}_{lo.size+hi.size}\))个组合。当然我们只是将两个大集合的组合统计了,大集合内部同样有如此的统计方式,由乘法原理得到\(f(n) = f(lo)\times f(hi) \times C^{lo.size}_{lo.size+hi.size}\)
typedef long long ll;
ll c[1010][1010];
class Solution {
private:
const int MOD = 1e9+7;
public:
void Calculate(int len){ //预计算组合数公式
for(int i = 1; i <= len; i++){
c[i][0] = c[i][i] = 1;
for(int j = 1; j < i; j++)
c[i][j] = (c[i - 1][j - 1] + c[i - 1][j]) % MOD;
} //递推过程类似于01背包
}
int dfs(vector<int>& arr){
vector<int> lo, hi;
if(arr.size() <= 1) return 1;
for (int i = 1; i < (int)arr.size(); i++){
if(arr[i] < arr[0]) lo.push_back(arr[i]);
else hi.push_back(arr[i]);
} //划分左右子树
ll cur = c[lo.size() + hi.size()][lo.size()];
cur = (cur * dfs(lo)) % MOD;
cur = (cur * dfs(hi)) % MOD;
return cur % MOD;
}
int numOfWays(vector<int>& nums) {
Calculate(nums.size());
return (dfs(nums) - 1 + MOD) % MOD; //记得-1
}
};
1568. 使陆地分离的最少天数 #Tarjan求割点
题目链接
题意
给你一个由若干 0 和 1 组成的二维网格 grid ,其中 0 表示水,而 1 表示陆地。岛屿由水平方向或竖直方向上相邻的 1 (陆地)连接形成。
如果 恰好只有一座岛屿 ,则认为陆地是 连通的 ;否则,陆地就是 分离的 。一天内,可以将任何单个陆地单元(1)更改为水单元(0)。现要求使陆地分离的最少天数。

分析
Tarjan割点的算法我还没学,暂时先咕
其实该题有个漏洞,所有岛屿都有小边角,只要将小边角切出一块即可,也就说最多天数为2。
LeetCode周赛#204 题解的更多相关文章
- Leetcode 周赛#202 题解
本周的周赛题目质量不是很高,因此只给出最后两题题解(懒). 1552 两球之间的磁力 #二分答案 题目链接 题意 有n个空篮子,第i个篮子位置为position[i],现希望将m个球放到这些空篮子,使 ...
- LeetCode周赛#203 题解
1561. 你可以获得的最大硬币数目 #贪心 题目链接 题意 有 3n 堆数目不一的硬币,你和你的朋友们打算按以下方式分硬币: 每一轮中,你将会选出 任意 3 堆硬币(不一定连续). Alice 将会 ...
- Leetcode 周赛#201 题解
1545 找出第N个二进制字符串的第K位 #分治 题目链接 题意 给定正整数\(n(\leq 20)\)与\(k\),二进制串\(S_n\)形成规则有: \(S_1 = "0"\) ...
- Leetcode 周赛#200 题解
1535 找出数组游戏的赢家 #模拟+优化 题目链接 题意 给你一个由 不同 整数组成的整数数组 arr 和一个整数 k(\(1\leq k\leq1e9\)) .每回合游戏都在数组的arr[0] 和 ...
- LeetCode双周赛#33 题解
5480. 可以到达所有点的最少点数目 #贪心 题目链接 题意 给定有向无环图,编号从0到n-1,一个边集数组edges(表示从某个顶点到另一顶点的有向边),现要找到最小的顶点集合,使得从这些点出发, ...
- Leetcode 双周赛#32 题解
1540 K次操作转变字符串 #计数 题目链接 题意 给定两字符串\(s\)和\(t\),要求你在\(k\)次操作以内将字符串\(s\)转变为\(t\),其中第\(i\)次操作时,可选择如下操作: 选 ...
- 【Leetcode周赛】从contest-111开始。(一般是10个contest写一篇文章)
Contest 111 (题号941-944)(2019年1月19日,补充题解,主要是943题) 链接:https://leetcode.com/contest/weekly-contest-111 ...
- 【Leetcode周赛】从contest-41开始。(一般是10个contest写一篇文章)
Contest 41 ()(题号) Contest 42 ()(题号) Contest 43 ()(题号) Contest 44 (2018年12月6日,周四上午)(题号653—656) 链接:htt ...
- 【Leetcode周赛】从contest-51开始。(一般是10个contest写一篇文章)
Contest 51 (2018年11月22日,周四早上)(题号681-684) 链接:https://leetcode.com/contest/leetcode-weekly-contest-51 ...
随机推荐
- java数据结构-07栈
一.什么是栈 栈是一种线性结构,栈的特点就是先进后出(FILO):就像弹夹装子弹一样,最先压进去的在最底下,最后才被射出. 二.相关接口设计 三.栈的实现 栈可以用之前的数组.链表等设计,这里我使 ...
- Java之HttpClient调用WebService接口发送短信源码实战
摘要 Java之HttpClient调用WebService接口发送短信源码实战 一:接口文档 二:WSDL 三:HttpClient方法 HttpClient方法一 HttpClient方法二 Ht ...
- Luogu P4247 [清华集训2012]序列操作
题意 给定一个长度为 \(n\) 的序列 \(a\) 和 \(q\) 次操作,每次操作形如以下三种: I a b c,表示将 \([a,b]\) 内的元素加 \(c\). R a b,表示将 \([a ...
- 震惊!你还不知道SpringBoot真正的启动引导类
引言 SpringBoot项目中的启动类,一般都是XXApplication,例如「StatsApplication」,「UnionApplication」. 每个项目的启动类名称都不一样.但是它的启 ...
- DP百题练(一)
目录 DP百题练(一) 线性 DP 简述 Arithmetic Progressions [ZJOI2006]物流运输 LG1095 守望者的逃离 LG1103 书本整理 CH5102 移动服务 LG ...
- 【Luogu】P2292 [HNOI2004]L语言 题解
前置芝士:\(Trie\)字典树 这道题,说是AC自动机,实际上一个\(Trie+\)队列轻松搞定. 首先,我们对所有单词建一棵\(Trie\). 然后,定义一个空队列\(Q\),初始时把\(-1\) ...
- System.Net邮件发送功能踩过的坑
System.Net邮件发送功能踩过的坑 目录 System.Net邮件发送功能踩过的坑 1.EazyEmail邮件发送类库 2.邮件发送授权码与邮件密码 3.通过邮件密码来发送邮件 4.Wiresh ...
- JavaScript ES 模块:现代化前端编程必备技能
自从 ES 模块被添加到规范中后,JavaScript 中的模块就更加简单了.模块按文件分开,异步加载.导出是用 export 关键字定义的:值可以用 import 关键字导入. 虽然导入和导出单个值 ...
- C++ 数据结构 4:排序
1 基本概念 1.1 定义 排序是计算机内经常进行的一种操作,其目的是将一组"无序"的数据元素调整为"有序"的数据元素. 1.2 数学定义 假设含n个数据元素的 ...
- TCP拥塞控制原理
一.何为拥塞 路由器无法处理高速到达的数据而被迫丢弃数据的现象叫做拥塞. 二.何为拥塞控制 TCP流量控制时为了平衡一个链接中接收方和发送方的速度匹配问题,当发送方发现发送速度大于接收方的接收速度时动 ...