hdu3185 Dead Or Alive

本文介绍了一个关于生存资源分配的问题,通过枚举城市顺序并利用二进制操作来确定最大生存人数。具体方法包括使用二进制记录每个城市的限制条件及人群的分配要求,通过递归搜索来找出最优解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Dead Or Alive

Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 69    Accepted Submission(s): 6


Problem Description
Only several people survive after Doomsday, and a small number of cities have not been destroyed. yifenfei wants to arrange these people to live on the available cities.
Now all the people follow yifenfei to pass the cities.
But there are some restrictions on the cities. Some people will die when pass some cities and every city has a population limitation. If people can’t find a city to live after passing all cities they will be dead! So the order of passing the cities is important.
 

Input
The first line an integer T (1<=T<=100) which means the number of test cases.
Each case starts with two integers n and m, which means the number of people and the number of cities. (1<= n <= 31, 1 <= m <= 10)
Second line has m integers, the ith integer Ci means ith city’s population limitation. The cities are ordered from 0 to m-1. (1 <= Ci <= 6)
Then n lines follow. Each line starts with an integer k which means the number of cities this people will die when passing it. Then k integers follow.
 

Output
The maximum number of people can survive!
 

Sample Input

 
23 33 3 21 102 1 23 32 1 22 0 12 0 21 2
 
Sample Output

 
32
 

Source
 

Recommend

lcy   |   We have carefully selected several similar problems for you:   3190  3187  3184  3186  3188 


题解:不知为什么我之前过的人只有四个(有一个提交了2次),而且还是7年以前的。

这题我的算法可能不是正解,跑了7700多毫秒,其余4个人最多只跑了200多毫秒。

我们先枚举所有城市的顺序(用搜索搜出来),这有!m种。先用一个2进制记录下每个城市会死哪些人(用01表示记录即可),记为g[i],在搜索时我们用个2进制记录哪些人一定要放在这个城市前面(如果经过会死亡的话),0表示一定要放在前面,1表示不一定。

然后转移就是 now=la&(~g[i])  (2进制操作)  //la为上一个位置的放置要求,now为这个位置的放置要求。

最后取最小值即可


代码(前面讲的云里雾里的):

#include<bits/stdc++.h>
using namespace std;
int ans,n,m,gg[50],a[50],b[50],c[50];
long long g[50];
void dfs(int x,long long la,int k,int k1,int sum){
	long long now;
	int t1,kk;
	if(x>m){
		if(m-sum-k>0)k+=m-sum-k;
		ans=min(ans,k);
		return;
	}	
	for(int i=0;i<m;i++)
	 if(gg[i]==0){
	 	gg[i]=1;b[x]=i;
		now=la&(~g[i]);
		c[x]=now;
	 	t1=k1-__builtin_popcount(now);
	 	//printf("%d %d\n",i,t1);
	 	//k1:now中1的个数
		//t1:now和la中1相差的个数 
	 	if(n-k1+t1-k>sum){
	 		kk=n-k1+t1-k-sum;
	 		dfs(x+1,now,k+kk,k1-t1,sum+a[i]);
		 }
		  else
	 	dfs(x+1,now,k,k1-t1,sum+a[i]);
	 	gg[i]=0;
	 }
}
int main(){
	int t1,i,j,k1,t;
	scanf("%d",&t1);
	while(t1--){
		memset(gg,0,sizeof(gg));
		memset(g,0,sizeof(g));
		scanf("%d%d",&n,&m);
		ans=n;
		for(i=0;i<m;i++){
			scanf("%d",&a[i]);
		} 
		for(i=1;i<=n;i++){
			scanf("%d",&k1);
			for(j=1;j<=k1;j++){
				scanf("%d",&t);
				g[t]|=1<<(i-1);
			}
		}
		dfs(1,(1<<n)-1,0,n,0);
		printf("%d\n",n-ans);
	}
}

