【AGC009C】Division into Two

题面

洛谷

题解

首先有一个比较显然的\(n^2\)算法:

设\(f_{i,j}\)表示\(A\)序列当前在第\(i\)个,\(B\)序列当前在第\(j\)个的方案数,发现\(i,j\)大小没有限制不是很好转移,于是再设一个\(g_{i,j}\)表示\(B\)序列当前在第\(i\)个,\(A\)序列当前在第\(j\)个的方案数的,这样子我们就可以钦定\(i>j\)了,转移会方便许多。

转移如下:

\[\left\{
\begin{aligned}
f_{i+1,j}\leftarrow f_{i,j}(a_{i+1}-a_i\geq A)\\
g_{i+1,i}\leftarrow f_{i,j}(a_{i+1}-a_j\geq B)\\
g_{i+1,j}\leftarrow g_{i,j}(a_{i+1}-a_i\geq B)\\
f_{i+1,i}\leftarrow g_{i,j}(a_{i+1}-a_j\geq A)
\end{aligned}
\right.
\]


然后让我们想一想怎么优化这个东西。

不妨设\(A<B\),则对于\(\forall i\geq 3,a_i-a_{i-2}\geq A\),对于不符合的我们直接判掉。

设\(f_i\)表示当前\(B\)序列在\(i\)的方案数,我们对于可以转移过来的\(j\),必须要满足\(a_i-a_j\geq B\),而对于\(\forall k\in [j+1,i-1]\),则要满足\(a_k-a_{k-1}\geq A\),考虑到这样的\(j\)是一个区间,而区间左右端点单调,我们把这个区间搞出来然后前缀和优化即可。

复杂度\(O(n)\)。

代码

\\O(n^2)
int main () {
scanf("%d %lld %lld", &N, &A, &B);
if (A > B) swap(A, B);
for (int i = 1; i <= N; i++) scanf("%lld", a + i);
for (int i = 3; i <= N; i++) if (a[i] - a[i - 2] < A) return puts("0") & 0;
a[0] = -1e18;
int ans = 0;
f[1][0] = 1, g[1][0] = 1;
for (int i = 1; i <= N; i++)
for (int j = 0; j < i; j++) {
if (a[i + 1] - a[i] >= A) f[i + 1][j] = (f[i + 1][j] + f[i][j]) % Mod;
if (a[i + 1] - a[j] >= B) g[i + 1][i] = (g[i + 1][i] + f[i][j]) % Mod;
if (a[i + 1] - a[i] >= B) g[i + 1][j] = (g[i + 1][j] + g[i][j]) % Mod;
if (a[i + 1] - a[j] >= A) f[i + 1][i] = (f[i + 1][i] + g[i][j]) % Mod;
}
for (int i = 0; i < N; i++) ans = (ans + (f[N][i] + g[N][i]) % Mod) % Mod;
printf("%d\n", ans);
return 0;
}
\\O(n)
int main () {
scanf("%d %lld %lld", &N, &A, &B);
for (int i = 1; i <= N; i++) scanf("%lld", a + i);
if (A > B) swap(A, B);
for (int i = 3; i <= N; i++) if (a[i] - a[i - 2] < A) return puts("0") & 0;
f[0] = s[0] = 1;
int l = 0, r = 0;
for (int i = 1; i <= N; i++) {
while (r < i - 1 && a[i] - a[r + 1] >= B) ++r;
if (l <= r) f[i] = (s[r] - (l ? s[l - 1] : 0) + Mod) % Mod;
s[i] = (s[i - 1] + f[i]) % Mod;
if (i != 1 && a[i] - a[i - 1] < A) l = i - 1;
}
int ans = 0;
for (int i = N; ~i; i--) {
ans = (ans + f[i]) % Mod;
if (i != N && a[i + 1] - a[i] < A) break;
}
printf("%d\n", ans);
return 0;
}

【AGC009C】Division into Two的更多相关文章

  1. 【HDOJ】【3480】Division

    DP/四边形不等式 要求将一个可重集S分成M个子集,求子集的极差的平方和最小是多少…… 首先我们先将这N个数排序,容易想到每个自己都对应着这个有序数组中的一段……而不会是互相穿插着= =因为交换一下明 ...

  2. 【BZOJ】【1385】【Baltic2000】Division expression

    欧几里得算法 普通的求个gcd即可……思路题 因为要求尽量是整数……所以 $\frac{x_1}{x_2*x_3*x_4*....*x_n}$是最大的结果了,因为$x_2$必须为分母,$x_1$必须为 ...

  3. 【UVA11859】Division Game(SG函数,Nim游戏)

    题意:给定一个n*m的矩阵,两个游戏者轮流操作. 每次可以选一行中的1个或多个大于1的整数,把它们中的每个数都变成它的某个真因子,不能操作的输. 问先手能否获胜 n,m<=50,2<=a[ ...

  4. 【BZOJ4421】[Cerc2015] Digit Division 动态规划

    [BZOJ4421][Cerc2015] Digit Division Description 给出一个数字串,现将其分成一个或多个子串,要求分出来的每个子串能Mod M等于0. 将方案数(mod 1 ...

  5. 【HDU6037】Expectation Division(动态规划,搜索)

    [HDU6037]Expectation Division(动态规划,搜索) 题面 Vjudge 你有一个数\(n\),\(n\le 10^{24}\),为了方便会告诉你\(n\)分解之后有\(m\) ...

  6. 【题解】HDU5845 Best Division (trie树)

    [题解]HDU5845 Best Division (trie树) 题意:给定你一个序列(三个参数来根),然后请你划分子段.在每段子段长度小于等于\(L\)且子段的异或和\(\le x\)的情况下最大 ...

  7. 【BZOJ】1385 [Baltic2000]Division expression

    [算法]欧几里德算法 [题解]紫书原题 #include<cstdio> #include<algorithm> using namespace std; ; int T,t, ...

  8. 【例题 7-1 UVA - 725】Division

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举分母从0到99999. 得到分子,判断合法 [代码] /* 1.Shoud it use long long ? 2.Have ...

  9. 【NLP】Python NLTK处理原始文本

    Python NLTK 处理原始文本 作者:白宁超 2016年11月8日22:45:44 摘要:NLTK是由宾夕法尼亚大学计算机和信息科学使用python语言实现的一种自然语言工具包,其收集的大量公开 ...

随机推荐

  1. snap应用多版本卸载

    Ubuntu18.04新增了几个内置软件使用Snap格式.同样的沙箱式处理方式,除了Canonical主推的Snap,还有Fedora的Flatpak和AppImage.一般正常使用没问题,就是第一次 ...

  2. Thread.currentThread().setContextClassLoader为什么不生效与java.lang.NoClassDefFoundError之Java类加载的Parent first Classloader

    众所周知,Java的类加载机制采用了双亲委派模型,导致在进行类加载的时候会有多个加载器,这种复杂的机制,有时候会导致‘Exception in thread main java.lang.NoClas ...

  3. url拼接参数格式

    在一些情况下,需要直接往url上拼接请求参数. http://www.yanggb.com?flag=1&type=normal&role=customer 通过上面的例子就可以看出, ...

  4. fancybit个人简介

    程序员一枚 熟悉C C++ C# js lua等多种常见开发语言 熟悉Unity游戏开发 node.js pomelo和C# scut 网游后端框架 做过.net和php网站后端 二次元文化爱好者 有 ...

  5. VS报错,Metadata file 'xxx.dll' could not be found

    错误提示“Metadata file 'xxx.dll' could not be found”步骤如下:1.右键单击解决方案,然后单击“属性”.2.单击左侧的配置.3.确保选中了它找不到的项目的“生 ...

  6. 脱离Office约束,C#结合Mpxj组件完美解析MSProject(.mpp)文件

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  7. 解决 Electron 5.0 版本出现 require is not defined 的问题

    Electron已经发布了5.0正式版,升级后发现原来能运行的代码报错提示require is not defined 经查相关资料,原来官方在5.0版本修改了nodeIntegration的默认值, ...

  8. APS系统如何落地?用户实际痛点解析!

    APS软件在中国的发展,在很长的时间内处于非常尴尬的状态:大企业都了解APS很重要,但只有非常少的企业肯真正实施APS软件,处于叫好不叫座的状态.直到工业4.0概念流行后,APS才逐渐被国内企业所认可 ...

  9. Android源码分析(十六)----adb shell 命令进行OTA升级

    一: 进入shell命令界面 adb shell 二:创建目录/cache/recovery mkdir /cache/recovery 如果系统中已有此目录,则会提示已存在. 三: 修改文件夹权限 ...

  10. mysql后台线程详解

    1.mysql后台线程 mysql后台线程主要用于维持服务器的正常运行和完成用户提交的任务,主要包括:master thread,read thread,write thread,redo log t ...