【题解】洛谷 P2476 [SCOI2008] 着色方案 [记搜]

P2476 [SCOI2008] 着色方案 - 洛谷 (luogu.com.cn)(支持升蓝)

相当难的一道题。。想了半小时毫无头绪。

由于本人特别喜欢品时间复杂度做题,一遇到这种记搜题就惨败。


如果 k 再小一点,那我会毫不犹豫的记搜。

但现在符合记搜要求大小的只有 ,考虑从 上下手。

又发现颜色并不重要 ,重要的只有上一次涂的什么颜色,这一次不要重复了。

考虑将所有颜色不分品种,只根据 的大小分成 5 种,设计个状态:

cpp 复制代码
dp[a][b][c][d][e][last]
// 表示某颜色只能涂 1 下的有 a 个,能涂 2 下的颜色 b 个,...,以此类推
// 上一次的涂的颜色是 last , 如 last = 4 表示上一次用的是能涂 4 下的颜色中的某一个

计算一下状态数:,实际还要少得多,可以过。

详见代码:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;

typedef long long LL;
const LL P = 1e9 + 7;
const int N = 17;

LL dp[N][N][N][N][N][6], inf;

LL dfs(int a, int b, int c, int d, int e, int last) {
	if (dp[a][b][c][d][e][last] != inf) {
		return dp[a][b][c][d][e][last];
	}
	if (a + b + c + d + e == 0) {
		return 1;
	}
	
	LL res = 0;
	if (a) {
		res = (res + (a - (last == 2)) * dfs(a - 1, b, c, d, e, 1)) % P;
		// 乘法原理,从所有只能涂一次的颜色中选一个,有 a 中种可能 
		// 选用能涂一下的颜色涂色,但是上一次用的是能涂 2 下的颜色
		// 那在这一次的时它会变成只能涂 1 下的颜色
		// 所以现在的 a 种能涂 1 下 的颜色,只能选 (a - 1) 种可能
	}
	if (b) {
		res = (res + (b - (last == 3)) * dfs(a + 1, b - 1, c, d, e, 2)) % P;
		// 涂 2 次的颜色少了一种,涂 1 次的颜色多了一种 
	}
	if (c) {
		res = (res + (c - (last == 4)) * dfs(a, b + 1, c - 1, d, e, 3)) % P;
	}
	if (d) {
		res = (res + (d - (last == 5)) * dfs(a, b, c + 1, d - 1, e, 4)) % P;
	}
	if (e) {
		res = (res + e * dfs(a, b, c, d + 1, e - 1, 5)) % P;
	}
	
	return dp[a][b][c][d][e][last] = res; 
}

int main () {
	ios::sync_with_stdio(false);
	cin.tie(0);
	
	memset(dp, -0x7f, sizeof(dp));
	inf = dp[0][0][0][0][0][0];
	
	int k; 
	cin >> k;
	int t[6] = {};
	for (int i = 1; i <= k; i ++) {
		int x;
		cin >> x;
		t[x] ++;
	}
	
	cout << dfs(t[1], t[2], t[3], t[4], t[5], 0) << "\n";
	
	return 0;
} 
相关推荐
YuTaoShao4 分钟前
【LeetCode 每日一题】1292. 元素和小于等于阈值的正方形的最大边长
算法·leetcode·职场和发展
Remember_9935 分钟前
【数据结构】深入理解Map和Set:从搜索树到哈希表的完整解析
java·开发语言·数据结构·算法·leetcode·哈希算法·散列表
浅念-6 分钟前
C++第一课
开发语言·c++·经验分享·笔记·学习·算法
charlie11451419110 分钟前
现代嵌入式C++教程:对象池(Object Pool)模式
开发语言·c++·学习·算法·嵌入式·现代c++·工程实践
HABuo12 分钟前
【linux进程控制(三)】进程程序替换&自己实现一个bash解释器
linux·服务器·c语言·c++·ubuntu·centos·bash
燃于AC之乐21 分钟前
我的算法修炼之路--8——预处理、滑窗优化、前缀和哈希同余,线性dp,图+并查集与逆向图
算法·哈希算法·图论·滑动窗口·哈希表·线性dp
格林威32 分钟前
多相机重叠视场目标关联:解决ID跳变与重复计数的 8 个核心策略,附 OpenCV+Halcon 实战代码!
人工智能·数码相机·opencv·算法·计算机视觉·分类·工业相机
一只小bit33 分钟前
Qt 多媒体:快速解决音视频播放问题
前端·c++·qt·音视频·cpp·页面
郝学胜-神的一滴33 分钟前
深入理解网络分层模型:数据封包与解包全解析
linux·开发语言·网络·程序人生·算法
永远都不秃头的程序员(互关)33 分钟前
【K-Means深度探索(九)】K-Means与数据预处理:特征缩放与降维的重要性!
算法·机器学习·kmeans