<think>好的,我需要帮助用户找到HDU OJ 1004问题的C语言解决方案。首先,我得回忆一下HDU 1004的具体内容。这个问题通常是关于统计出现次数最多的颜色或字符串。题目大意是输入多个测试用例,每个用例包含n个字符串,要求输出出现次数最多的那个。如果有多个并列最多,则按输入顺序输出最先出现的那个。 接下来,我需要确认用户的需求。他们明确提到需要C语言的实现方法,所以我要确保提供的代码是用C写的,并且符合题目的要求。同时,用户可能还希望了解解题思路,这样他们能更好地理解代码逻辑。 首先,我应该考虑如何高效统计每个字符串的出现次数。在C语言中,可以使用结构体数组来存储字符串及其出现次数。每次输入一个新的字符串时,遍历已有的结构体数组,检查是否已经存在。如果存在,则增加计数;如果不存在,则添加到数组中。最后,遍历整个数组找出出现次数最多的字符串。 接下来,需要注意内存管理。由于题目中每个字符串的最大长度可能有限,比如通常OJ题目中的字符串长度限制为100字符左右,所以需要为每个字符串分配足够的空间,比如char color[101]这样的数组。结构体数组的大小也需要足够大,比如假设最多有1000个不同的字符串,但具体可能需要根据题目要求调整。 然后,处理输入输出。HDU OJ的输入通常是多个测试用例,每个用例以n=0结束。在代码中,需要使用循环读取每个测试用例,直到n为0时停止。对于每个用例,先读取n,然后读取n个字符串。需要注意的是,每次处理完一个测试用例后,要重置结构体数组,避免上一个用例的数据影响当前结果。 在代码结构方面,主函数应该包含读取输入、处理每个测试用例的循环。处理每个字符串时,使用一个结构体数组来跟踪每个字符串的出现次数。使用strcmp来比较字符串,strcpy来复制字符串到结构体中。 此外,还需要考虑效率问题。如果n很大,比如达到1000,每次都要遍历整个数组,时间复杂度是O(n²),但在OJ中这可能仍然可以通过,因为实际测试数据可能不会太大。如果时间超限,可能需要优化数据结构,比如使用哈希表,但在C语言中实现哈希表较为复杂,对于新手来说可能较难理解,所以可能还是以简单方法为主。 然后,编写代码示例。需要包含必要的头文件,如stdio.h和string.h。定义结构体,比如struct ColorCount { char color[101]; int count; }; 然后声明结构体数组,并在每个测试用例开始时将其重置。 最后,测试代码的正确性。例如,输入多个测试用例,检查输出是否正确,特别是当有多个字符串出现次数相同的情况,是否输出最先出现的那个。 可能遇到的问题包括:字符串输入时的缓冲区问题,比如使用scanf读取字符串时可能会遇到空格,但题目中说明每个颜色是一个单独的字符串,所以可以使用scanf的%s格式符,它自动以空格或换行分隔。此外,结构体数组的索引需要在每次测试用例开始时重置为0,以确保每个用例独立处理。 总结一下,解题步骤为: 1. 读取测试用例数量n,直到n为0。 2. 对于每个用例,初始化结构体数组和索引。 3. 读取n个字符串,逐个处理,更新结构体数组中的计数。 4. 遍历结构体数组,找到出现次数最多的字符串。 5. 输出结果,并处理多个测试用例之间的换行要求。 现在,根据这些思路,编写C语言代码,并确保代码符合HDU OJ的要求,比如输入输出格式正确,没有多余的打印内容,处理多个测试用例等。</think>### HDU OJ 1004问题C语言解决方案 **问题描述** HDU OJ 1004要求统计多个测试用例中每个字符串出现的次数,输出出现次数最多的字符串。如果有多个并列最多的字符串,输出最先出现的那个。 --- #### 解题思路 1. **数据结构选择**:使用结构体数组存储字符串及其出现次数。 ```c struct ColorCount { char color[101]; // 存储字符串(假设最大长度为100) int count; // 出现次数 }; ``` 2. **输入处理**:循环读取测试用例,直到输入`n=0`时结束。 3. **字符串统计**: - 对每个新字符串,遍历结构体数组检查是否已存在。 - 若存在则增加计数,否则添加到数组中。 4. **结果输出**:遍历结构体数组,找到最大计数的字符串。 --- #### C语言实现代码 ```c #include <stdio.h> #include <string.h> #define MAX_STR_LEN 100 #define MAX_COLORS 1000 // 假设最多有1000种不同字符串 struct ColorCount { char color[MAX_STR_LEN + 1]; int count; }; int main() { int n; while (scanf("%d", &n) == 1 && n != 0) { struct ColorCount colors[MAX_COLORS]; int color_count = 0; // 当前已记录的不同字符串数量 for (int i = 0; i < n; i++) { char current_color[MAX_STR_LEN + 1]; scanf("%s", current_color); // 检查是否已存在 int found = 0; for (int j = 0; j < color_count; j++) { if (strcmp(colors[j].color, current_color) == 0) { colors[j].count++; found = 1; break; } } // 若不存在则添加 if (!found) { strcpy(colors[color_count].color, current_color); colors[color_count].count = 1; color_count++; } } // 查找出现次数最多的字符串 int max_count = 0; char result[MAX_STR_LEN + 1]; for (int i = 0; i < color_count; i++) { if (colors[i].count > max_count) { max_count = colors[i].count; strcpy(result, colors[i].color); } } printf("%s\n", result); } return 0; } ``` --- #### 代码说明 1. **输入处理**: - 使用`while (scanf("%d", &n) == 1 && n != 0)`循环读取测试用例。 - 每次循环初始化结构体数组`colors`和计数器`color_count`[^1]。 2. **字符串统计**: - 对每个输入的字符串,遍历结构体数组检查是否已存在(`strcmp`比较)。 - 不存在时添加到数组末尾,并初始化计数为1。 3. **性能优化**: - 时间复杂度为$O(n^2)$,适用于$n \leq 1000$的情况。 - 若$n$较大,可改用哈希表优化到$O(n)$,但实现复杂度较高。 4. **输出规则**: - 当多个字符串出现次数相同时,按输入顺序输出第一个出现的字符串(隐含逻辑:后添加的字符串不会覆盖先前的最大值)。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值