D-Mocha and Railgun

题目

https://ac.nowcoder.com/acm/contest/33186/D

解题

找最长弧

通过几何画板绘图

alt

发现当发射方向垂直于发射点与圆心的连线时,破坏圆弧最长

补个直观证明图,来自雨巨讲题

alt

分类讨论

每个样例给发射点QQ点坐标(x,y)(x,y),发射半径dd

可求得发射点到圆心的圆心距L=x2+y2L=\sqrt{x^2+y^2}

当原点在发射半径外时

L>dL>d

alt

在三角形AOEAOE上,可求得α=arcsin(L+d)\alpha=\arcsin(L+d)

在三角形IOJIOJ上,可求得β=arcsin(Ld)\beta=\arcsin(L-d)

对所求圆弧有=r(αβ)弧长=r*(\alpha-\beta)

当原点在发射半径内时

L<dL<d

alt

在三角形AOEAOE上,可求得α=arcsin(d+L)\alpha=\arcsin(d+L)

在三角形IOJIOJ上,可求得β=arcsin(dL)\beta=\arcsin(d-L)

对所求圆弧有=r(α+β)弧长=r*(\alpha+\beta)

代码

由上述逻辑得

#include<bits/stdc++.h>
using namespace std;
int T;
int main(){
    cin>>T;
    double r,x,y,d,L;
    while(T--){
        cin>>r>>x>>y>>d;
        L=sqrt(x*x+y*y);
        if(L>d){
            printf("%.8lf\n",r*(asin((L+d)/r)-asin((L-d)/r)));
        }
        else{
            printf("%.8lf\n",r*(asin((d+L)/r)+asin((d-L)/r)));
        }
    }
    return 0;
}

然而其实有arcsin(x)=arcsin(x)\arcsin(-x)=-\arcsin(x),故可以化简

#include<bits/stdc++.h>
using namespace std;
int T;
int main(){
    cin>>T;
    double r,x,y,d,L;
    while(T--){
        cin>>r>>x>>y>>d;
        L=sqrt(x*x+y*y);
        printf("%.8lf\n",r*(asin((L+d)/r)-asin((L-d)/r)));
    }
    return 0;
}