P11233 [CSP-S 2024] 染色 题解
零.背景:
本人在考场上想出正解,因为一些奇怪的心态问题,并没有写代码(详细内容见本人CSP-J/S游记中所描述的),以写一篇题解,特此纪念。
一.状态定义与分析:
抓住题目的重点就是有颜色,有贡献(即值与值之间的累加),
所以考虑定义状态
我们定义:
前两个从
二.考虑优化:
如果你实在没法发现规律的话,你也可以通过上述方法输出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;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现