零.背景:

本人在考场上想出正解,因为一些奇怪的心态问题,并没有写代码(详细内容见本人CSP-J/S游记中所描述的),以写一篇题解,特此纪念。

一.状态定义与分析:

抓住题目的重点就是有颜色,有贡献(即值与值之间的累加),

所以考虑定义状态 \(dp_{i,j}(j\in\{0,1\})\) ,表示在第 \(i\) 位填入颜色 \(j\) 可以获得的最大贡献。所以可以很自然的得到如下方程(以颜色0为例):

\[dp_{i,0} = max\{dp_{i-1,0},dp_{i-1,1},dp_{j+1,1}+a_{i}+s_{i-1}-s_{j+1}\}
\]

我们定义:\(s_{i}\) 表示将区间 \([1,i]\) 染成同一种颜色可以获得的贡献,所以这是可以用前缀和预处理的。

前两个从 \(i-1\) 转移过来的部分其实就是表示不选前面相等的数所带来的最大贡献,所以最终的转移是 \(O(n^{2})\) 的,可以得到50分。

二.考虑优化:

如果你实在没法发现规律的话,你也可以通过上述方法输出dp值找规律,而这道题,其实只需要从离 \(a_{i}\) 最近的下标\(j\)使得 \((a_{i}=a_{j})\) 转移就可以了,这样从逻辑上也是说的通的,因为只有这样,我们才能把前面的最优状态是否选择全都包含完(发现没,又回到DP的本质了,无后效性,上一个状态一定能包含前面局面的整体最优,我们只需要考虑当前决策就可以了,dp真的很神奇)。

所以我们定义一个 \(last_{i}\) 表示最近的与 \(a_{i}\) 相同的元素的下标,所以方程优化为:

\[dp_{i,0} = max\{dp_{i-1,0},dp_{i-1,1},dp_{last_{i}+1,1}+a_{i}+s_{i-1}-s_{last_{i}+1}\}
\]

那怎么得到这样一个 \(last\) 数组呢?观察数据范围可以发现 \(a_{i}\le1e6\) 可以开一个桶(value数组)去存当前的最新值,即每次循环按如下方式更新:

\[last_{i}=value_{a_{i}}
\]
\[value_{a_{i}}=i
\]

所以又可以 \(O(n)\) 处理 \(O(1)\) 查询了,所以最后整个问题就被我们在 \(O(n)\) 的复杂度下解决了。真的不能再做一些小优化了吗,其实是可以的,如果我们思考填颜色这个过程,你会发现,蓝色与红色的填色过程本质上是完全对称等价的(输出也可以验证这一点),所以我们可以完全优化成为一个标标准准的一维线性dp。到此整道题本人的完整思路已经结束,接下来就是看代码实现了(其实并不复杂)。

三.代码实现:

代码仅供理解算法过程做参考,切勿复制。(码风良好,可读性强)

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<vector>
#define int long long inline int read() {
int x=0,f=1;char ch=getchar();
while(ch > '9' || ch < '0'){if(ch == '-'){f = -1;}ch = getchar();}
while(ch >= '0'&&ch <= '9'){x = x * 10 + ch - 48; ch = getchar();}
return x * f;
} template <typename type> inline void write(type x)
{
if (x < 0) putchar('-'), x = -x;
if (x > 9) write(x / 10);
putchar(x % 10 + 48);
return;
} const int N = 2e5 + 10;
const int M = 1e6 + 10;
int n,T,a[N],s[N],dp[N],last_same[N],val[M]; signed main()
{
T = read();
while(T--){
n = read();
for(int i = 1;i <= n;++i) {
a[i] = read();
val[a[i]] = -1;
} for (int i = 1;i <= n;++i) {
last_same[i] = val[a[i]];
val[a[i]] = i;
s[i] = s[i - 1];
if(a[i] == a[i - 1]) s[i] += a[i];
} for(int i = 1;i <= n;++i){
dp[i] = dp[i - 1];
if(last_same[i] != -1){
if(last_same[i] == i - 1) dp[i] += a[i];
else dp[i] = std::max(dp[i],dp[last_same[i] + 1] + a[i] + s[i - 1] - s[last_same[i] + 1]);
}
}
write(dp[n]);
putchar('\n');
}
return 0;
}

