Description
很久很久以前,巨龙突然出现,带来灾难带走了公主又消失不见,王国十分危险,世间谁最勇敢,一位勇者赶来大声喊:“我要带上最好的剑,翻过最高的山,闯进最深的森林,把公主带回到面前”……最后,英雄 达拉崩巴斑得贝迪卜多比鲁翁 ,他战胜了巨龙 昆图库塔卡提考特苏瓦西拉松 ,国王把公主 米娅莫拉苏娜丹妮谢莉红 嫁给了 达拉崩巴斑得贝迪卜多比鲁翁 。
于是国王第二天要在 蒙达鲁克硫斯伯古比奇巴勒城 举办酒宴,一共准备了 N 桶酒。此时一位忠诚的仆人来报,有人在其中一桶酒里下毒,并且此毒要经过24小时后才会毒发身亡。现在距离晚会酒宴开始还有正好24小时,为在酒宴开始前找出哪一桶酒有毒,国王决定找一些小白鼠来试酒。(仆人不可能说谎,即一定有且只有一桶酒有毒)
聪明的 达拉崩巴斑得贝迪卜多比鲁翁 本着善良的天性,决定用尽量少的小白鼠来试酒。
请问最少需要多少只小白鼠试酒?
Input
样例输入有多组。
第一行输入整数 T,表示有 T 组用例,
接下来 T 行, 对于魅族测试用例输入一个正整数 n 表示一共有 n 桶酒。
Output
对于每个用例输出一行一个数字,代表所需的最少小白鼠数量。
Sourse
北京理工大学2018年暑假集训算法总结赛 - Problem D
测试输入 | 期待的输出 | 时间限制 | 内存限制 | 额外进程 | |
---|---|---|---|---|---|
测试用例 1 | 以文本方式显示
| 以文本方式显示
| 1秒 | 64M | 0 |
思路
考虑使用二进制编号法,对酒桶从0开始编号,每一只小白鼠可以确定一个二进制位,因此只需考虑最后一个编号所对应的二进制数的位数即可。
可以通过哪几只小鼠死亡,反推出酒桶的编号(即死亡小鼠所对应的二进制位为1)。
关于二进制编号法最优的说明:每一只小白鼠可以有死活两种状态,所以n只小白鼠最多有2^n种可能状态,最多能判断2^n个桶里有毒的编号;又因为二进制编号后能判断2^n个编号,因此存在且最优。
对于输入酒的桶数n,需要 (向上取整) 只小白鼠。
注意事项
- 头文件包括stdio.h和math.h
- c语言中 log 函数默认以e为底,计算2为底数需要用到换底公式
- 注意n的数量级,建议使用long long
- 在部分特殊情况向上取整失效时,可以以double类型读入和处理整数
- number为非整数时,要转换为类型int并且+1
- 当只有一个桶时,由于有且仅有一个桶有毒,所以该桶一定有毒,需要试毒0次。
整体代码
#include <stdio.h>
#include <math.h>
int main() {
int t;
long long n;
scanf("%d", &t);
double number;
while (t--) {
scanf("%lld", &n);
number= log(n) / log(2);
if (number!= (int)number) number++;
printf("%d\n", (int)number);
}
return 0;
}