给你两个数组 nums
和 andValues
,长度分别为 n
和 m
。
数组的 值 等于该数组的 最后一个 元素。
你需要将 nums
划分为 m
个 不相交的连续 子数组,对于第 ith
个子数组 `[li,
ri],子数组元素的按位
AND运算结果等于
andValues[i],换句话说,对所有的
1 <= i <= m,
nums[li] &
nums[li + 1] & ... & nums[ri] == andValues[i],其中
&表示按位
AND`运算符。
返回将 nums
划分为 m
个子数组所能得到的可能的 最小 子数组 值 之和。如果无法完成这样的划分,则返回 -1
。
示例 1:
输入: nums = [1,4,3,3,2], andValues = [0,3,3,2]
输出: 12
解释:
唯一可能的划分方法为:
[1,4]
因为1 & 4 == 0
[3]
因为单元素子数组的按位AND
结果就是该元素本身[3]
因为单元素子数组的按位AND
结果就是该元素本身[2]
因为单元素子数组的按位AND
结果就是该元素本身
这些子数组的值之和为 4 + 3 + 3 + 2 = 12
示例 2:
输入: nums = [2,3,5,7,7,7,5], andValues = [0,7,5]
输出: 17
解释:
划分 nums
的三种方式为:
[[2,3,5],[7,7,7],[5]]
其中子数组的值之和为5 + 7 + 5 = 17
[[2,3,5,7],[7,7],[5]]
其中子数组的值之和为7 + 7 + 5 = 19
[[2,3,5,7,7],[7],[5]]
其中子数组的值之和为7 + 7 + 5 = 19
子数组值之和的最小可能值为 17
示例 3:
输入: nums = [1,2,3,4], andValues = [2]
输出: -1
解释:
整个数组 nums
的按位 AND
结果为 0
。由于无法将 nums
划分为单个子数组使得元素的按位 AND
结果为 2
,因此返回
-1
。
提示:
1 <= n == nums.length <= 104
1 <= m == andValues.length <= min(n, 10)
1 <= nums[i] < 105
0 <= andValues[j] < 105
划分数组得到最小的值之和
题目
给你两个数组 nums
和 andValues
,长度分别为 n
和 m
。
数组的 值 等于该数组的 最后一个 元素。
你需要将 nums
划分为 m
个 不相交的连续 子数组,对于第 ith
个子数组 [li, ri]
,子数组元素的按位AND
运算结果等于 andValues[i]
,换句话说,对所有的 1 <= i <= m
,nums[li] & nums[li + 1] & ... & nums[ri] == andValues[i]
,其中 &
表示按位AND
运算符。
返回将 nums
划分为 m
个子数组所能得到的可能的 最小 子数组 值 之和。如果无法完成这样的划分,则返回 -1
。
示例 1:
输入: nums = [1,4,3,3,2], andValues = [0,3,3,2]
输出: 12
解释:
唯一可能的划分方法为:
[1,4]
因为1 & 4 == 0
[3]
因为单元素子数组的按位AND
结果就是该元素本身[3]
因为单元素子数组的按位AND
结果就是该元素本身[2]
因为单元素子数组的按位AND
结果就是该元素本身
这些子数组的值之和为 4 + 3 + 3 + 2 = 12
示例 2:
输入: nums = [2,3,5,7,7,7,5], andValues = [0,7,5]
输出: 17
解释:
划分 nums
的三种方式为:
[[2,3,5],[7,7,7],[5]]
其中子数组的值之和为5 + 7 + 5 = 17
[[2,3,5,7],[7,7],[5]]
其中子数组的值之和为7 + 7 + 5 = 19
[[2,3,5,7,7],[7],[5]]
其中子数组的值之和为7 + 7 + 5 = 19
子数组值之和的最小可能值为 17
示例 3:
输入: nums = [1,2,3,4], andValues = [2]
输出: -1
解释:
整个数组 nums
的按位 AND
结果为 0
。由于无法将 nums
划分为单个子数组使得元素的按位 AND
结果为 2
,因此返回-1
。
提示:
1 <= n == nums.length <= 104
1 <= m == andValues.length <= min(n, 10)
1 <= nums[i] < 105
0 <= andValues[j] < 105
题解
方法一:
思路
划分型dp
设$f_{i,j}$是前i个数分成j组的最小代价。
$f_{i,j} = f_{i’,j-1}+nums[i]$,其中$i’$是最优决策点,满足$nums[i’] & nums[i’+1] & … & nums[i] == andValues[j-1]$
由于“与和”的性质,对于固定一段的子数组,随着长度增长,与和减少。并且不同的与和值不超过logU个。利用灵神的模板,可以得到每种“与值”的左右端点,用m颗线段树,第i个线段树维护$f_{…,i}$的区间最小值。
代码
1 | template<class t,class u> ostream& operator<<(ostream& os,const pair<t,u>& p) { |