团体程序设计天梯赛L3-005 垃圾箱分布

L3-005. 垃圾箱分布

时间限制
200 ms
内存限制
65536 kB
代码长度限制
8000 B
判题程序
Standard
作者
陈越

大家倒垃圾的时候,都希望垃圾箱距离自己比较近,但是谁都不愿意守着垃圾箱住。所以垃圾箱的位置必须选在到所有居民点的最短距离最长的地方,同时还要保证每个居民点都在距离它一个不太远的范围内。

现给定一个居民区的地图,以及若干垃圾箱的候选地点,请你推荐最合适的地点。如果解不唯一,则输出到所有居民点的平均距离最短的那个解。如果这样的解还是不唯一,则输出编号最小的地点。

输入格式:

输入第一行给出4个正整数:N(<= 103)是居民点的个数;M(<= 10)是垃圾箱候选地点的个数;K(<= 104)是居民点和垃圾箱候选地点之间的道路的条数;DS是居民点与垃圾箱之间不能超过的最大距离。所有的居民点从1到N编号,所有的垃圾箱候选地点从G1到GM编号。

随后K行,每行按下列格式描述一条道路:
P1 P2 Dist
其中P1和P2是道路两端点的编号,端点可以是居民点,也可以是垃圾箱候选点。Dist是道路的长度,是一个正整数。

输出格式:

首先在第一行输出最佳候选地点的编号。然后在第二行输出该地点到所有居民点的最小距离和平均距离。数字间以空格分隔,保留小数点后1位。如果解不存在,则输出“No Solution”。

输入样例1:
4 3 11 5
1 2 2
1 4 2
1 G1 4
1 G2 3
2 3 2
2 G2 1
3 4 2
3 G3 2
4 G1 3
G2 G1 1
G3 G2 2
输出样例1:
G1
2.0 3.3
输入样例2:
2 1 2 10
1 G1 9
2 G1 20
输出样例2:
No Solution

解题思路:最短路

#include <iostream>
#include <cstdio>
#include <string>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <vector>
#include <set>
#include <stack>
#include <map>
#include <climits>

using namespace std;

#define LL long long
const int INF=0x3f3f3f3f;

int cnt,n,m,k,d;
int s[10000],nt[100000],e[100000],l[100000];
int dis[10000],visit[10000];

struct node
{
    int id,dis;
    friend bool operator <(node a,node b)
    {
        return a.dis>b.dis;
    }
}pre1,nt1;

struct node1
{
    int id,midis;
    double ave;
}a[10000];

int get(char ch[])
{
    int ans=0,len=strlen(ch),p=1;
    if(ch[0]=='G')
    {
        for(int i=len-1; i>=1; i--)
        {
            ans+=(ch[i]-'0')*p;
            p*=10;
        }
        ans+=n;
    }
    else
    {
        for(int i=len-1; i>=0; i--)
        {
            ans+=(ch[i]-'0')*p;
            p*=10;
        }
    }
    return ans;
}

void Dijkstra(int ss)
{
    pre1.id=ss,pre1.dis=0;
    priority_queue<node>q;
    memset(visit,0,sizeof visit);
    memset(dis,INF,sizeof dis);
    dis[ss]=0;
    q.push(pre1);
    while(!q.empty())
    {
        pre1=q.top();
        q.pop();
        visit[pre1.id]=1;
        for(int i=s[pre1.id]; ~i; i=nt[i])
        {
            int ee=e[i];
            if(visit[ee]) continue;
            if(dis[ee]>dis[pre1.id]+l[i])
            {
                dis[ee]=dis[pre1.id]+l[i];
                nt1.dis=dis[ee];
                nt1.id=ee;
                q.push(nt1);
            }
        }
    }
}

bool cmp(node1 a,node1 b)
{
    if(a.midis!=b.midis) return a.midis>b.midis;
    else if(a.ave!=b.ave) return a.ave<b.ave;
    else return a.id<b.id;
}

int main()
{
    while(~scanf("%d %d %d %d",&n,&m,&k,&d))
    {
        cnt=1;
        memset(s,-1,sizeof s);
        memset(nt,-1,sizeof nt);
        char s1[10],s2[10];
        int ll;
        for(int i=1; i<=k; i++)
        {
            scanf("%s %s %d",s1,s2,&ll);
            int xx=get(s1),yy=get(s2);
            nt[cnt]=s[xx],s[xx]=cnt,e[cnt]=yy,l[cnt++]=ll;
            nt[cnt]=s[yy],s[yy]=cnt,e[cnt]=xx,l[cnt++]=ll;
        }
        int res=0;
        for(int i=n+1;i<=n+m;i++)
        {
            Dijkstra(i);
            double sum=0;
            int flag=1,mi=INF;
            for(int j=1;j<=n;j++)
            {
                if(dis[j]>d) {flag=0;break;}
                mi=min(mi,dis[j]);
                sum+=dis[j];
            }
            if(flag)
            {
                a[res].id=i;
                a[res].midis=mi;
                a[res++].ave=sum/n;
            }
        }
        if(!res) {printf("No Solution\n");continue;}
        sort(a,a+res,cmp);
        printf("G%d\n",a[0].id-n);
        printf("%.1lf %.1lf\n",(double)a[0].midis,a[0].ave);
    }
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值