题目
问题描述
在x轴上存在n个点 ( x 1 , 0 ) , ( x 2 , 0 ) , ( x 3 , 0 ) . . . ( x n , 0 ) (x_{1},0),(x_{2},0),(x_{3},0)...(x_{n},0) (x1,0),(x2,0),(x3,0)...(xn,0),以相邻两点 ( x i , 0 ) (x_{i},0) (xi,0)与 ( x i + 1 , 0 ) (x_{i+1},0) (xi+1,0)连线作为直径向上作半圆,问是否有除x轴上的交点。
输入格式
- 第1行是整数n,表示序列长度。
- 第2~n+1行每行一个整数表示序列中各数。
输出格式
- 对于每组输入输出一行,若相交,输出yes,否则输出no。
数据范围
100%的数据满足: 1 ≤ n ≤ 1 0 3 1\le{n}\le{10^3} 1≤n≤103, − 1 0 6 ≤ x i ≤ 1 0 6 -10^6\le x_{i} \le 10^6 −106≤xi≤106,最多五组数据。
输入/输出例子
输入
4
0 10 5 15
输出
yes
提示
无
解题思路
思路
因为 1 ≤ n ≤ 1 0 3 1\le{n}\le{10^3} 1≤n≤103
所以我们直接暴力枚举。
分析
根据数学书的某一页,两个圆一共有
5
5
5 种状态,分别是外离,外切,内切,相交,内含。其中有交点的是外切,内切,相交。它们的图像如下所示:
再根据数学书的某一页,若两圆相切,则切点在两圆连心线上,而两圆圆心都在
x
x
x 轴上,所以其连心线也一定和
x
x
x 轴重合,显然不为所求。所以我们就是要判断是否有两圆相交。
设两圆半径分别为
R
R
R 和
r
r
r,两圆的圆心距为
d
d
d,若两圆相交,满足:
R
−
r
<
d
<
R
+
r
R-r<d<R+r
R−r<d<R+r
由于我们知道了直径,所以圆心的位置是可以算出来的,所以圆心距也能算出来。其具体公式为:
O = x 1 + r O=x_{1}+r O=x1+r
所以我们就可以开始暴力出奇迹了。
Code
#include<bits/stdc++.h>
#define int long long
using namespace std;
int n,a[10001];
signed main()
{
scanf("%lld",&n);
for(int i=1;i<=n;i++)
scanf("%lld",&a[i]);
for(int i=1;i<n;i++)
for(int j=1;j<n;j++)
if(i!=j)
{
double r1=abs(a[i+1]-((a[i+1]+a[i]))/2.0),r2=abs(a[j+1]-((a[j+1]+a[j]))/2.0),d=abs((a[j+1]+a[j])/2.0-(a[i+1]+a[i])/2.0);
if(d<abs(r2+r1)&&d>abs(r2-r1))
puts("yes"),exit(0);
}
puts("no");
}