1475 建设国家 DP

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1475

这题转化过来就是,给定n个点,每个点都有一个过期时间,一个价值。现在安排他们成一直线,使得总和最大。

一开始就是贪心,这是一个很经典的贪心。

http://www.cnblogs.com/liuweimingcprogram/p/6358456.html

和这题差不多,然后这个比较特别,有一个地方是可以接两个,我就暴力枚举了。然后蜜汁wa

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <assert.h>
#define IOS ios::sync_with_stdio(false)
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;

#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
#include <bitset>
const int maxn = 1e3 + 20;
struct node {
    int val, cost;
    node(int a, int b) : val(a), cost(b) {}
    node() {}
    bool operator < (const struct node & rhs) const {
        if (val != rhs.val) return val > rhs.val;
        else return cost < rhs.cost;
    }
}a[maxn];
int w[maxn];
bool del[maxn];
vector<struct node>vc;
void work() {
    int n, H;
    cin >> n >> H;
    for (int i = 1; i <= n; ++i) {
        cin >> a[i].cost >> a[i].val;
        a[i].cost = H - a[i].cost;
    }
    sort(a + 1, a + 1 + n);
    int ans = 0;
    for (int i = 1; i <= n; ++i) {
        int t = a[i].cost;
        if (t > n) {
            ans += a[i].val; //必定可以,空位也少了一个了,就放去第n天
            del[i] = true;
            continue;
        }
        while (t >= 1 && w[t]) {
            t--;
        }
        if (t) {
            w[t] = a[i].val;
            ans += a[i].val;
            del[i] = true;
        }
    }
    for (int i = 1; i <= n; ++i) {
        if (del[i]) continue;
        vc.push_back(a[i]);
    }
//    cout << w[1] << endl;
//    cout << w[2] << endl;
//    cout << w[3] << endl;
//    cout << w[4] << endl;
//    cout << ans << endl;
    int tans = 0;
    for (int i = 1; i <= n; ++i) {
        tans += w[i];
        int tofind = -inf;
        if (w[i] == 0) continue;
        for (int j = 0; j < vc.size(); ++j) {
            if (vc[j].cost < i) continue;
            tofind = max(tofind, vc[j].val);
        }
        ans = max(ans, tans + tofind);
    }
    cout << ans << endl;
}
int main() {
#ifdef local
    freopen("data.txt", "r", stdin);
//    freopen("data.txt", "w", stdout);
#endif
    work();
    return 0;
}

看了题解后居然是dp

用dp[i][j]表示前i个点中,选出了j个点,的最大答案。

那么,要求解dp[i][j],则有dp[i][j] = dp[i - 1][j],就是不选第i个点。

如果要选,那么有两种情况,

1、把这个点作为结束点,也就是两个点放在一起。这需要这个点的保质期 >= j - 1即可。因为现在求解的是选了j个点,那么以前就是选了j - 1个点,排在一起,所以只需要保质期 >= j - 1,然后更新答案,不更新dp数组

2、接在后一排,那么需要保质期 >= j

特别地,一开始就过期的,就不能选。

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <assert.h>
#define IOS ios::sync_with_stdio(false)
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;

#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
#include <bitset>
const int maxn = 1e3 + 20;
int dp[maxn][maxn];
struct node {
    int cost, val;
    bool operator < (const struct node & rhs) const {
        if (cost != rhs.cost) return cost < rhs.cost;
        else return val > rhs.val;
    }
}a[maxn];
void work() {
    int n, h;
    cin >> n >> h;
    for (int i = 1; i <= n; ++i) {
        cin >> a[i].cost >> a[i].val;
        a[i].cost = h - a[i].cost;
    }
    sort(a + 1, a + 1 + n);
//    memset(dp, -0x3f, sizeof dp);
    dp[0][0] = 0;
    int ans = 0;
    for (int i = 1; i <= n; ++i) {
        if (a[i].cost == 0) continue;
        for (int j = 1; j <= i; ++j) {
            dp[i][j] = dp[i - 1][j];
            if (a[i].cost >= j - 1) {
                ans = max(ans, dp[i - 1][j - 1] + a[i].val);
            }
            if (a[i].cost >= j) {
                dp[i][j] = max(dp[i][j], dp[i - 1][j - 1] + a[i].val);
            }
        }
    }
    cout << ans << endl;
}

int main() {
#ifdef local
    freopen("data.txt", "r", stdin);
//    freopen("data.txt", "w", stdout);
#endif
    work();
    return 0;
}

时间: 2024-12-28 21:31:43

1475 建设国家 DP的相关文章

1475 建设国家(优先队列)

1475 建设国家 基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题 小C现在想建设一个国家.这个国家中有一个首都,然后有若干个中间站,还有若干个城市. 现在小C想把国家建造成这样的形状:选若干(可以是0个)的中间站把他们连成一条直线,然后把首都(首都也是一个中间站)连在这一条直线的左端.然后每个点可以连一个城市,特别的是最右端的点可以连接两个城市. 现在有n个城市的规划供小C选择.但是,他们那儿的交通条件比较差,他们那儿一天是2*H个小时,每个城市里面的人每天

