Codeforces.1029D.Isolation(DP 分块)
\(Description\)
给定长为\(n\)的序列\(A_i\)和一个整数\(K\)。把它划分成若干段,满足每段中恰好出现过一次的数的个数\(\leq K\)。求方案数。
\(K\leq n\leq10^5\)。
\(Solution\)
设\(f[i]\)表示前\(i\)个数的答案,\(g[j]\)表示\(j\sim i\)恰好出现过一次的数的个数。有$$f[i]=\sum_{j\leq i,\ g[j]\leq K}f[j-1]$$
记\(las_i\)为\(A_i\)上次出现的位置下标。每次\(i\)移动时,\(g[j]\)的变化就是,\([las_i+1,i]\)区间\(+1\),\([las_{las_i}+1,las_i]\)区间\(-1\)。
也就是要动态修改\(g[j]\),求\(g[j]\leq K\)的\(f[j-1]\)的和。
数据结构什么的不好搞。考虑直接分块。
一种最简单的想法是,块内sort后维护前缀和,查询的时候二分。复杂度\(O(n\sqrt n\log(\sqrt n))\)(注意确实是\(\log(\sqrt n)\)),但过不去。
设\(s[i][j]\)表示第\(i\)块中,\(g[k]\leq j\)的\(f[k]\)的和,\(tag[i]\)表示第\(i\)块的整体修改标记。
考虑区间修改。对于整块直接打标记。对于零散部分,因为只是\(+1\),容易发现对于第\(i\)块,只有\(s[i][g[j]]\)的值改变了,且只是少掉了\(f[j-1]\)(\(j\)是影响到的下标,显然可以暴力枚举)。那么可以暴力更新\(s[i]\)。
对于整块的查询,假设是第\(i\)块,需要满足\(j+tag[i]\leq k\),即\(j\leq k-tag[i]\),那么\(s[i][k-tag[i]]\)就是答案了。
那么这样就可以啦。
其实还可以优化。
把每个修改拆成前缀修改,即:\([1,i]\)区间\(+1\),\([1,las_i]\)区间\(-2\),\([1,las_{las_i}]\)区间\(+1\)。
这样有什么好处呢。设\(i\)所在的块为\(p\)。那么对\(p\)块零散部分暴力修改,对\(1\sim p-1\)块统一打上标记\(tag\)。
可以发现这样\(s[i][j]\)的第二维是\(O(\sqrt n)\)级别的(只有同块内的会影响它,其它值都打到\(tag\)上了)!也就是空间只需要\(O(n)\)就够了。
而且如果我们把询问也拆成前缀的形式(其实本来就是前缀),那\(tag\)完全不需要直接打到\(1\sim p-1\)上,只需要在\(p\)上打就可以了。查询的时候维护一个\(tag\)的后缀和即可。
这样设块大小为\(B\),修改复杂度就只有\(O(B)\),查询复杂度还是\(O(B+\frac nB)\),但是某些修改比较多查询比较少的题目就可以调整\(B\)的大小解决啦。
虽然在这题复杂度依旧是\(O(n\sqrt n)\),但是常数不知道优秀到了哪里去。
中间过程(包括查询啊=-=)\(s\)的第二维可能是负的,但绝对值在\(\sqrt n\)内。
//233ms 2600KB
#include <cstdio>
#include <cctype>
#include <algorithm>
#define B 150
#define mod 998244353
#define Mod(x) x>=mod&&(x-=mod)
#define Add(x,v) (x+=v)>=mod&&(x-=mod)
#define gc() getchar()
typedef long long LL;
const int N=1e5+5,M=N/B+3;
int bel[N],f[N],g[N],tag[M],s[M][B+3<<1];
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
void Update(int p,int v)
{
int *s=::s[bel[p]];
for(int i=B; i<=B<<1; ++i) Add(s[i],v);
}
void Modify(int p,int v)
{
int bel=::bel[p],*s=::s[bel];
tag[bel]+=v;
for(int i=bel*B+1; i<=p; ++i)
{
if(v==1) Add(s[g[i]+B],mod-f[i-1]);
else Add(s[g[i]-1+B],f[i-1]), Add(s[g[i]-2+B],f[i-1]);
g[i]+=v;
}
}
int Query(int p,int K)
{
int bel=::bel[p],sum=tag[bel]; LL res=0;
for(int i=bel*B+1; i<=p; ++i) g[i]<=K&&(res+=f[i-1]);
while(bel--)
{
// assert(sum>=0);
// if(sum<=K) res+=s[bel][std::min(B<<1,K-sum+B)];//WA:sum may be >K
if(std::abs(sum-K)<=B) res+=s[bel][K-sum+B];
else if(sum<K) res+=s[bel][B<<1];
sum+=tag[bel];
}
return res%mod;
}
int main()
{
static int las[N],pre[N];
int n=read(),K=read();
for(int i=1; i<=n; ++i) bel[i]=(i-1)/B;
f[0]=1;
for(int i=1; i<=n; ++i)
{
int a=read(); las[i]=pre[a], pre[a]=i;
Update(i,f[i-1]), Modify(i,1);
if(las[i])
{
Modify(las[i],-2);
if(las[las[i]]) Modify(las[las[i]],1);
}
f[i]=Query(i,K);
}
printf("%d\n",f[n]);
return 0;
}
Codeforces.1029D.Isolation(DP 分块)的更多相关文章
- Codeforces 1129D - Isolation(分块优化 dp)
Codeforces 题目传送门 & 洛谷题目传送门 又独立切了道 *2900( 首先考虑 \(dp\),\(dp_i\) 表示以 \(i\) 为结尾的划分的方式,那么显然有转移 \(dp_i ...
- Codeforces Round #355 (Div. 2) D. Vanya and Treasure dp+分块
题目链接: http://codeforces.com/contest/677/problem/D 题意: 让你求最短的从start->...->1->...->2->. ...
- CF1129D Isolation(分块+DP)
一个很显然的DP方程式:f[i]=Σf[j],其中j<i且在[j+1,i]中出现1次的数不超过k个 乍一看挺神仙的,只会O(n^2),就是对于每个位置从后向前扫一遍,边扫边统计出现1次的数的个数 ...
- codeforces 682D(DP)
题目链接:http://codeforces.com/contest/682/problem/D 思路:dp[i][j][l][0]表示a串前i和b串前j利用a[i] == b[j]所得到的最长子序列 ...
- codeforces 666A (DP)
题目链接:http://codeforces.com/problemset/problem/666/A 思路:dp[i][0]表示第a[i-1]~a[i]组成的字符串是否可行,dp[i][1]表示第a ...
- Codeforces 176B (线性DP+字符串)
题目链接: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=28214 题目大意:源串有如下变形:每次将串切为两半,位置颠倒形成 ...
- Codeforces 55D (数位DP+离散化+数论)
题目链接: http://poj.org/problem?id=2117 题目大意:统计一个范围内数的个数,要求该数能被各位上的数整除.范围2^64. 解题思路: 一开始SB地开了10维数组记录情况. ...
- Codeforces 264B 数论+DP
题目链接:http://codeforces.com/problemset/problem/264/B 代码: #include<cstdio> #include<iostream& ...
- CodeForces 398B 概率DP 记忆化搜索
题目:http://codeforces.com/contest/398/problem/B 有点似曾相识的感觉,记忆中上次那个跟这个相似的 我是用了 暴力搜索过掉的,今天这个肯定不行了,dp方程想了 ...
随机推荐
- Docker相关安装和卸载
安装: 1.Docker要求CentOS系统的内核版本高于 3.10 ,通过 uname -r 命令查看你当前的内核版本是否支持安账docker 2.更新yum包:sudo yum update 3. ...
- OAuth2.0授权码模式
OAuth2.0简单说就是一种授权的协议,OAuth2.0在客户端与服务提供商之间,设置了一个授权层(authorization layer).客户端不能直接登录服务提供商,只能登录授权层,以此将用户 ...
- LeetCode28——实现strStr()
6月中下旬辞职在家,7 月份无聊的度过了一个月.8 月份开始和朋友两个人写项目,一个后台和一个 APP ,APP 需要对接蓝牙打印机.APP 和蓝牙打印机都没有搞过,开始打算使用 MUI 开发 APP ...
- git使用cherry-pick和revert抢救错误代码提交
大多数的新手在新接触git时都会出现这样的问题.代码写完了,提交到dev分支进行测试.一高兴忘记切回来,继续在dev分支开发,写完之后提交时猛的发现,我靠,我怎么在dev上面写代码,此时内心必然是一阵 ...
- 使用yarn来替代npm
Yarn是由Facebook.Google.Exponent 和 Tilde 联合推出了一个新的 JS 包管理工具 ,正如官方文档中写的,Yarn 是为了弥补 npm 的一些缺陷而出现的 安装yarn ...
- Python3.0的新特性
网上关于Python3与Python2的区别的文章都烂大街了,但基本上都是抄来抄去,为了追本溯源,直接看官网最靠谱,官网文档的结构性更强. 本文是对Python3.0官网文档 What's New I ...
- 解决微信浏览器缓存站点入口文件(IIS部署Vue项目)
最近开发的微信公众号项目中(项目采用Vue + Vux 构建,站点部署在IIS8.5上),遇到个非常奇葩的问题,发布站点内容后,通过微信打开网址发现是空白页面(后来验证是微信浏览器缓存了入口文件-in ...
- “金九银十”已过,总结我的天猫、蚂蚁、头条面试经历(Java岗)
跳槽时时刻刻都在发生,但是我建议大家跳槽之前,先想清楚为什么要跳槽.切不可跟风,看到同事一个个都走了,自己也盲目的开始面试起来(期间也没有准备充分),到底是因为技术原因(影响自己的发展,偏移自己规划的 ...
- Java自学-集合框架 泛型Generic
ArrayList上使用泛型 步骤 1 : 泛型 Generic 不指定泛型的容器,可以存放任何类型的元素 指定了泛型的容器,只能存放指定类型的元素以及其子类 package property; pu ...
- jQuery 的58种事件方法你都用过了吗
jQuery 事件方法 事件方法触发或将函数附加到所选元素的事件处理程序. 下表列出了用于处理事件的所有jQuery方法. 方法 描述 bind() 在3.0版中已弃用. 请改用on()方法.将事件处 ...