二分算法的应用——最大化平均值 POJ 2976 Dropping tests
最大化平均值
有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的更多相关文章
- POJ - 2976 Dropping tests && 0/1 分数规划
POJ - 2976 Dropping tests 你有 \(n\) 次考试成绩, 定义考试平均成绩为 \[\frac{\sum_{i = 1}^{n} a_{i}}{\sum_{i = 1}^{n} ...
- POJ 2976 Dropping tests 【01分数规划+二分】
题目链接:http://poj.org/problem?id=2976 Dropping tests Time Limit: 1000MS Memory Limit: 65536K Total S ...
- POJ - 2976 Dropping tests(01分数规划---二分(最大化平均值))
题意:有n组ai和bi,要求去掉k组,使下式值最大. 分析: 1.此题是典型的01分数规划. 01分数规划:给定两个数组,a[i]表示选取i的可以得到的价值,b[i]表示选取i的代价.x[i]=1代表 ...
- POJ 2976 Dropping tests【二分 最大化平均值】
题意:定义最大平均分为 (a1+a2+a3+---+an)/(b1+b2+---+bn),求任意去除k场考试的最大平均成绩 和挑战程序设计上面的最大化平均值的例子一样 判断是否存在x满足条件 (a1+ ...
- POJ 2976 Dropping tests (最大化平均值)
题目链接:click here~~ [题目大意]给你n个分数的值,要求最小不选k个,使得最后分数相加结果平均值最大 [解题思路]:最大化平均值:參见:click here~~ 代码: #include ...
- POJ 2976 Dropping tests 01分数规划 模板
Dropping tests Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6373 Accepted: 2198 ...
- POJ 2976 Dropping tests(01分数规划)
Dropping tests Time Limit: 1000MS Memory Limit: 65536K Total Submissions:17069 Accepted: 5925 De ...
- POJ 2976 Dropping tests(01分数规划入门)
Dropping tests Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 11367 Accepted: 3962 D ...
- POJ 2976 Dropping tests (0/1分数规划)
Dropping tests Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 4654 Accepted: 1587 De ...
随机推荐
- Unity3D游戏开发——收集当前关卡游戏中分散的物件
运用场景 许多游戏中会有一些供玩家拾起的物件,例如装备.血包.道具等.当玩家与物件进行碰撞后,则会进入仓库. 本篇介绍了简单的碰撞过程. 原理 基本的碰撞机制,用到OnTriggerEnter()碰撞 ...
- 团队作业7——第二次项目冲刺(Beta版本12.07——12.08)
1.当天站立式会议照片 本次会议在5号公寓3楼召开,本次会议内容:①:熟悉每个人想做的模块.②:根据项目要求还没做的完成. 2.每个人的工作 经过会议讨论后确定了每个人的分工 组员 任务 陈福鹏 实现 ...
- Java 线程结束 & 守护线程
/* 停止线程: 1,stop方法. 2,run方法结束. 怎么控制线程的任务结束呢? 任务中都会有循环结构,只要控制住循环就可以结束任务. 控制循环通常就用定义标记来完成. 但是如果线程处于了冻结状 ...
- VS提示“无法启动IIS Express Web服务器”的解决方法
有时在使用Visual Studio运行项目时,会提示“无法启动IIS Express Web服务器”,如图: 可以依次尝试以下方法(我的情况使用第一种就解决了): 1.可能原因:误操作执行了:Sc ...
- windows多线程(一) 创建线程 CreateThread
一 线程创建函数 CreateThread 修改说明: 这里 说了另一种创建线程方法,使用_beginthreadex()更安全的创建线程,在实际使用中尽量使用_beginthreadex()来创建线 ...
- 链表java实现
链表是用指针将多个节点联系在一起,通过头节点和尾节点还有节点数量,可以对链表进行一系列的操作.是线性表的链式存储实现. 1.链表是多个不连续的地址组成在一起根据指针链接在一起的,由多个节点组成,每个节 ...
- 配置SSH Forward提升安全性
目标 MacBook ---(SSH)---> BastionServer ---(SSH)---> RemoteServer 说明 BastionServer.RemoteServe ...
- P2261 [CQOI2007]余数求和
我是题面 题意还是很清晰,很容易理解 1e9范围明显不能暴力,除非你能把常数优化到\(\frac1 {10}\),但我实在想象不到用了这么多取模怎么把常数优化下去 我们可以把\(k\%i\)变成\(k ...
- P4611 [COCI2011-2012#7] TRAMPOLIN
题目背景 有很多超级英雄:蝙蝠侠,蜘蛛侠,超人等.其中,有一位叫牛.今天他想模仿蜘蛛侠,所以他选择了一排高大的摩天楼来跳. 题目描述 具体而言,他选择了一个由 N 个摩天大楼构成的序 列,从左到右编号 ...
- iPhone X 的原深感模组
物理与数字世界正走向融合,我们每天醒来的时间.睡眠时长.心率和步数等数据都会被分享.上传并转化为分析数据.无处不自的 AI.互联互通和软件平台将改变用户对现实的感知. 2018 年的 CES 展(国际 ...