zoukankan      html  css  js  c++  java
  • Codeforces 830A. Office Keys (贪心二分 or DP)

    原题链接:http://codeforces.com/contest/830/problem/A

    题意:在一条数轴上分别有n个人和k把钥匙(n<=k),以及一个目的地,每个人要各自拿到一个钥匙后到达目的地。每个人的移动速度都是1, 问所有人都到达目的地的最短时间。

    思路:转化一下题意,就是求耗时最长的人所用的最短时间。

    我们可以二分答案x,然后对排序后的人以及钥匙进行枚举,进行从左至右搭配。 这里check函数中返回false的条件是从左至右所有人都能在x的时间内到达目的地,而计算这些人到达目的地的时间是:取最近的一对人与钥匙计算,为了每个人耗时最少(贪心),这是最优的。

    AC代码:

     1 #include<iostream>
     2 #include<cstring>
     3 #include<cstdio>
     4 #include<algorithm>
     5 #include<cmath>
     6 using namespace std;
     7 typedef long long LL;
     8 const int MAXN=2005;
     9 int n,k;
    10 int a[MAXN],b[MAXN],S;
    11 bool check(LL x){
    12     int p=0;
    13     for(int i=1;i<=n;i++){
    14         while(p<=k){
    15             p++;
    16             if(1LL*(abs(a[i]-b[p])+abs(b[p]-S))<=x) break;
    17         }
    18         if(p>k) return false;
    19     }
    20     return true;
    21 }
    22 int main()
    23 {
    24     scanf("%d %d %d", &n , &k, &S);
    25     for(int i=1;i<=n;i++) scanf("%d", &a[i]);
    26     for(int i=1;i<=k;i++) scanf("%d", &b[i]);
    27     sort(a+1, a+n+1);
    28     sort(b+1, b+k+1);
    29     LL l=0,r=2e9+5;
    30     LL mid;
    31     while(l<r){
    32         mid=(l+r)/2;
    33         if(check(1LL*mid)) r=mid;
    34         else l=mid+1; 
    35     }
    36     printf("%d
    ", l);
    37 }

    这道题也可以用dp来做: dp[i][j]表示前i个人有前j个钥匙的最优解,利用01背包思想可以在O(n*k)时间内求出答案。

     1 #include<iostream>
     2 #include<cstring>
     3 #include<cstdio>
     4 #include<algorithm>
     5 #include<cmath>
     6 using namespace std;
     7 typedef long long LL;
     8 const int INF=2e9+10;
     9 const int MAXN=2005;
    10 int n,k;
    11 int a[MAXN],b[MAXN],S;
    12 int dp[MAXN][MAXN];
    13 int main(){
    14     scanf("%d %d %d", &n , &k, &S);
    15     for(int i=1;i<=n;i++) scanf("%d", &a[i]);
    16     for(int i=1;i<=k;i++) scanf("%d", &b[i]);
    17     sort(a+1,a+1+n);
    18     sort(b+1,b+1+k);
    19     for(int i=0; i<=n; i++)
    20         for(int j=0; j<=k; j++)
    21             dp[i][j] = INF;
    22     for(int i=0; i<=k; i++) dp[0][i]=0;
    23     for(int i=1; i<=n; i++)
    24         for(int j=i; j<=k; j++){
    25             dp[i][j] = min(dp[i][j-1],max(dp[i-1][j-1],abs(a[i]-b[j])+abs(b[j]-S)));
    26         }
    27     printf("%d
    ", dp[n][k]);
    28     return 0;
    29 }
  • 相关阅读:
    node-sass 安装失败
    js中复制功能总结
    设置NODE_ENV=test环境变量
    js eslint语法规范错误提示代码
    npm安装node包时怎么显示安装进度
    前端面试题总结三
    5种方式将数字转成千分位
    前端面试题总结二(js原型继承)
    前端面试题总结一(js变量和函数声明提前相关)
    1109 Group Photo (25分)
  • 原文地址:https://www.cnblogs.com/MasterSpark/p/7482085.html
Copyright © 2011-2022 走看看