题目描述
Wonderland居民决定举行一届地区性程序设计大赛。仲裁委员会志愿负责这次赛事并且保证会组织一次有史以来最公正的比赛。为此,所有参赛者的电脑和网络中心会以星状网络连接,也就是说,对每个参赛者,组委会会用一根长度一定的网线将他的计算机与中心连接,使得他们到网络中心的距离相等。
为了买网线,组委会与当地的网络公司联系,要向他们购买一定数目的等长网线,这些网线要尽可能的长,使得组织者可以让选手们彼此远离。
于是公司指派管理网线事务的负责人解决此事。负责人清楚地知道仓库里每根网线的长度(精确到厘米:cm),他也可以将他们以厘米的精度切割——前提是他得知道切成多长。但是现在,这个长度他算不出来,于是他彻底迷茫了。
你要做的,就是帮助困惑的负责人。编一个程序求出为了得到一定数目的等长网线,每根网线最大的可能长度。
输入格式
第一行由两个整数N和K组成,由一个空格间隔。N(1≤N≤10000)是仓库里光缆的数目,K(1≤K≤10000)是需要的网线数目。
接下来的N行每行只有一个实数,告诉你每根缆线的长度(单位:m)。这些网线至少长1m,最多不超过100km。
所有的长度精确到cm,且小数点后有且仅有两位。
输出格式
输出你求得的最大网线长度 (单位:m)。长度要精确到cm,并且输出时小数点后要恰有两位。
如果无论如何也不可能切割出需要数目的网线(每根至少1cm长),那么就输出“0.00”(不包括引号)。
输入样例
4 11
8.02
7.43
4.57
5.39
输出样例
2.00
老样子......找规律!!!!!!!!!!
像观察可不可以二分(最近发现二分好简单)
这么说了,显然是可以,那么~
自己画图去!!!!!
#include<iostream>
using namespace std;
double line[10000+10];
int n, k;
int l, r, mid;
提前准备好一个数组,和几个变量
int main()
{
cin >> n >> k;
for(int i=1;i<=n;i++){
scanf("%lf", &line[i]);
line[i]*=100;
if(line[i]>r) r = line[i];
}
l = 0;
r++;
int sum;
while(r - l > 1){
sum=0;
mid = l + (r - l)/2;
for(int i=1;i<=n;i++){
sum += line[i]/mid;
}
if(sum >= k) l = mid;
else r = mid;
}
double ans = l;
ans = ans * 1.0 / 100;
printf("%.2lf", ans);
return 0;
}
程序进行判断二分+实际计算结果
while以上的都是老样子,可以看我其他文章去学习二分
来吧完整程序
#include<iostream>
using namespace std;
double line[10000+10];
int n, k;
int l, r, mid;
int main()
{
cin >> n >> k;
for(int i=1;i<=n;i++){
scanf("%lf", &line[i]);
line[i]*=100;
if(line[i]>r) r = line[i];
}
l = 0;
r++;
int sum;
while(r - l > 1){
sum=0;
mid = l + (r - l)/2;
for(int i=1;i<=n;i++){
sum += line[i]/mid;
}
if(sum >= k) l = mid;
else r = mid;
}
double ans = l;
ans = ans * 1.0 / 100;
printf("%.2lf", ans);
return 0;
}
其实个人认为不用写函数,但只想偷个小懒的可以去我其他文章里找,下次直接套用就好了