P11233 [CSP-S 2024] 染色 题解的更多相关文章

  1. BZOJ 4033: [HAOI2015]树上染色题解

    BZOJ 4033: [HAOI2015]树上染色题解(树形dp) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1327400 原题地址: BZOJ 403 ...

  2. BZOJ4033:[HAOI2015]树上染色——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4033 有一棵点数为N的树,树边有边权.给你一个在0~N之内的正整数K,你要在这棵树中选择K个点,将 ...

  3. BZOJ2243:[SDOI2011]染色——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2243 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点 ...

  4. BZOJ 2303: [Apio2011]方格染色 题解

    题目大意: 有n*m的方格,中间的数要么是1,要么是0,要求任意2*2的方格中的数异或和为1.已知一部分格子中的数,求合法的填数的方案数. 思路: 由题意得:a[i][j]^a[i][j+1]^a[i ...

  5. [SDOI2011]染色 题解

    题目大意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段) 思路: 树剖之后 ...

  6. 洛谷P2486 [SDOI2011]染色 题解 树链剖分+线段树

    题目链接:https://www.luogu.org/problem/P2486 首先这是一道树链剖分+线段树的题. 线段树部分和 codedecision P1112 区间连续段 一模一样,所以我们 ...

  7. bzoj 1006: 神奇的国度 MCS

    题目大意: 弦图的最小染色. 题解: 裸题. #include <vector> #include <cstdio> #include <cstring> #inc ...

  8. FFT/NTT中档题总结

    被DeepinC%怕了,把一些题放到这里来 T1Normal 其实这道题放到中档题也不太合适,个人感觉真的很难,机房里好像都是颓的题解 因为期望的可加性,把每个点的贡献单独处理,即求期望深度 考虑$y ...

  9. 洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈)

    洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1311990 原题地址:洛谷P1155 双栈排序 ...

  10. 【题解】[HAOI2018]染色(NTT+容斥/二项式反演)

    [题解][HAOI2018]染色(NTT+容斥/二项式反演) 可以直接写出式子: \[ f(x)={m \choose x}n!{(\dfrac 1 {(Sx)!})}^x(m-x)^{n-Sx}\d ...

随机推荐

  1. WindowsPE文件格式入门06.手写最小PE

    https://bpsend.net/thread-346-1-1.html 实现目标 实现目标:手写实现不大于 200 Byte大小的PE文件(又名:畸形PE/变形PE),要求MessageBox弹 ...

  2. C#判断当前时间是否在规定时间段范围内(二维数组超简版)

    直接上C#代码 TimeSpan nowTime = DateTime.Now.TimeOfDay; string[,] arr = { { "7:50", "8:10& ...

  3. 第1.3讲、什么是 Attention?——从点菜说起 🍽️

    假设你在一家餐厅点餐,菜单上有好多菜(句子中的每个单词),你现在要点菜(做决策),但不可能每道菜都吃一样多. 你会怎么做?你会: 看下菜单(获取所有信息) 根据自己的口味打分(我喜欢辣.不要太咸) 按 ...

  4. IntelliJ IDEA 设置代码自动补全不区分大小写

    在使用IntelliJ IDEA进行开发时,输入一个关键字时会有提示相关的类和方法名,但是默认是区分大小写的,如何让IDEA不区分大小写呢? IntelliJ IDEA版本:IntelliJ IDEA ...

  5. 利用Word制作Kindle用的6寸PDF电纸书

      在word中以自己喜好的字体.字形和行间距等排版,完成后设置布局: 页面(纸张)大小自定义为宽X高=9cm X 12cm 页边距设为4个0.5cm 不要插入页码,占空间 另存为PDF 导入kind ...

  6. WDA SEARCH step by step

    之前写了不少的东西,其实大多数都是给自己看的,我的习惯是把资料放到网上,用的时候直接看博客. 之前硬盘轻轻摔了一下,几年的资料没了,然后就再也不用硬盘了. 昨天有人突然问我关于WDA的问题,毕竟奇怪, ...

  7. 开源交流丨批流一体数据集成框架ChunJun数据传输模块详解分享

    课件获取:关注公众号"ChunJun",后台私信 "课件" 获得直播课件 视频回放:点击这里 ChengYing开源项目地址:github 丨 gitee 喜欢 ...

  8. sublime text 2 snippet 设置

    1.标准文件写法 <snippet> <content><![CDATA[ 你需要插入的代码片段${1:name} ]]></content> < ...

  9. Kong入门学习实践(3)路由转发

    最近在学习Kong网关,因此根据老习惯,我会将我的学习过程记录下来,一来体系化整理,二来作为笔记供将来翻看.由于我司会直接使用Kong企业版,学习过程中我会使用Kong开源版. 本篇,我们学习快速配置 ...

  10. JS ES6数组中删除指定元素

    https://www.cnblogs.com/linsx/p/8331623.html arr.splice(arr.findIndex(item => item.id === data.id ...