最大化平均值

有n个物品的重量和价值分别wi 和 vi。从中选出 k 个物品使得 单位重量 的价值最大。

限制条件:
<= k <= n <= ^
<= w_i <= v_i <= ^6 输入:
n = 3
k = 2
{W, V} = {(2,2), (5,3), (2,1)} 输出:
0.75 (如果选0号和2号,平均价格是 (2 + 1) / (2 + 2) = 0.75)

题解:

一般先想到的肯定是:把物品按照  单位价值  进行排序,然后从大到小贪心地进行选取。但是这个方法对应输入得到的 是 5/7=0.714。不可行。

转换成二分搜索的问题,由之前的博客中,这种题目关键就是 编写二分的条件C(x)。

  • C(x) = 可以选择使得 单位重量的价格 不小于 x

假设 n组数据,那他们 单位重量的价格是:

  • sum(vi) / sum(wi)

因此就变成了:

  • sum(vi) / sum(wi) >= x

转换为:

  • sum(vi - x * wi) >= 0

对 (vi - x * wi )的值进行排序贪心地进行选取。因此:

  • C(x) = ((vi - x*wi) 从大到小排列中的前  k 个的和不小于0),即说明改 x 可以达到 k 个物品的 单位重量的价值
  • 然后,就是用 二分搜索,来进行得到 最大满足这一条件的 x
#include <iostream>
#include <functional>
#include <algorithm>
using namespace std; /*
3
2
2 2 5 3 2 1
*/
const int maxn = + ;
int n, k;
int w[maxn], v[maxn];
double ave[maxn]; //判断是否满足条件
bool C(double x)
{
for (int i = ; i < n; i++)
{
ave[i] = v[i] - x * w[i];
}
//如果要自定义排序,里面别写成int了.....答案会出错...
sort(ave, ave + n, greater<double>()); // for (int i = 0; i < n; i++) {
// cout << ave[i] << " ";
// }
// cout << endl; //按从大到小取k个数求和
double sum = ;
for (int i = ; i < k; i++) {
sum += ave[i];
} //观察是否 可以取到 x为平均值
return sum >= ;
} void solve()
{
int INF = ;
cin >> n >> k;
for (int i = ; i < n; i++)
{
cin >> w[i] >> v[i];
INF += v[i];
} double lh = , rh = INF, mid;
for (int i = ; i < ; i++)
{
mid = (lh + rh) / 2.0;
if (C(mid)) {
lh = mid;
}
else {
rh = mid;
}
}
printf("%.2f\n", lh);
} int main()
{
solve();
return ;
}

习题:POJ 2976 Dropping tests

来源:http://poj.org/problem?id=2976

题意:题目就是说,n个题目你可以少做k个题目, 然后 n个题目分别为你得到的分数  ai, 和题目的分数 bi , 有个公式是:

, 要求就是:均值最大为多少,需要四舍五入。输入多组数据,n, k同时为0时,终止输入。和上面的例题属于一模一样的题目, 用二分解决最大化平均值的问题。

#include <iostream>
#include <iomanip>
#include <algorithm>
#include <functional>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std; const int maxn = + ;
typedef long long LL;
int n, k;
LL a[maxn], b[maxn];
double INF;
double ave[maxn]; bool C(double x)
{
for (int i = ; i < n; i++)
{
ave[i] = a[i] - x * b[i];
} //降序
sort(ave, ave + n, greater<double>()); double sum = ;
for (int i = ; i < n - k; i++)
{
sum += ave[i];
} return sum >= ;
} void solve()
{
double lh = , rh = INF, mid;
for (int i = ; i < ; i++)
{
mid = (lh + rh) / ;
if (C(mid))
{
lh = mid;
}
else
{
rh = mid;
}
}
//和下面那种都可以
printf("%.f\n", lh * );
// cout << fixed << setprecision(0) << lh * 100 << endl;
} int main()
{
while (cin >> n >> k && (n || k))
{
for (int i = ; i < n; i++) {
cin >> a[i];
INF += a[i];
}
for (int i = ; i < n; i++) {
cin >> b[i];
}
solve();
}
return ;
}

