Luogu P11230 CSP-J 2024 接龙 题解 [ 线性 dp ] [ 前缀和 ]
接龙:一个前缀和优化 dp 或者单调队列优化 dp 的题目。
怎么周围的人都秒了 T3 不会 T4 啊,只有我觉得 T4 很套路,T3 比较难写吗。
暴力 dp
为了避免多维的状态定义,我们把每个人的子序列化为长度最多为 \(2\times 10^5\) 的一维序列,并记录下每一张牌对应的人是谁。记这个一维序列的长度为 \(tot\),显然 \(tot=\sum_{i=1}^{n} l_i\)。
我们定义状态 \(dp_{i,j}\) 表示当前进行到第 \(i\) 轮,且该轮以一维序列中的第 \(j\) 张牌结尾是否可行。
有一个显然的转移,我们遍历第 \(j\) 张牌的前面 \(k-1\) 张牌(必须要是同一个人的),假设某张前面的牌所写的数字为 \(x\),那么我们可以遍历 \(r-1\) 层中不是同一个人的且值为 \(x\) 的人的牌转移过来,只要这些牌中有一个的结果为 \(1\),那么第 \(j\) 张牌的值就是 \(1\)。
时间复杂度 \(O(rn^2k)\)。
第一步优化
注意到我们的限制有一个是“不能从同一个人那里转移过来”,考虑一个和前几年提高组假期计划那题很像的想法。
原题中是记录前 \(3\) 大的值,目的是避免走到重复的景点。而这题我们可以借鉴它的思路,记录两个属于不同人的且可以转移过来牌。只要这两张牌中,有一张和我现在的牌所属的人不同,就是可以转移过来的。
这个可以通过在每一轮结束之后预处理一遍得到。
时间复杂度 \(O(rnk)\)。
第二步优化
到这里我们就快做完这题了。观察到一个数的结果为 \(1\),当且仅当前 \(k-1\) 位的预处理值中有一个可以转移。熟悉 dp 的人肯定能一下就想到单调队列优化,进行转移,时间复杂度就被降到了 \(O(rn)\)。
但这是普及组,不能考单调队列。于是我们换一种普及组思路。因为这题可以预处理,而查询操作是查询 \(l\) 到 \(r\) 之间有没有元素 \(1\),因此我们预处理之后前缀和一下就能做到 \(O(1)\) 查询有没有元素 \(1\) 了。
时间复杂度 \(O(rn)\)。
代码
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
int t,n,k,q,f[105][200005][2],a[200005],bl[200005],tot,can[200005];
bitset<200005>dp[105];
void init()
{
memset(f,-1,sizeof(f));
for(int r=1;r<=100;r++)
{
//count can
if(r==1)
{
for(int i=1;i<=tot;i++)can[i]=can[i-1]+(a[i]==1);
}
else
{
for(int i=1;i<=tot;i++)can[i]=can[i-1]+((f[r-1][a[i]][0]!=-1&&f[r-1][a[i]][0]!=bl[i])||(f[r-1][a[i]][1]!=-1&&f[r-1][a[i]][1]!=bl[i]));
}
//count dp
int pre=0;
for(int i=1;i<=tot;i++)
{
if(bl[i]!=bl[i-1])pre=i;
int rn=i-1;
int ln=max(pre,i-k+1);
dp[r][i]=((can[rn]-can[ln-1])>0);
}
//count f
for(int i=1;i<=tot;i++)
{
if(dp[r][i]==0)continue;
if(f[r][a[i]][0]==-1)f[r][a[i]][0]=bl[i];
else if(f[r][a[i]][1]==-1&&f[r][a[i]][0]!=bl[i])f[r][a[i]][1]=bl[i];
}
// for(int i=1;i<=tot;i++)
// {
// cout<<"can["<<r<<"]["<<i<<"]="<<can[i]<<endl;
// }
// for(int i=1;i<=tot;i++)
// {
// cout<<"f["<<r<<"]["<<i<<"][0]="<<f[r][i][0]<<endl;
// cout<<"f["<<r<<"]["<<i<<"][0]="<<f[r][i][1]<<endl;
// }
}
}
void solve()
{
scanf("%d%d%d",&n,&k,&q);
tot=0;
for(int i=1;i<=n;i++)
{
int l;
scanf("%d",&l);
for(int j=1;j<=l;j++)
{
tot++;
scanf("%d",&a[tot]);
bl[tot]=i;
}
}
init();
while(q--)
{
int r,c;
scanf("%d%d",&r,&c);
printf("%d\n",(f[r][c][0]!=-1||f[r][c][1]!=-1));
}
}
int main()
{
freopen("chain.in","r",stdin);
freopen("chain.out","w",stdout);
scanf("%d",&t);
while(t--)solve();
return 0;
}
Luogu P11230 CSP-J 2024 接龙 题解 [ 线性 dp ] [ 前缀和 ]的更多相关文章
- CF10D-LCIS题解--线性DP+打印方案
题目链接: https://www.luogu.org/problemnew/show/CF10D 方法一 分析 \(LCS\)和\(LIS\)已经成烂大街的知识了,可是当这两个合并起来成为\(LCI ...
- 上午小测3 T1 括号序列 && luogu P5658 [CSP/S 2019 D1T2] 括号树 题解
前 言: 一直很想写这道括号树..毕竟是在去年折磨了我4个小时的题.... 上午小测3 T1 括号序列 前言: 原来这题是个dp啊...这几天出了好几道dp,我都没看出来,我竟然折磨菜. 考试的时候先 ...
- luogu P1549 棋盘问题(2) 题解
luogu P1549 棋盘问题(2) 题解 题目描述 在\(N * N\)的棋盘上\((1≤N≤10)\),填入\(1,2,-,N^2\)共\(N^2\)个数,使得任意两个相邻的数之和为素数. 例如 ...
- CSP J/S 初赛总结
CSP J/S 初赛总结 2021/9/19 19:29 用官方答案估计 J 涂卡的时候唯一的一支 2B 铅笔坏了,只能用笔芯一个个涂 选择 \(-6\ pts\) 判断 \(-3\ pts\) 回答 ...
- luogu2657-Windy数题解--数位DP
题目链接 https://www.luogu.org/problemnew/show/P2657 分析 第一道数位DP题,发现有点意思 DP求\([L,R]\)区间内的XXX个数,很套路地想到前缀和, ...
- Blocks题解(区间dp)
Blocks题解 区间dp 阅读体验...https://zybuluo.com/Junlier/note/1289712 很好的一道区间dp的题目(别问我怎么想到的) dp状态 其实这个题最难的地方 ...
- 模拟赛 提米树 题解 (DP+思维)
题意: 有一棵棵提米树,满足这样的性质: 每个点上长了一定数量的Temmie 薄片,薄片数量记为这个点的权值,这些点被标记为 1 到 n 的整数,其 中 1 号点是树的根,没有孩子的点是树上的叶子. ...
- 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)
洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...
- 方格取数(number) 题解(dp)
题目链接 题目大意 给你n*m个方格,每个格子有对应的值 你从(1,1)出发到(n,m)每次只能往下往上往右,走过的点则不能走 求一条路线使得走过的路径的权值和最大 题目思路 如果只是简单的往下和往右 ...
- 2019 CSP J/S第2轮 视频与题解
CSP入门组和提高组第二轮题解 转自网络
随机推荐
- Go Vue3 CMS管理后台(前后端分离模式)
本后台使用前后端分离模式开发,前端UI为Vue3+Ant Design Vue,后端Api为Go+Gin,解耦前后端逻辑,使开发更专注 技术栈 前端:Vue3,Ant Design Vue,Axios ...
- Javascript 常用封装(二)
1.字符串占位宽度 计算占位宽度:字符串的占位宽度除了涉及到具体的字符串内容,还与字体大小有关,可以将其放入Dom中来获取实际占位宽度 //计算字符串的占位宽度 function getTextWid ...
- iOS自动化打包输出工具
自动化打包输出工具 做开发的小伙伴有时候会接到自动化打包的需求,公司一般是要求根据一个配置文件来实现自动化配置iOS项目,比如往Xcode工程添加或修改代码.添加Framework.library.S ...
- 【论文系列】PPO知识点梳理 (尽我可能细致通俗理解!)
零.题记 这篇博客一方面为了记录当前的知识点,另一方面PPO算法实在是太重要了,不但要从理论上理解它到底是怎样实现的,还需要从代码方面进行学习和记录,这里我就通俗的将这个知识点进行简单的记录,用来日后 ...
- CVE-2023-32233 Linux 内核 UAF 漏洞分析与利用
Linux 内核 nftable 模块在处理匿名 set 时存在 UAF. 漏洞分析 漏洞成因是 nf_tables_deactivate_set 在释放匿名 set 时没有将 set 的标记设 ...
- 【分块】LibreOJ 6281 数列分块入门5
前言 对一个 int 类型的非负整数进行开方下取整,最多只会开方四次大小就不会再发生变化.一个大于 \(0\) 的正整数开方下取整最后的结果比如是 \(1\),而 \(1\) 开方的结果仍然会是 \( ...
- 【Amadeus原创】域用户完美执行应用程序
企业环境中,为了安全起见一般都没有赋予域用户或者企业的PC客户端用户管理员权限. 但偶尔会有个别的程序一定需要管理员身份才能执行,如财务某些程序或专业的应用程序.那么如何不赋予用户管理员权限及密码但又 ...
- showModalBottomSheet setState 无法更新ui和高度设置问题
showModalBottomSheet setState 无法更新ui问题 首先理解showModalBottomSheet,本质上可以理解为路由入栈,在show之后弹出的页面其实是另一个页面了,此 ...
- 在 .NET 下,Fiddler 不再抓取 Web Service 流量问题
在 .NET 下,Fiddler 不再抓取 Web Service 流量问题 问题现象 原来的一个应用中,需要访问 SOAP 服务.在原来的 .NET Framework 版本中,使用 Fiddler ...
- 在 MySQL 创造类似 PipelineDB 的流视图(continuous view)
公司的系统采用的是 Google Cloud SQL 提供的 MySQL 数据库,由于历史原因,数据库成本极高,需要对它进行优化缩减成本. 相比 PostgresSQL,MySQL 主要缺少以下特性, ...