题面

给你一个长度为 \(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. Python爬虫常用库介绍(requests、BeautifulSoup、lxml、json)

    1.requests库 http协议中,最常用的就是GET方法: import requests response = requests.get('http://www.baidu.com') pri ...

  2. spark 自定义 accumulator

    默认的accumulator 只是最简单的 int/float 有时候我需要一个map来作为accumulator 这样,就可以处理 <string, int>类型的计数了. 此外我还需要 ...

  3. 在stable diffussion中完美修复AI图片

    无论您的提示和模型有多好,一次性获得完美图像的情况很少见. 修复小缺陷的不可或缺的方法是图像修复(inpainting).在这篇文章中,我将通过一些基本示例来介绍如何使用图像修复来修复缺陷. 需要的软 ...

  4. Angular 18+ 高级教程 – Naming Conversion

    前言 命名规范对项目维护是很重要的. Angular 对项目的渗透很大的, 必须做好命名规范, 不然会很乱. InjectionToken InjectionToken = UPPER_SNAKE_C ...

  5. JavaScript Library – Swiper

    前言 官网已经有很好的教程了, 这篇只是记入一些我用过的东西和冷门知识. 参考 官网安装 官网 Demo 安装 yarn add swiper JS import Swiper from 'swipe ...

  6. CSS & JS Effect – fade in

    参考: stackoverflow – Is there a CSS-only (pure CSS) workaround to apply fade-in and fade-out on objec ...

  7. 安全 – 常见 Web 攻击

    前言 最近在研究 WAF, 顺便记入一下常见的 Web 攻击. SQL Injection SQL injection 是指程序员直接拿 client input, 拼接到 SQL query 中. ...

  8. 【赵渝强老师】Docker Swarm集群的数据持久化

    如果Docker Swarm集群中运行了mysql.nginx等服务,这些服务的数据如果没有挂载到宿主机中,那么容器一旦停止运行,那就意味着数据丢失. 有什么方法可以解决swarm集群中运行的服务能够 ...

  9. /proc/vmalloc

    root@pita2_mr813_tina35:/# cat /proc/vmallocinfo 0xffffff80007d0000-0xffffff8000902000 1253376 load_ ...

  10. 在Vue3中如何实现四种全局状态数据的统一管理?

    四种全局状态数据 在实际开发当中,会遇到四种全局状态数据:异步数据(一般来自服务端).同步数据.同步数据又分为三种:localstorage.cookie.内存.在传统的 Vue3 当中,分别采用不同 ...