zoukankan      html  css  js  c++  java
  • HDU 1756 点在多边形内

    题意:

    判断点在多边形内

    题解:

    随机一个区域外的点,与所有线段求交,如果点数为奇数,则在多边形内,反之,在多边形外

    需要注意:

    1、与多边形的顶点相交时,重新随机一个点

    2、点若在多边形上的话,直接返回奇数(我就在这里tle了。。一直随机,不停了。。。)

    View Code
      1 #include <iostream>
      2 #include <cstdlib>
      3 #include <cstdio>
      4 #include <cstring>
      5 #include <algorithm>
      6 #include <cmath>
      7 #include <ctime>
      8 
      9 #define N 222
     10 #define EPS 1e-7
     11 
     12 using namespace std;
     13 
     14 struct PO
     15 {
     16     double x,y;
     17 }p[N];
     18 
     19 struct LI
     20 {
     21     PO a,b;
     22 }li[N];
     23 
     24 int n,m;
     25 
     26 inline int doublecmp(double x)
     27 {
     28     if(x>EPS) return 1;
     29     else if(x<-EPS) return -1;
     30     return 0;
     31 }
     32 
     33 inline double cross(PO &a,PO &b,PO &c)
     34 {
     35     return (c.x-a.x)*(b.y-a.y)-(c.y-a.y)*(b.x-a.x);
     36 }
     37 
     38 inline void read()
     39 {
     40     for(int i=1;i<=n;i++) scanf("%lf%lf",&p[i].x,&p[i].y);
     41     for(int i=1;i<n;i++) li[i].a=p[i],li[i].b=p[i+1];
     42     li[n].a=p[n]; li[n].b=p[1];
     43     scanf("%d",&m);
     44 }
     45 
     46 inline int segcross(LI &a,LI &b)
     47 {
     48     int p1,p2,d1,d2;
     49     //b跨立a 
     50     p1=doublecmp(cross(a.a,a.b,b.a));
     51     p2=doublecmp(cross(a.a,a.b,b.b));
     52     //a跨立b 
     53     d1=doublecmp(cross(b.a,b.b,a.a));
     54     d2=doublecmp(cross(b.a,b.b,a.b));
     55     if(p1==0||p2==0) return -1;//多边形的边的端点在射线上 
     56     else if(p1*p2<=0&&d1*d2<=0) return 1;
     57     else return 0;
     58 }
     59 
     60 inline bool inside(PO &a,PO &b,PO &c)//点在直线上 
     61 {
     62     if(!doublecmp(cross(a,b,c))&&doublecmp((b.x-a.x)*(c.x-a.x))<=0&&doublecmp((b.y-a.y)*(c.y-a.y))<=0) return true;
     63     return false;
     64 }
     65 
     66 inline int getnum()
     67 {
     68     int ans,i=0,tmp;
     69     while(i<=n)
     70     {
     71         li[i].b.x=rand()+1000.0;
     72         li[i].b.y=rand()+1000.0;
     73         ans=0;
     74         for(i=1;i<=n;i++)
     75         {
     76             if(inside(li[0].a,li[i].a,li[i].b)) return 1;//点在多边形上 
     77             tmp=segcross(li[0],li[i]);
     78             if(tmp==-1) break;
     79             else if(tmp==1) ans++;
     80         }
     81     }
     82     return ans;
     83 }
     84 
     85 inline void go()
     86 {
     87     int ans;
     88     while(m--)
     89     {
     90         scanf("%lf%lf",&li[0].a.x,&li[0].a.y);
     91         if(getnum()&1) puts("Yes");
     92         else puts("No");
     93     }
     94 }
     95 
     96 int main()
     97 {
     98     while(scanf("%d",&n)!=EOF) read(),go();
     99     return 0;
    100 }
  • 相关阅读:
    OpenCV程序在生产环境中运行
    C#调用C++导出(dllexport)方法
    IIS7.5 GZip配置
    wcf学习笔记--初识wcf
    Greenplum installation guide
    Cloudera 5.8.2 Installation guide
    WPF DataGrid 合并单元格
    wpf DataGrid CheckBox列全选
    WPF button 圆角制作
    WPF passwordbox 圆角制作
  • 原文地址:https://www.cnblogs.com/proverbs/p/2923788.html
Copyright © 2011-2022 走看看