2369. 检查数组是否存在有效划分
题目描述
给你一个下标从 0 开始的整数数组 nums
,你必须将数组划分为一个或多个 连续 子数组。
如果获得的这些子数组中每个都能满足下述条件 之一 ,则可以称其为数组的一种 有效 划分:
- 子数组 恰 由
2
个相等元素组成,例如,子数组[2,2]
。 - 子数组 恰 由
3
个相等元素组成,例如,子数组[4,4,4]
。 - 子数组 恰 由
3
个连续递增元素组成,并且相邻元素之间的差值为1
。例如,子数组[3,4,5]
,但是子数组[1,3,5]
不符合要求。
如果数组 至少 存在一种有效划分,返回 true
,否则,返回 false
。
示例 1:
输入:nums = [4,4,4,5,6] 输出:true 解释:数组可以划分成子数组 [4,4] 和 [4,5,6] 。 这是一种有效划分,所以返回 true 。
示例 2:
输入:nums = [1,1,1,2] 输出:false 解释:该数组不存在有效划分。
提示:
2 <= nums.length <= 105
1 <= nums[i] <= 106
解法
方法一:记忆化搜索
我们设计一个函数 \(dfs(i)\),表示从下标 \(i\) 开始是否存在一种有效划分。那么答案就是 \(dfs(0)\)。
函数 \(dfs(i)\) 的执行过程如下:
- 如果 \(i \ge n\),返回 \(true\)。
- 如果 \(i\) 和 \(i+1\) 下标的元素相等,那么可以选择将 \(i\) 和 \(i+1\) 作为一个子数组,递归调用 \(dfs(i+2)\)。
- 如果 \(i\), \(i+1\) 和 \(i+2\) 下标的元素相等,那么可以选择将 \(i\), \(i+1\) 和 \(i+2\) 作为一个子数组,递归调用 \(dfs(i+3)\)。
- 如果 \(i\), \(i+1\) 和 \(i+2\) 下标的元素依次递增 \(1\),那么可以选择将 \(i\), \(i+1\) 和 \(i+2\) 作为一个子数组,递归调用 \(dfs(i+3)\)。
- 如果上述情况都不满足,返回 \(false\),否则返回 \(true\)。
即:
\[
dfs(i) = \textit{OR}
\begin{cases}
true,&i \ge n\\
dfs(i+2),&i+1 < n\ \textit{and}\ \textit{nums}[i] = \textit{nums}[i+1]\\
dfs(i+3),&i+2 < n\ \textit{and}\ \textit{nums}[i] = \textit{nums}[i+1] = \textit{nums}[i+2]\\
dfs(i+3),&i+2 < n\ \textit{and}\ \textit{nums}[i+1] - \textit{nums}[i] = 1\ \textit{and}\ \textit{nums}[i+2] - \textit{nums}[i+1] = 1
\end{cases}
\]
为了避免重复计算,我们使用记忆化搜索的方法。
时间复杂度 \(O(n)\),空间复杂度 \(O(n)\)。其中 \(n\) 为数组的长度。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
|
方法二:动态规划
我们可以将方法一中的记忆化搜索转换为动态规划。
设 \(f[i]\) 表示数组的前 \(i\) 个元素是否存在一种有效划分,初始时 \(f[0] = true\),答案就是 \(f[n]\)。
状态转移方程如下:
\[
f[i] = \textit{OR}
\begin{cases}
true,&i = 0\\
f[i-2],&i-2 \ge 0\ \textit{and}\ \textit{nums}[i-1] = \textit{nums}[i-2]\\
f[i-3],&i-3 \ge 0\ \textit{and}\ \textit{nums}[i-1] = \textit{nums}[i-2] = \textit{nums}[i-3]\\
f[i-3],&i-3 \ge 0\ \textit{and}\ \textit{nums}[i-1] - \textit{nums}[i-2] = 1\ \textit{and}\ \textit{nums}[i-2] - \textit{nums}[i-3] = 1
\end{cases}
\]
时间复杂度 \(O(n)\),空间复杂度 \(O(n)\)。其中 \(n\) 为数组的长度。
1 2 3 4 5 6 7 8 9 10 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
1 2 3 4 5 6 7 8 9 10 11 12 |
|