题面

给你一个长度为 \(n\) 的数组 \(a\) 和一个长度为 \(m\) 的数组 \(b\) (所有 \(1 \le i < m\) 满足 \(b_i > b_{i+1}\) )。最初, \(k\) 的值是 \(1\) 。您的目标是通过重复执行这两种操作中的一种,使数组 \(a\) 为空:

  • \(1\) - 如果 \(k\) 的值小于 \(m\) ,并且数组 \(a\) 不是空的,那么可以将 \(k\) 的值增加 \(1\) 。这不会产生任何费用。
  • \(2\) - 从数组 \(a\) 中移除一个非空的前缀,使得其总和不超过 \(b_k\) 。需要花费 \(m - k\) 。

你需要最小化操作的总成本,使数组 \(a\) 为空。如果无法通过任何操作序列实现,则输出 \(-1\) 。否则,输出操作的最小总成本,以及产生这个最小成本的操作序列的数目,取模为 \(10^9 + 7\) 。

如果在任何一步中选择了不同的操作类型,或在任何一步中删除的前缀大小不同,则认为两个操作序列是不同的。

题解

D1(Easy Version)

题目只要求求出最小总成本,所以考虑 \(dp\),我们根据 \(b_i\) 的单调性,容易发现选择时对 \(b_i\) 按顺序对成本进行松弛即可,唯一需要确定的是状态的来源。

考虑什么时候转移是合法的,必然是一个区间和小于 \(b_i\) 时合法,如果一个区间和小于 \(b_i\),有一个较小的区间被这个区间所包含,那么一定是不优的,因此我们尽可能的将区间扩大,找到最大满足区间和小于 \(b_i\) 的区间,可以枚举左端点或右端点做双指针转移,对每个 \(b_i\) 转移一次即可。

D2(Hard Version)

在 D1 的条件下,我们需要求出转移的方案数,那么对于我们转移的方式应该枚举右端点,这样才能确定每一次转移后的方案数。

对于一个段 \(l\) 转移到 \(r\),若对于某个 \(b_i\) 能使 \(l\) 转移到 \(r\):

  • 当 \(dp_r - dp_l > m - i\) 时,将 \(dp_l\) 的方案数赋值到 \(dp_r\) 上。

  • 当 \(dp_r - dp_l < m - i\) 时,我们跳过这样的 \(l\)。

  • 当 \(dp_r - dp_l = m - i\) 时,我们将连续的 \(l\) 方案数求和,转移到 \(dp_r\) 上。

因此,我们可以记录一个 \(maxx\) 代表当前区间段 \(dp_l = dp_{maxx}\) 可以转移到 \(dp_r\) 的区间段的右端点,如果正在考虑的 \(l\) 小于 \(maxx\),那么可以对这个区间段内满足 \(dp_r - dp_l = m - i\) 的部分求和,可以证明 \(l\) 是单调递增的,如果 \(l\) 大于 \(maxx\),更新 \(maxx\),方案数替换为下一个合法的区间,令 \(maxx = l\) 继续转移。

容易发现,上述操作和双指针无异,时间复杂度 \(O(nm)\)。

参考代码

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 3e5 + 10, mod = 1e9 + 7;
int n, m;
ll a[N], b[N]; void solve()
{
cin >> n >> m;
vector<ll> pre(n + 1);
for (int i = 1; i <= n; i ++ ) cin >> a[i], pre[i] = pre[i - 1] + a[i];
for (int i = 1; i <= m; i ++ ) cin >> b[i];
vector<ll> dp(n + 1, 1e18), f(n + 1);
dp[0] = 0, f[0] = 1;
for (int i = 1; i <= m; i ++ )
{
ll sum = 0;
for (int l = 0, r = 1, maxx = 0; r <= n; r ++ )
{
while (pre[r] - pre[l] > b[i]) (sum -= f[l]) %= mod, l ++ ;
if (maxx < l) maxx = l, sum = 0;
while (maxx < r && dp[maxx] == dp[l]) (sum += f[maxx]) %= mod, maxx ++ ;
if (l < r)
{
if (dp[r] > dp[l] + m - i) dp[r] = dp[l] + m - i, f[r] = sum;
else if (dp[r] == dp[l] + m - i) (f[r] += sum) %= mod;
}
}
}
if (dp[n] == 1e18) cout << "-1\n";
else cout << dp[n] << ' ' << (f[n] + mod) % mod << '\n';
} int main()
{
ios::sync_with_stdio(false);
cin.tie(0), cout.tie(0);
int T = 1;
cin >> T;
while (T -- ) solve();
return 0;
}