二分算法的应用——最大化平均值 POJ 2976 Dropping tests的更多相关文章

  1. POJ - 2976 Dropping tests && 0/1 分数规划

    POJ - 2976 Dropping tests 你有 \(n\) 次考试成绩, 定义考试平均成绩为 \[\frac{\sum_{i = 1}^{n} a_{i}}{\sum_{i = 1}^{n} ...

  2. POJ 2976 Dropping tests 【01分数规划+二分】

    题目链接:http://poj.org/problem?id=2976 Dropping tests Time Limit: 1000MS   Memory Limit: 65536K Total S ...

  3. POJ - 2976 Dropping tests(01分数规划---二分(最大化平均值))

    题意:有n组ai和bi,要求去掉k组,使下式值最大. 分析: 1.此题是典型的01分数规划. 01分数规划:给定两个数组,a[i]表示选取i的可以得到的价值,b[i]表示选取i的代价.x[i]=1代表 ...

  4. POJ 2976 Dropping tests【二分 最大化平均值】

    题意:定义最大平均分为 (a1+a2+a3+---+an)/(b1+b2+---+bn),求任意去除k场考试的最大平均成绩 和挑战程序设计上面的最大化平均值的例子一样 判断是否存在x满足条件 (a1+ ...

  5. POJ 2976 Dropping tests (最大化平均值)

    题目链接:click here~~ [题目大意]给你n个分数的值,要求最小不选k个,使得最后分数相加结果平均值最大 [解题思路]:最大化平均值:參见:click here~~ 代码: #include ...

  6. POJ 2976 Dropping tests 01分数规划 模板

    Dropping tests   Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 6373   Accepted: 2198 ...

  7. POJ 2976 Dropping tests(01分数规划)

    Dropping tests Time Limit: 1000MS   Memory Limit: 65536K Total Submissions:17069   Accepted: 5925 De ...

  8. POJ 2976 Dropping tests(01分数规划入门)

    Dropping tests Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 11367   Accepted: 3962 D ...

  9. POJ 2976 Dropping tests (0/1分数规划)

    Dropping tests Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4654   Accepted: 1587 De ...

随机推荐

  1. 第1阶段冲刺成果—简单运算game(APP)

    第1阶段冲刺成果 由于我们团队都没有Android的基础,所以在这一块花了很长的时间去学习探索,就连简单的Android的电脑配置也花了很长的时间,所以其他的DONE的都没有完成,这是失败的地方.但是 ...

  2. vue 过渡效果

    Vue中提供了`<transition>`和`<transition-group>`来为元素增加过渡动画.文档写的很清楚,但是实际使用起来还是费了一番功夫.这里做一个简单的记录 ...

  3. vue 笔记1

    created 钩子可以用来在一个实例被创建之后执行代码: new Vue({ data: { a: 1 }, created: function () { // `this` 指向 vm 实例 co ...

  4. java.lang.NoClassDefFoundError: org/hibernate/annotations/common/reflection/ClassLoadingException

    下载高版本的: hibernate-commons-annotations-5.0.1.Final.jar 低版本缺包

  5. jdk10配置

    解压 vi /etc/profile JAVA_HOME=/home/elasticsearch/jdk- CLASSPATH=$JAVA_HOME/lib/ PATH=$PATH:$JAVA_HOM ...

  6. 血液检测 & 创业骗局

    血液检测 & 创业骗局 硅谷血液检测公司 Theranos http://www.sohu.com/a/236659372_100053377 https://www.jianshu.com/ ...

  7. Java乐观锁、悲观锁

    乐观锁 乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号 ...

  8. Embarcadero Delphi 7 Enterprise 7.0.4.453 中文版

    在 win7下可能报错请大家按下面的方式 delphi7运行不正常的提示unable to rename'c:\program files\Borland\delphi7\Bin\delphi32.$ ...

  9. properties文件读取与写入

    将peoperties文件的读取和写入封装成了一个工具类: import java.io.BufferedInputStream; import java.io.FileInputStream; im ...

  10. P3835 【模板】可持久化平衡树

    题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作) 查询x数的 ...