UVa 12123 - Magnetic Train Tracks

如果你能通过白书中级习题的前一题 , 那么这个题目还是很简单的 , 因为博主的数学部分没有进行专题研究 , 目前的理解还是比较基础的 , 可能对新手会有借鉴意义 , 但也不妨大神们来虐虐我啊(#^.^#)


提示:

1. 锐角三角形是比较难限制的 , 因为要求三个角都是锐角是吧

2. 但是如果我能找到一个钝角 , 那么我就一定能确定一个钝角三角形

3. 时间肯定不能n^3去搞 , 所以你需要想办法高效的枚举 , 首先难以避免的两重循环是枚举一个基准点(然后极角排序 , 这是个老套路) , 因为是找一这个点为钝角的两条边 , 所以还需要枚举一个点 , 以此来构成一条边 , 然后我们要想办法高效的找到另外有多少条边满足要求 , 对了 , 就是利用枚举的单调性


注意: 这是我看前一个题目的疑惑 , 是关于枚举顺序的 , 枚举要求不重复 , 所以顺序很重要 ,所以我们都只记录某一条边一个方向的钝角(有点抽象 , 看代码)


#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <string>
#include <vector>
#include <deque>
#include <stack>
#include <algorithm>

using namespace std;
const double pi = acos(-1);
const double eps = 1e-10;

int dcmp(double a) { return fabs(a)<eps?0:(a<0?-1:1); }

struct points
{
	double x , y;
	void read(){ scanf("%lf%lf",&x,&y); } 
	points(double x = 0 , double y = 0):x(x),y(y){}
};

typedef points Vector;

Vector operator -(Vector a , Vector b) { return Vector(a.x-b.x , a.y-b.y); }

double angle(Vector a)
{
	return atan2(a.y, a.x);
}

int n , m;
points p[1500];
double s[3000];

int main(int argc, char *argv[]) {
	int Case=0;
	while(cin>>n && n)
	{
		for(int i=0;i<n;i++) p[i].read();
		
		long long res =0;
		for(int i=0;i<n;i++)
		{
			m=0;
			for(int j=0;j<n;j++) if(i!=j) s[m++] = angle(p[j]-p[i]);
			sort(s, s+m);
			for(int j=0;j<m;j++) s[j+m] = pi*2+s[j];
			for(int j=0 , k=0 , l=0;j<m;j++)
			{
				while(dcmp(s[k+1]-s[j]-pi/2.0)<0) k++;
				while(dcmp(s[l+1]-s[j]-pi)<0) l++;
				res+= l-k;
			}
		}
		long long now = (long long)n*(n-1)*(n-2)/6;
		cout<<"Scenario "<<++Case<<":\nThere are "<<now-res<<" sites for making valid tracks\n";
	}
	
	
	return 0;
}


你可能感兴趣的:(几何,uva)