接龙:一个前缀和优化 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 ] [ 前缀和 ]的更多相关文章

  1. CF10D-LCIS题解--线性DP+打印方案

    题目链接: https://www.luogu.org/problemnew/show/CF10D 方法一 分析 \(LCS\)和\(LIS\)已经成烂大街的知识了,可是当这两个合并起来成为\(LCI ...

  2. 上午小测3 T1 括号序列 && luogu P5658 [CSP/S 2019 D1T2] 括号树 题解

    前 言: 一直很想写这道括号树..毕竟是在去年折磨了我4个小时的题.... 上午小测3 T1 括号序列 前言: 原来这题是个dp啊...这几天出了好几道dp,我都没看出来,我竟然折磨菜. 考试的时候先 ...

  3. luogu P1549 棋盘问题(2) 题解

    luogu P1549 棋盘问题(2) 题解 题目描述 在\(N * N\)的棋盘上\((1≤N≤10)\),填入\(1,2,-,N^2\)共\(N^2\)个数,使得任意两个相邻的数之和为素数. 例如 ...

  4. CSP J/S 初赛总结

    CSP J/S 初赛总结 2021/9/19 19:29 用官方答案估计 J 涂卡的时候唯一的一支 2B 铅笔坏了,只能用笔芯一个个涂 选择 \(-6\ pts\) 判断 \(-3\ pts\) 回答 ...

  5. luogu2657-Windy数题解--数位DP

    题目链接 https://www.luogu.org/problemnew/show/P2657 分析 第一道数位DP题,发现有点意思 DP求\([L,R]\)区间内的XXX个数,很套路地想到前缀和, ...

  6. Blocks题解(区间dp)

    Blocks题解 区间dp 阅读体验...https://zybuluo.com/Junlier/note/1289712 很好的一道区间dp的题目(别问我怎么想到的) dp状态 其实这个题最难的地方 ...

  7. 模拟赛 提米树 题解 (DP+思维)

    题意: 有一棵棵提米树,满足这样的性质: 每个点上长了一定数量的Temmie 薄片,薄片数量记为这个点的权值,这些点被标记为 1 到 n 的整数,其 中 1 号点是树的根,没有孩子的点是树上的叶子. ...

  8. 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)

    洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...

  9. 方格取数(number) 题解(dp)

    题目链接 题目大意 给你n*m个方格,每个格子有对应的值 你从(1,1)出发到(n,m)每次只能往下往上往右,走过的点则不能走 求一条路线使得走过的路径的权值和最大 题目思路 如果只是简单的往下和往右 ...

  10. 2019 CSP J/S第2轮 视频与题解

    CSP入门组和提高组第二轮题解 转自网络

随机推荐

  1. Caused by: org.gradle.api.internal.plugins.PluginApplicationException: Faile

    解决方法: 1.新建一个安卓应用,复制下面路径红色框的代码  去替换  导入应用中的代码,就是修改gradle版本: 2.在导入的应用中如下路径添加信息 代码: android.overridePat ...

  2. moectf2023的一些题

    [moectf]GUI 这是一道win32窗口程序 找到线程的回调函数 sub_740C94这个函数里有很多函数,意义不明,实际并不会对输入的字符串做出改变 回调函数里下断点,然后单步动调 sub_7 ...

  3. C#向JAVA发送form-data文件问题处理方案

    前言 和上一篇文章一样,.NET 和 JAVA之间的接口请求又遇到了新问题 我们有一个用来接收文件的接口,外部把文件流.文件名.目录,传进来,我们系统把生成的附件ID反回去,接口为POST-form- ...

  4. ie浏览器设置允许跨域

    前情 在访问测试搭建的测试环境的时候,发现接口因为跨域全部失败了,服务端又不想设置允许跨域,又急于使用,于是想到是不是可以使用跨域浏览器,上一次已解决chrome允许跨域,这一次来设置IE允许跨域 放 ...

  5. 开启生态新姿势 | 使用 WordPress 远程附件存储到 COS

    在看到这篇文章前,你大概已经听说过 WordPress 了,它是使用 PHP 语言开发的博客平台,用户可以在支持 PHP 和 MySQL 数据库的服务器上架设属于自己的网站,也可以把 WordPres ...

  6. 聊一聊坑人的 C# MySql.Data SDK

    一:背景 1. 讲故事 为什么说这东西比较坑人呢?是因为最近一个月接到了两个dump,都反应程序卡死无响应,最后分析下来是因为线程饥饿导致,那什么原因导致的线程饥饿呢?进一步分析发现罪魁祸首是 MyS ...

  7. VB 的一些歧义(不断更新)

    foo . bar 它可能是 foo.bar() 也可能是 foo(withObj.bar). f (a) , b 它可能是 call f(a)._DEFAULT(a)(Missing, b) 也可能 ...

  8. X64\X86\X86-64的区别

    x86是指intel的开发的一种32位指令集,从386开始时代开始的,一直沿用至今,是一种cisc指令集,所有intel早期的cpu,amd早期的cpu都支持这种指令集,ntel官方文档里面称为&qu ...

  9. Qt编写地图综合应用3-省市区域图

    一.前言 省市区域图也可以叫省市轮廓图,就是将每个省份.市区的边界区域变成轮廓展示,只是个大概的轮廓,和真是的地图基本一致,毕竟都是一个个点堆起来的,可能会有很小很小的误差,之前做大屏系统中间那个中国 ...

  10. 创建用于预测序列的人工智能模型,用Keras Tuner探索模型的超参数。

    上一篇:<创建用于预测序列的人工智能模型(五),调整模型的超参数> 序言:在完成初步的模型研发后,接下来的重点是探索和优化超参数.通过合理调整超参数(如学习率.动量参数.神经元数量等),可 ...