uva 1382 - Distant Galaxy

简介: 点击打开链接uva 1382 题意:给出平面上的n个点,找出一个矩形,使得边界上含有尽量多的点 思路: 1 很清楚,如果输入的n个点在同一行或者同一列的话那么ans = n。

点击打开链接uva 1382


题意:给出平面上的n个点,找出一个矩形,使得边界上含有尽量多的点
思路:
1 很清楚,如果输入的n个点在同一行或者同一列的话那么ans = n。还有一种情况就是n个点的横坐标和纵坐标只有2种,那么这种情况ans = n。
2 对于这一题我们考虑的是枚举矩形的上下边界(纵坐标),然后利用其它的方法求左右边界,见下图


3 对于竖线i,我们用left[i]表示竖线左边位于上下边界的点数(不包括位于竖线i), on[i]表示竖线上位于上下边界之间的点数(和on2[i]的区别就是on[i]不统计位于上下边界的点数),这样给定左右边界i和j的话,矩形边界上的点数为left[j]+on2[j]+on[i]-left[i],当有边界j确定的时候,on[i]-left[i]要最大
4 那么我们枚举完上下边界后,我们利用O(n)的时间去求left[] , on[] , on2[],然后枚举有边界j,维护最大的on[i]-left[i]

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;

const int MAXN = 110;
struct Node{
    int x;
    int y;
    bool operator<(const Node& s)const{
        return x < s.x; 
    }
};
Node node[MAXN];
int n , numy[MAXN];

int solve(){
    sort(node , node+n);
    sort(numy , numy+n);
    int num = unique(numy , numy+n)-numy;
    if(num <= 2)//如果纵坐标最多只有2个那么ans = n
       return n;
    //枚举上下界
    int miny , maxy , ans;
    int left[MAXN] , on[MAXN] , on2[MAXN];
    ans = 0;
    for(int i = 0 ; i < num ; i++){
       for(int j = i+1 ; j < num ; j++){
           miny = numy[i];
           maxy = numy[j];
           //求left , on , on2数组;
           int k = -1;
           memset(left , 0 , sizeof(left));
           memset(on , 0 , sizeof(on));
           memset(on2 , 0 , sizeof(on2));
           for(int t = 0 ; t < n ; t++){
              if(!t || node[t].x != node[t-1].x){
                 k++;
                 left[k] = k == 0 ? 0 : left[k-1]+on2[k-1]-on[k-1]; 
              }
              if(node[t].y > miny && node[t].y < maxy)
                 on[k]++;
              if(node[t].y >= miny && node[t].y <= maxy)
                 on2[k]++;
           }
           if(k <= 1)//如果横坐标最多只有2个那么ans = n
              return n;
           int Max = 0;
           for(int t = 0 ; t <= k ; t++){
              ans = max(ans , left[t]+on2[t]+Max); 
              Max = max(Max , on[t]-left[t]); 
           }
       }
    }
    return ans;
}

int main(){
    int Case = 1;
    while(scanf("%d" , &n) && n){
         for(int i = 0 ; i < n ; i++){
            scanf("%d%d" , &node[i].x , &node[i].y); 
            numy[i] = node[i].y;
         }
         printf("Case %d: %d\n" , Case++ , solve());
    } 
    return 0;
}




目录
相关文章
uva10038 Jolly Jumpers
uva10038 Jolly Jumpers
45 0
UVa11776 - Oh Your Royal Greediness!
UVa11776 - Oh Your Royal Greediness!
58 0
uva375 Inscribed Circles and Isosceles Triangles
uva375 Inscribed Circles and Isosceles Triangles
45 0
UVa 10082 WERTYU
UVa 10082 WERTYU
132 0
|
算法
UVA题目分类
题目 Volume 0. Getting Started 开始10055 - Hashmat the Brave Warrior 10071 - Back to High School Physics 10300 - Ecological Premium 458 - The Decoder 494...
1574 0
uva 11806 - Cheerleaders
点击打开链接 题意:在一个n行m列的矩形里面放k个相同的石子,要求第一行,最后一行,第一列,最后一列都要有石子。问有几种方法? 思路: 1 如果题目没有要求“第一行,最后一行,第一列,最后一列都要有石子”,那么答案就是C[n*m][k],我们用C[i][j]表示i个里面选择j个的组合数。
826 0
|
机器学习/深度学习 人工智能
uva 10870 Recurrences
点击打开uva 10870 思路:构造矩阵+矩阵快速幂 分析: 1 题目给定f(n)的表达式 f(n) = a1 f(n - 1) + a2 f(n - 2) + a3 f(n -3) + .
738 0
|
人工智能
uva 305 Joseph
点击打开链接uva 305 思路: 数学+打表 分析: 1 传统的约瑟夫问题是给定n个人和m,每次数m次把当前这个人踢出局,问最后留下的一个人的编号 2 这一题是前k个人是好人,后面k个是坏人。
1051 0
|
人工智能
uva 10189 Minesweeper
/* Minesweeper WA了n次才知道uva格式错了也返回wa没有pe啊尼玛 */ #include&lt;iostream&gt; #include&lt;stdio.h&gt; #include&lt;string.h&gt; using namespace std; char a[105][105]; int main() { int i,j,n,m,
941 0

热门文章

最新文章