zoukankan      html  css  js  c++  java
  • hdoj1050 Moving Tables(贪心)

    题目链接

    http://acm.hdu.edu.cn/showproblem.php?pid=1050

    题意

    有一条走廊,走廊两边各有200个房间,一边的房间编号是奇数,另一边是偶数。现在有n个箱子需要从一个房间移动到另一个房间,移动一个箱子需要10分钟,箱子可以同时移动,但某一段走廊每次只能移动1个箱子,比如(1,3)和(6,8)可以同时移动,而(1,5)和(6,8)不能同时移动,求最少需要多少时间可以把箱子移动完毕。

    思路

    先将箱子按开始的房间号由小到大排序,然后循环遍历,在遍历的过程中不断去除可以同时移动的箱子(表示箱子已移动完毕),记录循环的次数,直到所有的箱子均已移动完毕。循环次数*10即是答案。由于这是区间相交问题,所以要将所有的房间号映射到一维坐标系中,根据题目中的图可知,对于房间号为k的房间来说(k为偶数),房间k和房间k-1是相对的,也就是偶数k和k-1在坐标轴上对应的是同一位置,所以要将所有编号为偶数k的房间的编号转化成k-1.

    代码

     1 #include <algorithm>
     2 #include <iostream>
     3 #include <cstring>
     4 #include <cstdio>
     5 #include <vector>
     6 using namespace std;
     7 
     8 struct Node
     9 {
    10     int s, e;
    11     bool visit;    //记录箱子是否已经移动完毕,值为true则移动完毕
    12 
    13     Node(int s, int e) :s(s), e(e), visit(false) {}
    14     bool operator < (const Node& node)const
    15     {
    16         return s < node.s;    //将房间按开始房间号从小到大排序
    17     }
    18 };
    19 
    20 int main()
    21 {
    22     //freopen("hdoj1050.txt", "r", stdin);
    23     vector<Node> v;
    24     int m, n;
    25     cin >> m;
    26     while (m--)
    27     {
    28         cin >> n;
    29         v.clear();
    30         int s, e;
    31         for (int i = 0; i<n; i++)
    32         {
    33             cin >> s >> e;
    34             if (s > e)            //注意输入的开始房间号可能大于结束房间号
    35                 swap(s, e);
    36             if (s % 2 == 0)        //房间号为偶数的,要变成该偶数的前一个奇数
    37                 s--;
    38             if (e % 2 == 0)
    39                 e--;
    40             v.push_back(Node(s, e));
    41         }
    42 
    43         sort(v.begin(), v.end());
    44         int ans = 0;    //记录循环的次数
    45         int cnt = 0;    //记录已经移动完毕的箱子的个数
    46         int cur = 0;    //当前从第cur个箱子开始遍历
    47         int t;
    48         while (cnt != n)
    49         {
    50             for (int i = 0; i<n; i++)
    51             {
    52                 if (!v[i].visit)
    53                 {
    54                     cur = i;
    55                     t = v[i].e;
    56                     v[i].visit = true;
    57                     cnt++;
    58                     break;
    59                 }
    60             }
    61             for (int i = cur+1; i<n; i++)
    62             {
    63                 if (!v[i].visit && v[i].s > t)
    64                 {
    65                     cnt++;
    66                     t = v[i].e;
    67                     v[i].visit = true;
    68                 }
    69             }
    70             ans++;
    71         }
    72         cout << ans * 10 << endl;
    73     }
    74     return 0;
    75 }

    注意点

    1、要按开始房间号从小到大排序,而不是结束房间号。手动运行下面的数据有助于理解:

    1
    4
    2 10
    4 6
    8 16
    12 14

    2、输入的开始房间号可能大于结束房间号,此时要将两者交换;

    3、对于编号为偶数k的房间,需将其编号转为奇数k-1

    测试数据

    输入:

    3
    2
    1 3
    4 6
    4
    1 20
    2 10
    11 19
    15 16
    4
    2 10
    4 6
    8 16
    12 14

    输出:

    20
    30
    20
  • 相关阅读:
    信息论
    学习抓包
    深入学习垃圾kafka
    share data
    【转载】计算图像相似度——《Python也可以》之一
    聊聊java list的使用特性
    log4j多线程以及分文件输出日志
    【转载】JDBC的连接参数的设置导致rowid自动添加到sql
    背包问题
    【转】【动态规划】01背包问题
  • 原文地址:https://www.cnblogs.com/sench/p/7999792.html
Copyright © 2011-2022 走看看