CF2027D The Endspeaker (Hard Version) 题解的更多相关文章

  1. Codeforces 1196D2 RGB Substring (Hard version) 题解

    题面 \(q\) 个询问,每个询问给出一个字符串 \(s\),要你在 \(s\) 中用最小替换得到无穷字符串 RGBRGBRGB... 的长度为定值 \(k\) 的子串. 题解 一眼看过去可能是编辑距 ...

  2. CF1157C1-Increasing Subsequence (easy version)题解

    原题地址 题目大意:

  3. Hdoj 2501.Tiling_easy version 题解

    Problem Description 有一个大小是 2 x n 的网格,现在需要用2种规格的骨牌铺满,骨牌规格分别是 2 x 1 和 2 x 2,请计算一共有多少种铺设的方法. Input 输入的第 ...

  4. Palindromes _easy version 题解

    “回文串”是一个正读和反读都一样的字符串,比如“level”或者“noon”等等就是回文串.请写一个程序判断读入的字符串是否是“回文”. Input输入包含多个测试实例,输入数据的第一行是一个正整数n ...

  5. CF1203D2 Remove the Substring (hard version) 题解

    这题初赛让我白给了6分,于是我决定回来解决一下它. 说实话,看原题题面和看CCF代码真是两种完全不同的感受…… ------------思路分析: 把$s$串删去一部分之后,会把$s$串分成两部分,当 ...

  6. UVA12558 Egyptian Fractions (HARD version) (埃及分数,迭代加深搜索)

    UVA12558 Egyptian Fractions (HARD version) 题解 迭代加深搜索,适用于无上界的搜索.每次在一个限定范围中搜索,如果无解再进一步扩大查找范围. 本题中没有分数个 ...

  7. uva12558 Egyptian Fractions (HARD version)(迭代深搜)

    Egyptian Fractions (HARD version) 题解:迭代深搜模板题,因为最小个数,以此为乐观估价函数来迭代深搜,就可以了. #include<cstdio> #inc ...

  8. Codeforces Round #617 (Div. 3) 题解

    又是隔了一年才来补题的我 A.B水题就不用说了 C - Yet Another Walking Robot C题我居然卡了一会,最后决定用map水,结果出来看了看题解,居然真的是map...没想到会出 ...

  9. Codeforces Round #575 (Div. 3)

    本蒟蒻已经掉到灰名了(菜到落泪),希望这次打完能重回绿名吧...... 这次赛中A了三题 下面是本蒟蒻的题解 A.Three Piles of Candies 这题没啥好说的,相加除2就完事了 #in ...

  10. Codeforces Round #581(Div. 2)

    Codeforces Round #581(Div. 2) CF 1204 A. BowWow and the Timetable 题解:发现,$4$的幂次的二进制就是一个$1$后面跟偶数个$0$. ...

随机推荐

  1. Adobe Photoshop cc2022 Mac中文破解版下载 支持intel/M1/M2/M3

    Adobe Photoshop cc2022 Mac版,支持intel.M1.M2.M3芯片,全系都能使用,最低系统需求:11.0以上.建议安装此版本或者ps2024版本. 下载地址放到最后,先来看安 ...

  2. 使用 `Roslyn` 分析器和修复器对.cs源代码添加头部注释

    之前写过两篇关于Roslyn源生成器生成源代码的用例,今天使用Roslyn的代码修复器CodeFixProvider实现一个cs文件头部注释的功能, 代码修复器会同时涉及到CodeFixProvide ...

  3. sicp每日一题[2.3]

    Exercise 2.3 Implement a representation for rectangles in a plane. (Hint: You may want to make use o ...

  4. 1. Two Sum Go实现

    在数组中找到 2 个数之和等于给定值的数字,结果返回 2 个数字在数组中的下标. 1. 解法1 时间复杂度 O(n^2) 直接两次遍历所有节点,进行求和比较 代码如下: func twoSum(num ...

  5. react-pdf预览在线PDF的使用

    1.在react项目中安装react-pdf依赖包 建议安装8.0.2版本的react-pdf,如果安装更高版本的可能出现一些浏览器的兼容性问题: npm install react-pdf@8.0. ...

  6. Nodejs+npm详细安装

    Nodejs详细安装步骤1.去官网下载nodejs,下载地址:https://nodejs.org/en/download/ 根据自己要求下载,我这里下载的是windows installer的. 2 ...

  7. C++ STL list容器——链表

    list容器 简介 链表是一种物理存储单元上非连续.非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的.链表由一系列节点组成,节点可以在运行时动态生成.每个节点包括两部分:一个是存储 ...

  8. SpringMVC —— REST风格简介

    REST风格简介 REST(Representational State Transfer),表现形式转换 传统风格资源描述形式 REST风格描述形式 优点 隐藏资源的访问行为,无法通过地址得知对资源 ...

  9. MyBatis——案例——查询-多条件查询(多参数接收的三种方法)

    查询-多条件查询   编写接口方法:Mapper接口       参数:所有条件查询 List<Brand> selectByCondition(int status,String com ...

  10. 如何用VMWARE创建一个Linux虚拟机

    序言 各位好啊,我是会编程的蜗牛,作为java开发者,我们都是需要接触Linux服务器的,一般部署应用都是部署在Linux服务器上的~ 但一般的服务器要么需要购买,要么只是公司里的,那么有没有免费的L ...