BZOJ 1027 [JSOI2007]合金 ——计算几何

我们可以把每一种金属拆成一个二维向量,显然第三维可以计算出来,是无关的。

我们只需要考虑前两维的情况,显然可以构成点集所形成的凸包内。

然后我们枚举两两的情况,然后可以发现如果所有的点都在一侧是可以选的。

然后我们在点集中,找到一个有向最小环就可以了。

#include <map>
#include <cmath>
#include <queue>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define inf 0x3f3f3f3f
#define eps 1e-8
#define ll long long
#define mp make_pair
 
struct Vector{
    double x,y;
    void print()
    {
        printf("Vector - > (%.3f,%.3f)\n",x,y);
    }
};
 
struct Point{
    double x,y;
    void print()
    {
        printf("Point  (%.3f,%.3f)\n",x,y);
    }
};
 
double operator * (Vector a,Vector b)
{
    return a.x*b.y-a.y*b.x;
}
 
Vector operator - (Point a,Point b)
{Vector ret;ret.x=a.x-b.x;ret.y=a.y-b.y;return ret;}
 
double dot(Vector a,Vector b)
{
    return a.x*b.x+a.y*b.y;
}
 
Point a[505],b[505];
 
int n,m,c[505][505];
 
bool judge(Point x,Point y)
{
    F(i,1,m)
    {
        double cro=(x-b[i])*(y-b[i]);
        if (cro>eps) return 0;
        if (fabs(cro)<eps&&dot(x-b[i],y-b[i])>eps) return 0;
    }
    return 1;
}
 
bool SPJ()
{
    F(i,1,n) if (fabs(a[i].x-a[1].x)>eps||fabs(a[i].y-a[1].y)>eps) return 0;
    F(i,1,m) if (fabs(b[i].x-a[1].x)>eps||fabs(b[i].y-a[1].y)>eps) return 0;
    return 1;
}
 
int main()
{
    scanf("%d%d",&n,&m);
    F(i,1,n) scanf("%lf%lf%*lf",&a[i].x,&a[i].y);
    F(i,1,m) scanf("%lf%lf%*lf",&b[i].x,&b[i].y);
    if (!m) return printf("0\n"),0;
    if (SPJ()) return printf("1\n"),0;
    memset(c,0x3f,sizeof c);
    F(i,1,n) F(j,1,n) if (i!=j)
    if (judge(a[i],a[j])) c[i][j]=1;
    F(k,1,n) F(i,1,n) F(j,1,n)
        c[i][j]=min(c[i][k]+c[k][j],c[i][j]);
    int ans=inf;
    F(i,1,n) ans=min(ans,c[i][i]);
    printf("%d\n",ans==inf?-1:ans);
}

  

posted @ 2017-04-09 20:09  SfailSth  阅读(187)  评论(0编辑  收藏  举报