[Codeforces 8D] Two Friends
Brief Introduction:
有两人a、b,他们都在A点,a经过B点到C点,而b直接到C点。a走过的距离不超过la,b走过距离不超过lb,询问他们可能经过最长的公共距离。
Algorithm1:
我们首先可以发现一个公共距离是否可行是具有单调性的
从而可以考虑使用二分
于是我们将问题转化为已知三个圆,询问着三个圆是否有公共部分
对于这类问题,我们每次求出三个圆中两两的交点,判断其是否在第三个圆内即可
Algorithm2:
我们假设公共路径在AD上,D在线段BC上,我们可以发现在D从B移动到C时,最长公共距离的长度是凸性函数
我们由此想到三分法
而对于每一个特定的D点,其公共距离的长度同算法1一样具有单调性,使用二分法即可
Code1:
#include <bits/stdc++.h> using namespace std;
const double eps=1e-; #define point complex<double> point a,b,c;
double AB,BC,AC,ta,tb; void Read(point &k)
{
double x,y;cin >> x >> y;
k=point(x,y);
} bool intersect(point a,double Ra,point b,double Rb,point c,double Rc)
{
if(abs(a-b)-(Ra+Rb)>eps) return false;
if(abs(a-c)-Ra<eps && abs(b-c)-Rb<eps) return true;
if(abs(a-b)+Ra-Rb<-eps || abs(a-b)+Rb-Ra<-eps) return false; b-=a;c-=a;
point i=point(b.real()/abs(b),b.imag()/abs(b)); //对原图进行线性变换,求出新的基向量
b/=i;c/=i; double x=(Ra*Ra-Rb*Rb+abs(b)*abs(b))/(*abs(b)); //用勾股定理求交点 double h=sqrt(max(Ra*Ra-x*x,0.0)); if(abs(point(x,h)-c)-Rc<eps || abs(point(x,-h)-c)-Rc<eps) return true; //对上下两个交点都进行判断
return false;
} bool eval(point a,double Ra,point b,double Rb,point c,double Rc) //查看两两的交点是否在第三圆内
{
if(Ra<eps || Rb<eps || Rc<eps) return false;
if(intersect(a,Ra,b,Rb,c,Rc)) return true;
if(intersect(a,Ra,c,Rc,b,Rb)) return true;
if(intersect(b,Rb,c,Rc,a,Ra)) return true;
return false;
} int main()
{
cout.setf(ios::fixed);
cout.precision(); cin >> ta >> tb;
Read(a);Read(c);Read(b);
AC=abs(a-c);AB=abs(a-b);BC=abs(b-c); ta+=AB+BC;tb+=AC;
if(tb-(AB+BC)>-eps)
return cout << min(tb,ta),; double l=,r=min(ta,tb);
while(fabs(r-l)>eps) //对答案二分
{
double m=(r+l)*.;
if(eval(a,m,b,ta-BC-m,c,tb-m)) l=m;
else r=m;
}
cout << (r+l)*.;
return ;
}
Code2:
#include <bits/stdc++.h> using namespace std;
const double eps=1e-; struct Point
{
double x,y;
Point(){}
Point(double a,double b){x=a,y=b;}
void input(){cin >> x >> y;}
double dist(Point&a){return hypot(x-a.x,y-a.y);}
}; double ta,tb,w,AB,AC,BC,AU,UB,UC,lm,rm;
Point A,B,C; double eval(double k)
{
Point U=Point(k*B.x+(-k)*C.x,k*B.y+(-k)*C.y);
AU=A.dist(U),UB=U.dist(B),UC=U.dist(C);
if(AU+UB<ta && AU+UC<tb)
return min(ta-UB,tb-UC); double l=,r=;
while(fabs(l-r)>eps) //二分
{
w=(l+r)*0.5;
Point V=Point(w*U.x+(-w)*A.x,w*U.y+(-w)*A.y);
if(w*AU+V.dist(B)<ta && w*AU+V.dist(C)<tb) l=w;
else r=w;
}
return (l+r)*0.5*AU;
} int main()
{
cout.setf(ios::fixed);
cout.precision();
cin >> ta >> tb;
A.input();C.input();B.input();
AB=A.dist(B),AC=A.dist(C),BC=B.dist(C); ta+=AB+1e-;tb+=AC+1e-; //先加上eps,解决精度问题 if(tb>AB+BC)
{
cout << min(tb,ta+BC);
return ;
} double l=,r=;
while(fabs(l-r)>eps) //三分
{
lm=(*l+r)/,rm=(*r+l)/;
if(eval(lm)>eval(rm)) r=rm;
else l=lm;
}
cout << eval((l+r)*0.5);
return ;
}
Review:
1、使用complex类解决计算几何问题
使用abs、hypot函数
2、线性变换
首先确定原点A,求出其它坐标与原点的相对位置
其次求出单位向量P(Xb/abs,Yb/abs),将AB作为X轴
最后使用复数除法,将其它向量除去单位向量,确定新的坐标
3、判断三圆是否有公共部分:
两两使用勾股定理判交点是否在第三个圆内
4、在无法确定某个距离时,寻找其中的凸性或单调性,用三分或二分解决
计算几何的常用策略
5、精度问题:一般选择超出答案要求的2到3位,过少会WA,过多会TLE
[Codeforces 8D] Two Friends的更多相关文章
- codeforces 8D Two Friends 二分+ 判断三个圆是否有公共交点
题目链接 有两个人x, y, 现在在A点, x要直接去B点, y要先去C点在去B点, 现在给出x, y两人可以行走的最大距离T1, T2, 求出他们从A点出发之后, 可以走的最长的公共路径. 我们先看 ...
- Codeforces 8D Two Friends 三分+二分+计算几何
题目链接:点击打开链接 题意:点击打开链接 三分house到shop的距离,二分这条斜边到cinema的距离 #include<stdio.h> #include<string.h& ...
- CodeForces 8D Two Friends 判断三个圆相交
题意: 有两个人\(Alan\)和\(Bob\),他们现在都在\(A\)点,现在\(Bob\)想去\(B\)点,\(Alan\)想先到\(C\)点再去\(B\)点. \(Alan\)所走的总路程不能超 ...
- Questions(Updating)
有时候做题时会遇到一些未学习的零碎知识点,或存疑的疑惑 为防止遗忘,在此记录 1.复数除法与线性变换的关系 Accepted Codeforces 8D(2018.5.9) Definition: 复 ...
- Mistakes(Updating)
1.当调试时发现无法正常调用函数时,检查是否发生爆栈 对于每个栈仅有4MB的空间,开int只能开大约5*10^5. 大数组一定要开全局变量 2.当long long=int*int时会爆int,一定要 ...
- Codeforces Round #219 (Div. 1)(完全)
戳我看题目 A:给你n个数,要求尽可能多的找出匹配,如果两个数匹配,则ai*2 <= aj 排序,从中间切断,分成相等的两半后,对于较大的那一半,从大到小遍历,对于每个数在左边那组找到最大的满足 ...
- Codeforces Round #509 (Div. 2) F. Ray in the tube(思维)
题目链接:http://codeforces.com/contest/1041/problem/F 题意:给出一根无限长的管子,在二维坐标上表示为y1 <= y <= y2,其中 y1 上 ...
- python爬虫学习(5) —— 扒一下codeforces题面
上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...
- 【Codeforces 738D】Sea Battle(贪心)
http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...
随机推荐
- React context基本用法
React的context就是一个全局变量,可以从根组件跨级别在React的组件中传递.React context的API有两个版本,React16.x之前的是老版本的context,之后的是新版本的 ...
- Codeforces Round #524 (Div. 2) B. Margarite and the best present
B. Margarite and the best present 题目链接:https://codeforces.com/contest/1080/problem/B 题意: 给出一个数列:an=( ...
- java摘要
**idea 注册 Licensed to ilanyu License Server: http://idea.iteblog.com/key.php 1.文件上传下载 http://blog.cs ...
- javascript实现倒计时(转)
<html> <head> <title>倒计时</title> <meta charset="utf-8"> < ...
- Spring任务调度<task:scheduled-tasks>【含cron参数详解】 (转载)
Spring内部有一个task是Spring自带的一个设定时间自动任务调度 task使用的时候很方便,但是他能做的东西不如quartz那么的多! 可以使用注解和配置两种方式,配置的方式如下 引入Spr ...
- java中的构造块、静态块等说明
一:这篇博客写的时候我在学校已经一个星期了,为什么又会想到写这le,因为这几天又在重新学下有关spring.myBatis的知识,其中在实例化sessionFactory的时候用到了静态块,虽然在学习 ...
- c#之字符串函数
1.常用的字符串函数 Compare 比较字符串的内容,考虑文化背景(场所),确定某些字符是否相等 int Compare(string str1,string str2) int Compare(s ...
- 图论:Gale-Shapley算法
Gale-Shapley算法又叫做延迟认可算法,它可以解决这么一个问题 一共有N位男士和N位女士 每位男士对每位女士都有一个好感度,让他们结合成为N对夫妻,要求男士优先表白,最后问结合情况 第一轮,每 ...
- Kali Linux中前十名的Wifi攻击工具
无 线网络的攻与防一直是比较热门的话题,由于无线信号可以被一定范围内的任何人接收到(包括死黑阔),这样就给WIFI带来了安全隐患:路由器生产厂商和网 络服务供应商(ISPs)的配置大多是默认开启了WP ...
- python3 购物车练习
# 购物车# 功能要求:# 要求用户输入总资产,例如:2000# 显示商品列表,让用户根据序号选择商品,加入购物车# 购买,如果商品总额大于总资产,提示账户余额不足,否则,购买成功.# 可充值.某商品 ...