HDU 4435 charge-station (并查集)

  先说下题目的意思:

      在一个二维坐标系中有N个点,某人要来个走遍所有点的旅行,但是他的车每次加油后只能走M个单位距离;所以要在这个N点中选一些建立加油站;问题来了:i^th  点 建加油站的花费是  2^(i-1); 求最小话费 用二进制表示;(其中1号必须建立加油站)

  思路:有  10000>01111; 所以我们可以一开始都给这些个点染色(都建立加油站),然后从高位枚举这一位可以不建立加油站么?可以的话给他去除掉;依次类推;这样就可以维护这个“最小”;

  解法:上述思路的关键是给定一个染色方案如何判断是否合法:我的判断方法是并查集

        1)根据染色分俩堆;

        2)把建立加油站的点建立最小生成树,当距离大于M 时停止;

        3)看建立的树是否把所有的加油点囊括在内,有不再的肯定是false;

        4)没有加油站的点在   暴力判断下   有没有  离这个点的距离   小于 M/2 的加油站点;就可以了;没有就是false;

        5)至此结束;

#include <cstdio>
#include <string>
#include <algorithm>
#include <iostream>
#include <cstring>
#include <cmath>
#include <map>
using namespace std;
typedef long long LL;
struct point
{
    int x,y;
    void input()
    {
        scanf("%d%d",&x,&y);
    }
};
struct Edge
{
    int s,to;
    double dis;
    Edge(){}
    Edge(int s,int to,double dis):s(s),to(to),dis(dis){}
    bool operator < (const Edge &rht) const
    {
        return dis<rht.dis;
    }
};
point ko[129];
Edge  edge[129*129];
int DIS[129][129],fa[129], n,pos,m;
bool flag[129];
void inint(bool  t)
{
    pos=0;
    if(t)memset(flag,1,sizeof flag);
    for(int i=1;i<=128;i++)fa[i]=i;
}
int Find(int x)
{
    return x==fa[x]?x:fa[x]=Find(fa[x]);
}
double get_dis(point a,point b)
{
    int x=a.x-b.x;int y=a.y-b.y;
    return ceil(sqrt(1.0*(x*x+y*y)));
}
int Set1[129],pos1, Set2[129],pos2;
bool make()
{
    inint(false);
    pos1=pos2=0;
    for(int i=1;i<=n;i++)
        if(flag[i])  Set1[pos1++]=i;
        else         Set2[pos2++]=i;
    for(int i=0;i<pos1;i++) for(int j=i+1;j<pos1;j++)
    {
        double dis=DIS[Set1[i]][Set1[j]];

        edge[pos++]=Edge(Set1[i],Set1[j],dis);
    }
    sort(edge,edge+pos);
    for(int i=0;i<pos;i++)
    {
        Edge & tmp=edge[i];
        if(tmp.dis>m) break;
        int x=Find(tmp.s);
        int y=Find(tmp.to);
        if(x!=y) fa[x]=y;
    }
    int FA=Find(1);
    for(int i=0;i<pos1;i++)
        if(FA!=Find(Set1[i])) return false;
    for(int i=0;i<pos2;i++)
    {
        int j=0;
        for(;j<pos1;j++)
        {
            if(DIS[Set2[i]][Set1[j]]*2.0<=m) break;
        }
        if(j==pos1) return false;
    }
    return true;
}
void solve()
{
    for(int i=n;i>=2;i--)
    {
        flag[i]=false;
        if(make()) continue;
        flag[i]=true;
    }
    int i;
    for(i=n;i>=1;i--) if(flag[i]) break;
    for(;i>=1;i--)
        printf("%d",flag[i]);
    puts("");
}
int main()
{
    while(~scanf("%d %d",&n,&m))
    {
        inint(true);
        for(int i=1;i<=n;i++)ko[i].input();
        for(int i=1;i<=n;i++)for(int j=1;j<=n;j++)
            DIS[i][j]=get_dis(ko[i],ko[j]);
        if(!make()){puts("-1");continue;}
        solve();
    }
    return 0;
}

  

时间: 2024-12-20 12:26:18

HDU 4435 charge-station (并查集)的相关文章

hdu 3234 Exclusive-OR (并查集+异或性质)

Exclusive-OR Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 2177    Accepted Submission(s): 603 Problem Description You are not given n non-negative integers X0, X1, ..., Xn-1 less than 220 ,

HDU 4496 D-City (并查集)

题意:给你n个点m条边,问删除前i条边后有多少个连通分块. 思路:从后往前操作,从后往前添加i条边等于添加完m条边后删掉前m-i条边,可知刚开始没有边,所以sum[m]=n; #include <stdio.h> #include <iostream> #include <algorithm> #include <string.h> #include <queue> #include <math.h> #define M 100010

hdu 1811Rank of Tetris (并查集 + 拓扑排序)

1 /* 2 题意:这些信息可能有三种情况,分别是"A > B","A = B","A < B",分别表示A的Rating高于B,等于B,小于B. 3 4 现在Lele并不是让你来帮他制作这个高手榜,他只是想知道,根据这些信息是否能够确定出这个高手榜,是的话就输出"OK". 5 否则就请你判断出错的原因,到底是因为信息不完全(输出"UNCERTAIN"),还是因为这些信息中包含冲突(输出&quo

HDU 1232 畅通工程(并查集)

畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 30485    Accepted Submission(s): 16013 Problem Description 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府"畅通工程"的目标是使全省任何两个城镇间都可以实现交通(但不一定有

HDU 2017 Code Lock (并查集的应用+快速幂)

链接:HDU 3461 题目大意: 题目的大意是一个密码锁上有编号为1到N的N个字母,每个字母可以取26个小写英文字母中的一个.再给你M个区间[L,M],表示该区间的字母可以一起同步"增加"(从'a'变为'b'为增1,'z'增1为'a').假如一组密码按照给定的区间进行有限次的"增加"操作后可以变成另一组密码,那么我们认为这两组密码是相同的.该题的目标就是在给定N.M和M个区间的前提下计算有多少种不同的密码. 根据题意,如果一个可调整的区间都没有的话,答案应该是26

HDU 4775 Infinite Go(并查集,模拟)

HDU 4775 Infinite Go 题目链接 题意:围棋,两人轮流走,如果有一链被围死,就会被吃掉,问下完后最后黑色和白色各剩多少棋 思路:模拟,利用一个并查集来保存链,然后并记录下周围有多少个空格,然后去模拟,注意几个点,就是删除的时候,要把空格还回去,还有边界的位置是也算被围死的 代码: #include <stdio.h> #include <string.h> #include <queue> #include <map> using name

HDU 1198-Farm Irrigation(并查集)

Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 5820    Accepted Submission(s): 2523 Problem Description Benny has a spacious farm land to irrigate. The farm land is a rectangle

HDU 1198 Farm Irrigation (并查集)

Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 5809    Accepted Submission(s): 2516 Problem Description Benny has a spacious farm land to irrigate. The farm land is a rectangle,

hdu 3047 Zjnu Stadium 并查集高级应用

Zjnu Stadium Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 1631    Accepted Submission(s): 616 Problem Description In 12th Zhejiang College Students Games 2007, there was a new stadium built