1475 建设国家

题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1475 这题一开始我是用贪心去做的,但是一直没过...貌似贪心并不能得到最优解法. 然后这道题用优先队列来写 首先我们先把对于第i的中间站可以连接的城市用向量数组存起来..然后在只有满足中间站数量<i的情况下才向队列中和添加元素..然后若是队列元素大于i时把把城市人口最小的给删去..这样就能保证每次选择的时候都能保证选的是最优选择.. 这题真的是很妙..膜拜in

51nod建设国家

小C现在想建设一个国家.这个国家中有一个首都,然后有若干个中间站,还有若干个城市. 现在小C想把国家建造成这样的形状:选若干(可以是0个)的中间站把他们连成一条直线,然后把首都连在这一条直线的左端.然后每个点可以连一个城市,特别的是最右端的点可以连接两个城市. 现在有n个城市的规划供小C选择.但是,他们那儿的交通条件比较差,他们那儿一天是2*H个小时,每个城市里面的人每天都会去首都拿一样东西,从他们所在的城市出发,到了首都之后拿了东西就走(拿东西的时间可以忽略不计),他们要在2*H个小时之内返回

BZOJ 1096: [ZJOI2007]仓库建设(DP+斜率优化)

[ZJOI2007]仓库建设 Description L公司有N个工厂,由高到底分布在一座山上.如图所示,工厂1在山顶,工厂N在山脚.由于这座山处于高原内陆地区(干燥少雨),L公司一般把产品直接堆放在露天,以节省费用.突然有一天,L公司的总裁L先生接到气象部门的电话,被告知三天之后将有一场暴雨,于是L先生决定紧急在某些工厂建立一些仓库以免产品被淋坏.由于地形的不同,在不同工厂建立仓库的费用可能是不同的.第i个工厂目前已有成品Pi件,在第i个工厂位置建立仓库的费用是Ci.对于没有建立仓库的工厂,其

【BZOJ】1096: [ZJOI2007]仓库建设(dp+斜率优化)

http://www.lydsy.com/JudgeOnline/problem.php?id=1096 首先得到dp方程(我竟然自己都每推出了QAQ)$$d[i]=min\{d[j]+cost(j+1,i)\}+c[i]$$其中$d[i]$是前i个且在i建仓库的最小费用,$cost(j+1,i)$是将j+1-i的东西全都运到i的费用 而我们先考虑cost怎么求,orz,好神的前缀和,首先维护sum[i]表示1-i的物品,则j-i的东西从j全都运到i需要$$(sum[i]-sum[j]) \ti

poj 1185 火炮 (减少国家DP)

火炮 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 19690   Accepted: 7602 Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队. 一个N*M的地图由N行M列组成.地图的每一格可能是山地(用"H" 表示).也可能是平原(用"P"表示).例如以下图.在每一格平原地形上最多可以布置一支炮兵部队(山地上不可以部署炮兵部队):一支炮兵部队在地图上的攻

poj - 1170 - Shopping Offers(减少国家dp)

意甲冠军:b(0 <= b <= 5)商品的种类,每个人都有一个标签c(1 <= c <= 999),有需要购买若干k(1 <= k <=5),有一个单价p(1 <= p <= 999).有s(0 <= s <= 99).问完毕採购最少须要多少钱. 题目链接:http://poj.org/problem?id=1170 -->>已有b种物品,再将每种优惠分别看成一种新物品,剩下就是全然背包问题了.. 设dp[i]表示购买状态为 i 时

《“十三五”国家信息化规划》涉及网络与信息安全的内容摘录

就在今天(12月27日),国务院全文刊发了<"十三五"国家信息化规划>(参见:http://www.gov.cn/zhengce/content/2016-12/27/content_5153411.htm).在此摘录跟网络与信息安全相关的内容如下: [注:我最关注的几点是(信息量太大!)] 构建关键信息基础设施安全保障体系.实施网络安全审查制度,防范重要信息技术产品和服务网络安全风险.建立国家关键信息基础设施目录,制定关于国家关键信息基础设施保护的指导性文件,进一步明确关

【文件】“十三五”国家信息化规划

索 引 号: 000014349/2016-00257 主题分类: 工业.交通\信息产业(含电信) 发文机关: 国务院 成文日期: 2016年12月15日 标 题: 国务院关于印发“十三五”国家信息化规划的通知 发文字号: 国发[2016]73号 发布日期: 2016年12月27日 主 题 词:       国务院关于印发 “十三五”国家信息化规划的通知 国发[2016]73号 各省.自治区.直辖市人民政府,国务院各部委.各直属机构: 现将<“十三五”国家信息化规划>印发给你们,请认真贯彻执行