zoukankan      html  css  js  c++  java
  • COGS2085 Asm.Def的一秒

    时间限制:1 s   内存限制:256 MB

    【题目描述】

    “你们搞的这个导弹啊,excited!”

    Asm.Def通过数据链发送了算出的疑似目标位置,几分钟后,成群结队的巡航导弹从“无蛤”号头顶掠过,布满了天空。

    “一共发射了多少导弹?”

    “十亿美元。”斯科特·华莱士回答,“单价100万,现在天上有1000多枚。这玩意能自动搜索10个可疑点,找到目标就发动攻击。”

    “什么?10个?我给了它10万个点!”

    “这会让它的程序崩溃的。好在你还有时间手动输入路径。”

    “多久?”

    “零……还有一秒,他们又给续上了一秒。”

    “我想静静,别问我静静是谁。”

    Asm.Def在第一象限内找到了n个可疑点。他需要为导弹规划路径。

    如图所示,导弹一开始在(0,0)。它只能朝着一定的方向——即严格夹在图中两条射线间的方向(白色部分)前进。注意,它不能沿着这两条射线前进,当然也不能停在原地。

    当导弹到达某个可疑点后,它仍然只能朝着该范围内的方向前进,如图。

    Asm.Def想要让导弹经过尽可能多的可疑点。他需要在一秒钟内知道,最多能经过多少个可疑点。

    【输入格式】

    第1行1个整数n。

    第2行4个整数a b c d:代表两条射线的斜率分别是a/b和c/d。保证0<=a,b,c,d<=10^5,a/b<c/d(即a/b是靠下的那条射线),a/b≠0/0,c/d≠0/0.

    接下来n行,每行2个整数xi,yi(1<=xi,yi<=10^5),代表i号可疑点的坐标。

    【输出格式】

    一行一个整数,即最多能经过几个可疑点。

    【样例输入】

    15
    1 3 2 1
    3 1
    6 2
    4 2
    2 5
    4 5
    6 6
    3 4
    1 6
    2 1
    7 4
    9 3
    5 3
    1 3
    15 5
    12 4

    【样例输出】

    4

    【提示】

    这是最佳路径。注意,导弹不能前往位于射线上的点。

    对于30%的数据,n<=1000,a=0,b=1,c=1,d=0。

    对于60%的数据,n<=1000。

    对于100%的数据,n<=10^5。

    数学问题 计算几何  动态规划 LIS

    题面异常带感啊233

    如果两条射线分别平行于x和y轴,显然是一个最长单调上升序列问题

    现在两条射线角度不定,但只要以射线为x、y轴建立平面坐标系,把原坐标系上的点映射到新坐标系上,就又可以求LIS了

    如何求新坐标?

    过点作新坐标轴x的平行线,与另一坐标轴y交于点k,|(kx,ky)-(0,0)|即是其在新坐标轴y上的长度,用它除以新坐标轴y向量的单位长度,就是它的新y坐标。

    x坐标同理

    ↑注意由于输入的是分数,一通约分之后可以得到整数坐标,不需要用double存←博主后知后觉

    注意所求是单调上升而不是单调不降,所以x坐标相同的时候,要先更新y大的

     1 #include<iostream>
     2 #include<algorithm>
     3 #include<cstring>
     4 #include<cstdio>
     5 #include<cmath>
     6 #define LL long long
     7 using namespace std;
     8 const int mxn=100010;
     9 int read(){
    10     int x=0,f=1;char ch=getchar();
    11     while(ch<'0' || ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    12     while(ch>='0' && ch<='9'){x=x*10+ch-'0';ch=getchar();}
    13     return x*f;
    14 }
    15 struct point{
    16     LL x,y;
    17 /*    double x,y;
    18     point operator + (point b){return (point){x+b.x,y+b.y};}
    19     point operator - (point b){return (point){x-b.x,y-b.y};}
    20     double operator * (point b){return x*b.x+y*b.y;}*/
    21     bool operator < (point b)const{
    22         return x<b.x || (x==b.x && y>b.y);
    23     }
    24 }a[mxn],T[3];
    25 LL Cross(point a,point b){
    26     return a.x*b.y-a.y*b.x;
    27 }
    28 int n,m;
    29 int f[mxn],t[mxn];
    30 void add(int x,int v){
    31     while(x<=n){t[x]=max(t[x],v);x+=x&-x;}
    32 }
    33 int que(int x){
    34     int res=0;
    35     while(x){res=max(res,t[x]);x^=x&-x;}
    36     return res;
    37 }
    38 LL bas[mxn];
    39 int main(){
    40     freopen("asm_second.in","r",stdin);
    41     freopen("asm_second.out","w",stdout);
    42     int i,j;point tmp;
    43     n=read();
    44     T[1].y=read();T[1].x=read();T[2].y=read();T[2].x=read();
    45     for(i=1;i<=n;i++){
    46         a[i].x=read();a[i].y=read();
    47         tmp=(point){Cross(T[1],a[i]),Cross(a[i],T[2])};
    48         //a[i].x*t[1].y-a[i].y*t[1].x
    49         a[i]=tmp;
    50     }
    51     a[++n]=(point){0,0};
    52     sort(a+1,a+n+1);
    53     //
    54     for(i=1;i<=n;i++)bas[i]=a[i].y;
    55     sort(bas+1,bas+n+1);
    56     int ed=unique(bas+1,bas+n+1)-bas-1;
    57     for(i=1;i<=n;i++){
    58         if(a[i].y<0)a[i].y=-1;
    59         else a[i].y=lower_bound(bas+1,bas+ed+1,a[i].y)-bas-1;
    60     }
    61     int ans=0;
    62     for(i=1;i<=n;i++){
    63 //        printf("i:%d %lld %lld
    ",i,a[i].x,a[i].y);
    64         if(a[i].x<=0 || a[i].y<=0)continue;
    65         int res=que(a[i].y-1);
    66         f[i]=max(f[i],res+1);
    67         add(a[i].y,f[i]);
    68         ans=max(ans,f[i]);
    69     }
    70     printf("%d
    ",ans);
    71     return 0;
    72 }
  • 相关阅读:
    只用一个字节 计算象棋将帅之间可能的位置
    后缀数组学习
    java 构造不可变类集的使用方法
    topcoder SRM 639 div2
    navicat和pymysql
    表查询
    表的关系对应
    MySQl数据类型和条件限制
    复习之网络编程
    协程
  • 原文地址:https://www.cnblogs.com/SilverNebula/p/6667286.html
Copyright © 2011-2022 走看看