零.背景:

本人在考场上想出正解,因为一些奇怪的心态问题,并没有写代码(详细内容见本人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. SuperSocket 2.0 的发布标志着.NET Socket 服务器框架迈入了一个全新的时代

    2025 年 4 月 20 日 - SuperSocket 团队自豪地宣布正式发布 SuperSocket 2.0,这是这款适用于 .NET 的高性能.可扩展套接字服务器应用程序框架的一个重要里程碑. ...

  2. Innosetup 安装 VC_redist 运行时库

    #普通安装vc_redis.x86.exe(会提示用户做出选择),在innosetup的[Run]属性中添加下面这一行 Filename: "{app}VC_redist.x86.exe&q ...

  3. 精选 12 款开源、免费、美观的 Vue 后台管理系统模板!

    前言 在当前软件开发领域,Vue.js 凭借其高效.灵活和易于上手的特性,成为了前端开发的热门选择.对于需要快速搭建企业级后台管理系统的开发者而言,使用现成的 Vue 后台管理系统模板无疑是一个明智之 ...

  4. HarmonyOS运动开发:精准估算室内运动的距离、速度与步幅

    前言 在室内运动场景中,由于缺乏 GPS 信号,传统的基于卫星定位的运动数据追踪方法无法使用.因此,如何准确估算室内运动的距离.速度和步幅,成为了运动应用开发中的一个重要挑战.本文将结合鸿蒙(Harm ...

  5. 2023人形全能赛v831代码(包括YOLOv2识别和扫码以及颜色识别)

    v831 import time, math from maix import nn, camera, display, image import serial class YOLOv2: def _ ...

  6. Java安全_RCE漏洞

    [!NOTE] 本次学习使用开源项目: https://github.com/JoyChou93/java-sec-code/blob/master/src/main/java/org/joychou ...

  7. 🚀 开源提示词优化神器来了!一键优化Function Calling和MCP提示词,让你的AI应用性能飞跃

    还在为Function Calling调用不准确而头疼?MCP提示词写得不够规范?今天给大家推荐一个开源的提示词优化平台,专门解决这些痛点! 背景:为什么需要专业的提示词优化? 在AI应用开发中,我们 ...

  8. 基于 C# 编写的轻量级工控网关和 SCADA 组态软件

    前言 今天大姚给大家分享一个基于 C# 编写.开源(LGPL-3.0 license)的轻量级工控网关和 SCADA 组态软件:SharpSCADA. 项目介绍 SharpSCADA是一个基于 C# ...

  9. Java 常用线程池

    快捷创建线程池方式 Java通过Executors提供四种线程池,分别为: 1.newCachedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需求,可以灵活回收空闲线程,若无可回收 ...

  10. Spark知识点汇总

    一.Spark架构设计 二.Spark常用算子 tips1: 数据处理的生命周期tips2: repartition vs coalesce区别: 可以使用 repartition 算子随意调整(提升 ...