uva 1467 - Installations( 贪心+暴力)
2014-01-25 00:48
211 查看
题目链接:uva 1467 - Installations
题目大意:给出n个任务,每个任务有所需时间和截止时间,单个任务超过截止时间要罚款,求一个完成任务的序列,使得最大罚款和第二大罚款数之和最小。
解题思路:一开始想用二分求解最大罚款值,后来发现不靠谱。首先贪心,按照任务的的截止时间排序,这是一个顾全大局的做法,这样的做法比较优,但不是最优解,有可能牺牲某个任务放在后面做会比较好,暴力枚举第一个到第一第二大的位置,每次把该任务置于p的位置后面(即牺牲该任务以减少第一第二的值),但是这样做会使得牺牲的任务罚款最多,不一定更优,所以维护最小值。
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
const int N = 505;
const int INF = 0x3f3f3f3f;
struct job {
int s, d;
}j
;
int n, p;
bool cmp(const job& a, const job& b) {
return a.d < b.d;
}
void init() {
scanf("%d", &n);
for (int i = 0; i < n; i++) scanf("%d%d", &j[i].s, &j[i].d);
sort(j, j + n, cmp);
}
int handle(int x) {
int t = 0, a = 0, b = 0, k;
for (int i = 0; i <= p; i++) {
if (i == x) continue;
t += j[i].s;
k = max(0, t - j[i].d);
b = max(b, k);
if (b > a) swap(a, b);
}
t += j[x].s;
k = max(0, t - j[x].d);
b = max(b, k);
return a + b;
}
int solve() {
int t = 0, a = 0, b = 0;
for (int i = 0; i < n; i++) {
t += j[i].s;
if (t - j[i].d >= b) {
b = t - j[i].d;
p = i;
}
if (b > a) swap(a, b);
}
int ans = a + b;
for (int i = 0; i < p; i++)
ans = min(ans, handle(i));
return ans;
}
int main() {
int cas;
scanf("%d", &cas);
while (cas--) {
init();
printf("%d\n", solve());
}
return 0;
}
题目大意:给出n个任务,每个任务有所需时间和截止时间,单个任务超过截止时间要罚款,求一个完成任务的序列,使得最大罚款和第二大罚款数之和最小。
解题思路:一开始想用二分求解最大罚款值,后来发现不靠谱。首先贪心,按照任务的的截止时间排序,这是一个顾全大局的做法,这样的做法比较优,但不是最优解,有可能牺牲某个任务放在后面做会比较好,暴力枚举第一个到第一第二大的位置,每次把该任务置于p的位置后面(即牺牲该任务以减少第一第二的值),但是这样做会使得牺牲的任务罚款最多,不一定更优,所以维护最小值。
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
const int N = 505;
const int INF = 0x3f3f3f3f;
struct job {
int s, d;
}j
;
int n, p;
bool cmp(const job& a, const job& b) {
return a.d < b.d;
}
void init() {
scanf("%d", &n);
for (int i = 0; i < n; i++) scanf("%d%d", &j[i].s, &j[i].d);
sort(j, j + n, cmp);
}
int handle(int x) {
int t = 0, a = 0, b = 0, k;
for (int i = 0; i <= p; i++) {
if (i == x) continue;
t += j[i].s;
k = max(0, t - j[i].d);
b = max(b, k);
if (b > a) swap(a, b);
}
t += j[x].s;
k = max(0, t - j[x].d);
b = max(b, k);
return a + b;
}
int solve() {
int t = 0, a = 0, b = 0;
for (int i = 0; i < n; i++) {
t += j[i].s;
if (t - j[i].d >= b) {
b = t - j[i].d;
p = i;
}
if (b > a) swap(a, b);
}
int ans = a + b;
for (int i = 0; i < p; i++)
ans = min(ans, handle(i));
return ans;
}
int main() {
int cas;
scanf("%d", &cas);
while (cas--) {
init();
printf("%d\n", solve());
}
return 0;
}
相关文章推荐
- 在应用程序中使用Spring启动嵌入式Jetty并让Web程序共享同一个Application Context(一)
- [wikioi1553]互斥的数(数学分析+散列/数学分析+二分)
- 用Eclipse编译你的ROS程序
- 文字向上无缝滚动
- Android--去除EditText边框,添加下划线,
- 干货!表达式树解析"框架"(3)
- 单元测试
- 2013年终总结
- 最常用的设计模式
- eclipse+spket+Extjs4.2.1开发环境搭建
- RH318之虚拟机创建及其它功能
- C#简单实现贪吃蛇程序(LinQ + Entity)
- JS逗号、冒号与括号
- Support Library Features
- CloudStack4.2登录报用户名或密码错误问题解析
- 尾数为0零BigDecimal不能装成正常数
- 国内最全最详细的hadoop2.2.0集群的HA高可靠的最简单配置
- 黑马程序员_java基础笔记(08)...GUI,网络编程,正则表达式
- JavaScript里的依赖注入
- 【iCore2 模块相关资料】发布模块DEMO 代码包,目前支持 iM_TFT30、 iM_LAN和 iM_RGB 三个模块