zoukankan      html  css  js  c++  java
  • 1298 圆与三角形

    给出圆的圆心和半径,以及三角形的三个顶点,问圆同三角形是否相交。相交输出"Yes",否则输出"No"。(三角形的面积大于0)。

     
    Input
    第1行:一个数T,表示输入的测试数量(1 <= T <= 10000),之后每4行用来描述一组测试数据。
    4-1:三个数,前两个数为圆心的坐标xc, yc,第3个数为圆的半径R。(-3000 <= xc, yc <= 3000, 1 <= R <= 3000)
    4-2:2个数,三角形第1个点的坐标。
    4-3:2个数,三角形第2个点的坐标。
    4-4:2个数,三角形第3个点的坐标。(-3000 <= xi, yi <= 3000)
    Output
    共T行,对于每组输入数据,相交输出"Yes",否则输出"No"。
    Input示例
    2
    0 0 10
    10 0
    15 0
    15 5
    0 0 10
    0 0
    5 0
    5 5
    Output示例
    Yes
    No
    /*
       圆与三角形是否相交,转化为线段与圆相交
    
       1.两点都在圆内  --相离
       2.一点在圆内,一点在圆外  --相交
       3.两点都在圆外
        ① .如果圆心到线段所在直线的距离>r  --相离
        ② .距离<=r
         如果圆心与线段两个端点以及线段本身组成的夹角都是锐角   --相交
         否则  --相离
     */
    #include <bits/stdc++.h>
    
    #define MAXK 5
    
    using namespace std;
    
    struct Point{
        double x,y;
        Point(){}
        Point(double _x,double _y){
            x=_x;
            y=_y;
        }
    };
    int t;
    double a,b,r;
    Point p[MAXK],O;
    bool ok;
    
    int judge(Point *p1, Point *p2,double r) {//点p1和p2都不在圆内
        double a, b, c, dist1, dist2, angle1, angle2; // ax + by + c = 0;
        if (p1->x == p2->x)
            a = 1, b = 0, c = -p1->x;//特殊情况判断,分母不能为零
        else if (p1->y == p2->y)
            a = 0, b = 1, c = -p1->y;//特殊情况判断,分母不能为零
        else {
            a = p1->y - p2->y;
            b = p2->x - p1->x;
            c = p1->x * p2->y - p1->y * p2->x;
        }
        dist1 = a * O.x + b * O.y + c;
        dist1 *= dist1;
        dist2 = (a * a + b * b) * r * r;
        if (dist1 > dist2) return 0;//点到直线距离大于半径r
        angle1 = (O.x - p1->x) * (p2->x - p1->x) + (O.y - p1->y) * (p2->y - p1->y);
        angle2 = (O.x - p2->x) * (p1->x - p2->x) + (O.y - p2->y) * (p1->y - p2->y);
        if (angle1 > 0 && angle2 > 0) return 1;//余弦都为正,则是锐角
        return 0;
    }
    
    inline void init(){
        ok=false;
    }
    
    int main(){
    //    freopen("in.txt","r",stdin);
        scanf("%d",&t);
        while(t--){
            init();
            scanf("%lf%lf%lf",&a,&b,&r);
            O.x=a;
            O.y=b;
            int flag1=0;//圆上
            int flag2=0;//圆外
            int flag3=0;//圆内
            for(int i=0;i<3;i++){
                scanf("%lf%lf",&p[i].x,&p[i].y);
                if((a-p[i].x)*(a-p[i].x)+(b-p[i].y)*(b-p[i].y)==r*r)
                    flag1++;
                else if((a-p[i].x)*(a-p[i].x)+(b-p[i].y)*(b-p[i].y)>r*r)
                    flag2++;
                else 
                    flag3++;
            }
            if(flag1!=0){//有在圆上的
                puts("Yes");
                continue;
            }else if(flag2!=0&&flag3!=0){//有在内也有在外的
                puts("Yes");
                continue;
            }else if(flag2==0&&flag3!=0){//全部在圆内
                puts("No");
                continue;
            }else{//都在圆外就要讨论了
                for(int i=0;i<3;i++){
                    if(judge(&p[i%3],&p[(i+1)%3],r)==true){
                        ok=true;
                        break;
                    }
                }
                if(ok==false){
                    puts("No");
                }else{
                    puts("Yes");
                }
            }
        }
        return 0;
    }
  • 相关阅读:
    redis 高级
    redis连接
    redis脚本
    Redis事务
    redis发布订阅
    加一
    C语言从代码中加载动态链接库
    GCC编译器
    Vim编辑器基础
    Linux用户的创建与授权及修改密码
  • 原文地址:https://www.cnblogs.com/wuwangchuxin0924/p/7858079.html
Copyright © 2011-2022 走看看