• 【 Gym 101116K 】Mixing Bowls(dfs)


    BUPT2017 wintertraining(15) #4H
    Gym - 101116K

    题意

    给定一个菜谱,大写的单词代表混合物,小写的代表基础原料。每个混合物由其它混合物或基础原料组成,不会间接或直接地需要自己。制备每个混合物必须先制备组成它的混合物。且混合物都要用一个碗装,当它作为原料去合成其它混合物后,碗就空出来了。求合成第一个混合物最少需要用几个碗。

    题解

    假设合成一个混合物需要dfs(i)个碗。
    因为最后需要合成第一个混合物,所以我们以它为根进行dfs。
    假设当前混合物的原料里有t个混合物,其中第j个合成的混合物需要need[j]个碗,且还要j-1个碗装前面j-1个混合物,因此此时需要j-1+need[j]个碗。
    因此我们把need按从大到小排序,求出合成子混合物时所需的最大碗数,最后混合成当前混合物需要t+1个碗,取两者最大值即可。

    代码

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    #include <iostream>
    #define N 1005
    using namespace std;
    struct node{
    	char name[25];
    	char son[15][25];
    	int d;
    }a[N];
    int n;
    int s[N];
    bool cmp(int a,int b){
    	return a>b;
    }
    int dfs(int x){
    	int t=0,need[15];
    	memset(need,0,sizeof need);
    	for(int j=1;j<=a[x].d;j++)
    		if(a[x].son[j][0]>='A'&&a[x].son[j][0]<='Z'){
    			t++;
    			for(int i=1;i<=n;i++)if(strcmp(a[i].name,a[x].son[j])==0){
    				need[t]=s[i]?s[i]:dfs(i);
    				break;
    			}
    		}
    	sort(need+1,need+1+t,cmp);
    	int ans=t+1;
    	for(int j=1;j<=t;j++){
    		ans=max(j-1+need[j],ans);
    	}
    	return s[x]=ans;
    }
    int main() {
    	scanf("%d",&n);
    	for(int i=1;i<=n;i++){
    		scanf("%s%d",a[i].name,&a[i].d);
    		for(int j=1;j<=a[i].d;j++)
    			scanf("%s",a[i].son[j]);
    	}
    	printf("%d
    ",dfs(1));
    	return 0;
    }
    
  • 相关阅读:
    javascript-事件代理
    控制DIV属性——实现盒子长、宽、背景等变化
    mac环境下VSCODE 全局搜索无效的问题
    VSCODE snippets的使用
    JS单线程与setTimeout执行原理(转载)
    ES6
    SQL聚合函数
    MySQL函数(转发)
    关于mysql中int(1)中int后面的数字(转发)
    关于xcode8.0 VVDocumenter无法使用的解决办法
  • 原文地址:https://www.cnblogs.com/flipped/p/6440812.html
Copyright © 2020-2023  润新知