列表

详情


NC20143. [JLOI2015]战争调度

描述

 脸哥最近来到了一个神奇的王国,王国里的公民每个公民有两个下属或者没有下属,这种关系刚好组成一个 n 层的完全二叉树。公民 i 的下属是 2 * i 和 2 * i +1。
最下层的公民即叶子 节点的公民是平民,平民没有下属,最上层的是国王,中间是各级贵族。现在这个王国爆发了战争,国王需要决定每一个平民是去种地以供应粮食还是参加战争,每一个贵族(包括国王自 己)是去管理后勤还是领兵打仗。
一个平民会对他的所有直系上司有贡献度,若一个平民 i 参加战争,他的某个直系上司 j 领兵打仗,那么这个平民对上司的作战贡献度为 wij。若一个平民 i 种地,他的某个直系上司 j 管理后勤,那么这个平民对上司的后勤贡献度为 fij,若 i 和 j 所参加的事务不同,则没有贡献度。
为了战争需要保障后勤,国王还要求不多于 m 个平民参加 战争。国王想要使整个王国所有贵族得到的贡献度最大,并把这件事交给了脸哥。但不幸的是, 脸哥还有很多 deadline 没有完成,他只能把这件事又转交给你。你能帮他安排吗?

输入描述

第一行两个数 n;m。
接下来 2^(n-1) 行,每行n-1 个数,第 i 行表示编号为 2^(n-1)-1+ i 的平民对其n-1直系上司的作战贡献度,其中第一个数表示对第一级直系上司,即编号为 (2^(n-1)-1+ i)/2 的贵族的作战贡献度 wij,依次往上。
接下来 2^(n-1)行,每行n-1个数,第i行表示编号为 2^(n-1)-1+ i的平民对其n-1个直系上司的后勤贡献度,其中第一个数表示对第一级直系上司,即编号为 (2^(n-1)-1+ i)/2 的贵族的后勤贡献度 fij ,依次往上。

输出描述

一行一个数表示满足条件的最大贡献值

示例1

输入:

3 4
503 1082
1271 369
303 1135
749 1289
100 54
837 826
947 699
216 389

输出:

6701

原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++11(clang++ 3.9) 解法, 执行用时: 19ms, 内存消耗: 8676K, 提交时间: 2019-04-15 13:59:05

#include<bits/stdc++.h>
using namespace std;
const int N=2005;
int n,m,w[N][N],f[N][N],dp[N][N],vis[N];
void dfs(int x,int y)
{
	for(int i=0;i<=(1<<y);i++)dp[x][i]=0;
	if(!y)
	{
		for(int i=1;i<=n;i++)
			if(vis[i])dp[x][1]+=w[x][i];
			else dp[x][0]+=f[x][i];
		return;
	}
    vis[y]=0;
	dfs(x*2,y-1);
	dfs(x*2+1,y-1);
    for(int i=0;i<=1<<(y-1);i++) 
    	for(int j=0;j<=1<<(y-1);j++) 
        	dp[x][i+j]=max(dp[x][i+j],dp[x*2][i]+dp[x*2+1][j]);
    vis[y]=1;
	dfs(x*2,y-1);
	dfs(x*2+1,y-1);
    for(int i=0;i<=1<<(y-1);i++) 
      	for(int j=0;j<=1<<(y-1);j++)
       		dp[x][i+j]=max(dp[x][i+j],dp[x*2][i]+dp[x*2+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",&w[i+(1<<n)][j]);
	for(int i=0;i<(1<<n);i++)
		for(int j=1;j<=n;j++)scanf("%d",&f[i+(1<<n)][j]);
	dfs(1,n);
	int ans=0;
	for(int i=0;i<=m;i++)ans=max(ans,dp[1][i]);
	cout<<ans;
	return 0;
}

上一题