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的更多相关文章

  1. codeforces 8D Two Friends 二分+ 判断三个圆是否有公共交点

    题目链接 有两个人x, y, 现在在A点, x要直接去B点, y要先去C点在去B点, 现在给出x, y两人可以行走的最大距离T1, T2, 求出他们从A点出发之后, 可以走的最长的公共路径. 我们先看 ...

  2. Codeforces 8D Two Friends 三分+二分+计算几何

    题目链接:点击打开链接 题意:点击打开链接 三分house到shop的距离,二分这条斜边到cinema的距离 #include<stdio.h> #include<string.h& ...

  3. CodeForces 8D Two Friends 判断三个圆相交

    题意: 有两个人\(Alan\)和\(Bob\),他们现在都在\(A\)点,现在\(Bob\)想去\(B\)点,\(Alan\)想先到\(C\)点再去\(B\)点. \(Alan\)所走的总路程不能超 ...

  4. Questions(Updating)

    有时候做题时会遇到一些未学习的零碎知识点,或存疑的疑惑 为防止遗忘,在此记录 1.复数除法与线性变换的关系 Accepted Codeforces 8D(2018.5.9) Definition: 复 ...

  5. Mistakes(Updating)

    1.当调试时发现无法正常调用函数时,检查是否发生爆栈 对于每个栈仅有4MB的空间,开int只能开大约5*10^5. 大数组一定要开全局变量 2.当long long=int*int时会爆int,一定要 ...

  6. Codeforces Round #219 (Div. 1)(完全)

    戳我看题目 A:给你n个数,要求尽可能多的找出匹配,如果两个数匹配,则ai*2 <= aj 排序,从中间切断,分成相等的两半后,对于较大的那一半,从大到小遍历,对于每个数在左边那组找到最大的满足 ...

  7. Codeforces Round #509 (Div. 2) F. Ray in the tube(思维)

    题目链接:http://codeforces.com/contest/1041/problem/F 题意:给出一根无限长的管子,在二维坐标上表示为y1 <= y <= y2,其中 y1 上 ...

  8. python爬虫学习(5) —— 扒一下codeforces题面

    上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...

  9. 【Codeforces 738D】Sea Battle(贪心)

    http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...

随机推荐

  1. hdu1281(棋盘游戏,车的放置)

    Problem Description 给定一个n * m的棋盘,在棋盘里放尽量多的国际象棋中的车,使他们不能相互攻击 已知有些格子不能放置,问最多能放置多少个车 并计算出必须棋盘上的必须点. Inp ...

  2. springboot与dubbo结合

    转:http://www.cnblogs.com/Alandre/p/6490142.html  写的很好! 本文提纲 一.为啥整合 Dubbo 实现 SOA 二.运行 springboot-dubb ...

  3. AnnotationConfigApplicationContext.的用法的核心代码

    public static void main(String[] args) {ApplicationContext ctx = new AnnotationConfigApplicationCont ...

  4. Eclipse来push,fetch,rebase代码

    如何与项目里的其他人一起合作项目,提交代码并更新呢?这里提出我比最近用到的两种工具:一种是Eclipse,另外一个是SourceTree.个人推荐从事Java开发的话,可以用Eclipse.当然,还有 ...

  5. Flink源码阅读(一)--Checkpoint触发机制

    Checkpoint触发机制 Flink的checkpoint是通过定时器周期性触发的.checkpoint触发最关键的类是CheckpointCoordinator,称它为检查点协调器. org.a ...

  6. 【Foreign】Research Rover [DP]

    Research Rover Time Limit: 25 Sec  Memory Limit: 256 MB Description Input Output 仅一行一个整数表示答案. Sample ...

  7. 通过监测DLL调用探测Mimikatz

    通过Sysmon的-l参数可以探测到DLL加载(ImageLoaded): REF: https://securityriskadvisors.com/blog/post/detecting-in-m ...

  8. kuangbin 带你飞 数学基础

    模版整理: 晒素数 void init() { cas = ; ; i < MAXD ; i++) is_prime[i] = true; is_prime[] = is_prime[] = f ...

  9. python 微博评论获取的时候日期格式化

    # -*- coding: utf-8 -*- # @Time : 2018/03/05 10:57 # @Author : cxa # @File : testDataTime.py # @Soft ...

  10. java 定时器的三种方式

    原地址:http://blog.csdn.net/haorengoodman/article/details/23281343/ /** * 普通thread * 这是最常见的,创建一个thread, ...