给你一个下标从 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
方法一:记忆化搜索
时间复杂度
方法二:动态规划
设
根据题意,当
答案为
时间复杂度
class Solution:
def validPartition(self, nums: List[int]) -> bool:
@cache
def dfs(i):
if i == n:
return True
res = False
if i < n - 1 and nums[i] == nums[i + 1]:
res = res or dfs(i + 2)
if i < n - 2 and nums[i] == nums[i + 1] and nums[i + 1] == nums[i + 2]:
res = res or dfs(i + 3)
if (
i < n - 2
and nums[i + 1] - nums[i] == 1
and nums[i + 2] - nums[i + 1] == 1
):
res = res or dfs(i + 3)
return res
n = len(nums)
return dfs(0)
class Solution:
def validPartition(self, nums: List[int]) -> bool:
n = len(nums)
dp = [False] * (n + 1)
dp[0] = True
for i in range(2, n + 1):
if nums[i - 1] == nums[i - 2]:
dp[i] = dp[i] or dp[i - 2]
if i > 2 and nums[i - 1] == nums[i - 2] == nums[i - 3]:
dp[i] = dp[i] or dp[i - 3]
if i > 2 and nums[i - 1] - nums[i - 2] == 1 and nums[i - 2] - nums[i - 3] == 1:
dp[i] = dp[i] or dp[i - 3]
return dp[-1]
class Solution {
private int n;
private int[] f;
private int[] nums;
public boolean validPartition(int[] nums) {
this.nums = nums;
n = nums.length;
f = new int[n];
Arrays.fill(f, -1);
return dfs(0);
}
private boolean dfs(int i) {
if (i == n) {
return true;
}
if (f[i] != -1) {
return f[i] == 1;
}
boolean res = false;
if (i < n - 1 && nums[i] == nums[i + 1]) {
res = res || dfs(i + 2);
}
if (i < n - 2 && nums[i] == nums[i + 1] && nums[i + 1] == nums[i + 2]) {
res = res || dfs(i + 3);
}
if (i < n - 2 && nums[i + 1] - nums[i] == 1 && nums[i + 2] - nums[i + 1] == 1) {
res = res || dfs(i + 3);
}
f[i] = res ? 1 : 0;
return res;
}
}
class Solution {
public boolean validPartition(int[] nums) {
int n = nums.length;
boolean[] dp = new boolean[n + 1];
dp[0] = true;
for (int i = 2; i <= n; ++i) {
if (nums[i - 1] == nums[i - 2]) {
dp[i] = dp[i] || dp[i - 2];
}
if (i > 2 && nums[i - 1] == nums[i - 2] && nums[i - 2] == nums[i - 3]) {
dp[i] = dp[i] || dp[i - 3];
}
if (i > 2 && nums[i - 1] - nums[i - 2] == 1 && nums[i - 2] - nums[i - 3] == 1) {
dp[i] = dp[i] || dp[i - 3];
}
}
return dp[n];
}
}
class Solution {
public:
vector<int> f;
vector<int> nums;
int n;
bool validPartition(vector<int>& nums) {
n = nums.size();
this->nums = nums;
f.assign(n, -1);
return dfs(0);
}
bool dfs(int i) {
if (i == n) return true;
if (f[i] != -1) return f[i] == 1;
bool res = false;
if (i < n - 1 && nums[i] == nums[i + 1]) res = res || dfs(i + 2);
if (i < n - 2 && nums[i] == nums[i + 1] && nums[i + 1] == nums[i + 2]) res = res || dfs(i + 3);
if (i < n - 2 && nums[i + 1] - nums[i] == 1 && nums[i + 2] - nums[i + 1] == 1) res = res || dfs(i + 3);
f[i] = res ? 1 : 0;
return res;
}
};
class Solution {
public:
bool validPartition(vector<int>& nums) {
int n = nums.size();
vector<bool> dp(n + 1);
dp[0] = true;
for (int i = 2; i <= n; ++i)
{
if (nums[i - 1] == nums[i - 2]) dp[i] = dp[i] || dp[i - 2];
if (i > 2 && nums[i - 1] == nums[i - 2] && nums[i - 2] == nums[i - 3]) dp[i] = dp[i] || dp[i - 3];
if (i > 2 && nums[i - 1] - nums[i - 2] == 1 && nums[i - 2] - nums[i - 3] == 1) dp[i] = dp[i] || dp[i - 3];
}
return dp[n];
}
};
func validPartition(nums []int) bool {
n := len(nums)
f := make([]int, n)
for i := range f {
f[i] = -1
}
var dfs func(int) bool
dfs = func(i int) bool {
if i == n {
return true
}
if f[i] != -1 {
return f[i] == 1
}
res := false
f[i] = 0
if i < n-1 && nums[i] == nums[i+1] {
res = res || dfs(i+2)
}
if i < n-2 && nums[i] == nums[i+1] && nums[i+1] == nums[i+2] {
res = res || dfs(i+3)
}
if i < n-2 && nums[i+1]-nums[i] == 1 && nums[i+2]-nums[i+1] == 1 {
res = res || dfs(i+3)
}
if res {
f[i] = 1
}
return res
}
return dfs(0)
}
func validPartition(nums []int) bool {
n := len(nums)
dp := make([]bool, n+1)
dp[0] = true
for i := 2; i <= n; i++ {
if nums[i-1] == nums[i-2] {
dp[i] = dp[i] || dp[i-2]
}
if i > 2 && nums[i-1] == nums[i-2] && nums[i-2] == nums[i-3] {
dp[i] = dp[i] || dp[i-3]
}
if i > 2 && nums[i-1]-nums[i-2] == 1 && nums[i-2]-nums[i-3] == 1 {
dp[i] = dp[i] || dp[i-3]
}
}
return dp[n]
}
function validPartition(nums: number[]): boolean {
const n = nums.length;
const vis = new Array(n).fill(false);
const queue = [0];
while (queue.length !== 0) {
const i = queue.shift() ?? 0;
if (i === n) {
return true;
}
if (!vis[i + 2] && i + 2 <= n && nums[i] === nums[i + 1]) {
queue.push(i + 2);
vis[i + 2] = true;
}
if (
!vis[i + 3] &&
i + 3 <= n &&
((nums[i] === nums[i + 1] && nums[i + 1] === nums[i + 2]) ||
(nums[i] === nums[i + 1] - 1 &&
nums[i + 1] === nums[i + 2] - 1))
) {
queue.push(i + 3);
vis[i + 3] = true;
}
}
return false;
}
function validPartition(nums: number[]): boolean {
const n = nums.length;
const dp = new Array(n + 1).fill(false);
dp[0] = true;
for (let i = 2; i <= n; ++i) {
if (nums[i - 1] == nums[i - 2]) {
dp[i] = dp[i] || dp[i - 2];
}
if (i > 2 && nums[i - 1] == nums[i - 2] && nums[i - 2] == nums[i - 3]) {
dp[i] = dp[i] || dp[i - 3];
}
if (
i > 2 &&
nums[i - 1] - nums[i - 2] == 1 &&
nums[i - 2] - nums[i - 3] == 1
) {
dp[i] = dp[i] || dp[i - 3];
}
}
return dp[n];
}