Solution -「JLOI 2015」「洛谷 P3262」战争调度
Link.
给定一棵 层的完全二叉树,你把每个结点染成黑色或白色,满足黑色叶子个数不超过 。对于一个叶子 ,若其 级父亲与其同为黑色,则对答案贡献 ;若同为白色,则对答案贡献 。求最大贡献和。
。
想要 DP,比如令 表示 子树内有 个叶子为黑色时的最大贡献和。但发现这根本没法转移 qwq。
那……爆搜呢?
从上往下搜索,直接钦定当前非叶结点是黑是白,搜到叶子时,在向上计算当前叶子是黑色/白色时的贡献,回溯时简单背包。复杂度 ,可过欸!
/* Clearink */
#include <cstdio>
const int MAXN = 10;
int n, m, a[1 << MAXN | 5][MAXN + 5], b[1 << MAXN | 5][MAXN + 5];
int f[1 << MAXN | 5][1 << MAXN | 5];
bool fight[1 << MAXN | 5];
inline void chkmax ( int& a, const int b ) { a < b && ( a = b, 0 ); }
inline void solve ( const int u, const int d ) {
for ( int i = 0; i <= 1 << d; ++ i ) f[u][i] = 0;
if ( !d ) {
for ( int i = 1; i <= n; ++ i ) {
if ( fight[u >> i] ) f[u][1] += a[u][i];
else f[u][0] += b[u][i];
}
} else {
for ( int k = 0; k <= 1; ++ k ) {
fight[u] = k;
solve ( u << 1, d - 1 ), solve ( u << 1 | 1, d - 1 );
for ( int i = 0; i <= 1 << d >> 1; ++ i ) {
for ( int j = 0; j <= 1 << d >> 1; ++ j ) {
chkmax ( f[u][i + j], f[u << 1][i] + f[u << 1 | 1][j] );
}
}
}
}
}
int main () {
scanf ( "%d %d", &n, &m ), -- n;
for ( int i = 0; i < 1 << n; ++ i ) {
for ( int j = 1; j <= n; ++ j ) {
scanf ( "%d", &a[( 1 << n ) + i][j] );
}
}
for ( int i = 0; i < 1 << n; ++ i ) {
for ( int j = 1; j <= n; ++ j ) {
scanf ( "%d", &b[( 1 << n ) + i][j] );
}
}
solve ( 1, n );
int ans = 0;
for ( int i = 0; i <= m; ++ i ) chkmax ( ans, f[1][i] );
printf ( "%d\n", ans );
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现