博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
UVA - 10674-Tangents
阅读量:6207 次
发布时间:2019-06-21

本文共 4988 字,大约阅读时间需要 16 分钟。



题意:给出两个圆,求它们的公切线,并依照一定格式输出

做法:模拟

代码:

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;const double eps=2e-5;const double pi=acos(-1.0);int dcmp(double x){return fabs(x)

0:x<0?-1:1;} struct dot { double x,y; dot(){} dot(double a,double b){x=a;y=b;} dot operator +(dot a){return dot(x+a.x,y+a.y);} dot operator -(dot a){return dot(x-a.x,y-a.y);} dot operator *(double a){return dot(x*a,y*a);} double operator *(dot a){return x*a.y-y*a.x;} dot operator /(double a){return dot(x/a,y/a);} double operator /(dot a){return x*a.x+y*a.y;} bool operator ==(dot a){return x==a.x&&y==a.y;} void in(){scanf("%f%f",&x,&y);} void out(){printf("%f %f\n",x,y);} dot norv(){return dot(-y,x);} dot univ(){double a=mod();return dot(x/a,y/a);} dot ro(double a){return dot(x*cos(a)-y*sin(a),x*sin(a)+y*cos(a));} double mod(){return sqrt(x*x+y*y);} double dis(dot a){return sqrt(pow(x-a.x,2)+pow(y-a.y,2));} }; typedef pair<dot,dot> LL; LL ans[10]; bool operator <(LL a,LL b) { return dcmp(a.first.x-b.first.x)!=0?dcmp(a.first.x-b.first.x)<0: dcmp(a.first.y-b.first.y)<0; } int work(dot a,double r1,dot b,double r2) { int cnt; double ang; if(dcmp(a.x-b.x)==0&&dcmp(a.y-b.y)==0&&dcmp(r1-r2)==0) return -1; if(dcmp(a.dis(b)+min(r1,r2)-max(r1,r2))<0) return 0; if(dcmp(a.dis(b)+min(r1,r2)-max(r1,r2))==0) { if(dcmp(r1-r2)>0) ans[0].first=ans[0].second=a+(b-a).univ()*r1; else ans[0].first=ans[0].second=b+(a-b).univ()*r2; return 1; } ang=acos((r1-r2)/a.dis(b)); ans[0].first=a+(b-a).ro(ang).univ()*r1; ans[1].first=a+(b-a).ro(-ang).univ()*r1; ang=pi-ang; ans[0].second=b+(a-b).ro(-ang).univ()*r2; ans[1].second=b+(a-b).ro(ang).univ()*r2; cnt=2; if(dcmp(a.dis(b)-r1-r2)==0) { ans[2].first=ans[2].second=a+(b-a).univ()*r1; cnt=3; } if(dcmp(a.dis(b)-r1-r2)>0) { ang=acos((r1+r2)/a.dis(b)); ans[2].first=a+(b-a).ro(ang).univ()*r1; ans[3].first=a+(b-a).ro(-ang).univ()*r1; ans[2].second=b+(a-b).ro(ang).univ()*r2; ans[3].second=b+(a-b).ro(-ang).univ()*r2; cnt=4; } sort(ans,ans+cnt); return cnt; } int main() { int i,cnt; dot a,b; double r1,r2; while(cin>>a.x>>a.y>>r1>>b.x>>b.y>>r2) { if(a.x==0&&a.y==0&&r1==0&&b.x==0&&b.y==0&&r2==0) return 0; cnt=work(a,r1,b,r2); cout<<cnt<<endl; for(i=0;i<cnt;i++) printf("%.5f %.5f %.5f %.5f %.5f\n",ans[i].first.x,ans[i].first.y, ans[i].second.x,ans[i].second.y,ans[i].first.dis(ans[i].second)); } }

Description

Problem B

Tangents
Input:
standard input
Output: standard output
Time Limit: 2 seconds
 

You can see in the pictures below that two different circles can have at most four common tangents. Given the center and radius of two circles your job is to find the length of their common tangents and also the points where they touch the two circles.

 

 

Input

The input file contains several lines of inputs.

 

Each line contains six integers x1 (-100<=x1<=100), y1 (-100<=y1<=100), r1 (0<r1<=200), x2 (-100<=x2<=100), y2 (-100<=y2<=100), r2 (0<r2<=200). Here (x1, y1) and (x2, y2) are the coordinates of the center of the first circle and second circle respectively, r1 is the radius of the first circle and r2 is the radius of the second circle.

 

Input is terminated by a line containing six zeroes.

 

Output

For each line of input you should produce one of more lines of output. The description of this output is given below.

 

First line of the output for each line of input contains an integer n, which denotes the number of different tangents between the two circles. If there is infinite number of tangents between the two circles then the value of n should be –1. If n is positive then next n lines contains the description of each tangent. The description of the tangent contains five floating-point numbers Sx, Sy, Tx, Ty, L in a single line. Here (Sx, Sy) is the point at which the tangent touches the first circle and (Tx, Ty) is the point where the tangent touches the second circle and L is the length of the tangent. All the floating-point numbers have five digits after the decimal point. Errors less than 2*10-5 will be ignored. The tangents should be printed in ascending order of Sx and in case of a tie they should be printed in ascending order of Sy.

 

Sample Input           Output for Sample Input

10 10 5 20 20 5

10 10 10 20 20 10

10 10 5 20 10 5

0 0 0 0 0 0

4

6.46447 13.53553 16.46447 23.53553 14.14214

10.00000 15.00000 20.00000 15.00000 10.00000

13.53553 6.46447 23.53553 16.46447 14.14214

15.00000 10.00000 15.00000 20.00000 10.00000

2

2.92893 17.07107 12.92893 27.07107 14.14214

17.07107 2.92893 27.07107 12.92893 14.14214

3

10.00000 5.00000 20.00000 5.00000 10.00000

10.00000 15.00000 20.00000 15.00000 10.00000

15.00000 10.00000 15.00000 10.00000 0.00000

 


Problem setter: Shahriar Manzoor. Member of Elite Problem Setters’ Panel

Special Thanks: Derek Kisman, Member of Elite Problem Setters’ Panel

Input

Output

Sample Input

Sample Output

Hint

Source

Root :: Prominent Problemsetters ::

转载地址:http://zczja.baihongyu.com/

你可能感兴趣的文章
Solr部署到tomcat
查看>>
python笔记8-多线程threading之封装式
查看>>
新技术与注会
查看>>
医美分期一定会倒下一大片?
查看>>
MongoDB 通过配置文件启动
查看>>
组合模型初探
查看>>
git解决冲突(rebase版)
查看>>
科研项目之经验之谈
查看>>
一行代码打开相册/相机
查看>>
使用Maven对JAVA程序打包-带主类、带依赖
查看>>
【Lintcode】二叉树的最大深度 - 比较简单,用递归比较好,不递归也能做,比较麻烦...
查看>>
批量 kill mysql 线程
查看>>
Anaconda的使用和包的更新;conda 创建虚拟环境
查看>>
JSON.parse 解析json字符串时,遇换行符报错
查看>>
Spring Cloud 概述
查看>>
Extensions in UWP Community Toolkit - Overview
查看>>
【详细】【转】C#中理解委托和事件
查看>>
网络通信
查看>>
华为手机设置桌面图标角标提醒的实现
查看>>
[日常] Go语言圣经-错误,函数值习题
查看>>