哨所巡视(二)

31 篇文章 0 订阅
7 篇文章 0 订阅

题面
Description

有N个哨所需要巡视,在哨所停留的时间为Ci,哨所之间有P双向道路连接,第j条双向道路连接了哨所Sj和Ej(1 <= Sj <= N; 1 <= Ej <= N; Sj != Ej),而且走完它需要Lj的时间。没有两个哨所是被一条以上的道路所连接。从某一个哨所出发,去巡视所有的基站,然后照原路回到出发的哨所,求巡视所有哨所所需最少时间,包括出发哨所两次停留的时间。

Input

第1行包含两个整数N和P。
接下来N行,每行包含一个整数Ci。
接下来P行,每行包含三个整数Sj, Ej和Lj。

Output

输出一个整数, 所需要的总时间。

Sample Input
5 7
10
10
20
6
30
1 2 5
2 3 5
2 4 12
3 4 17
2 5 15
3 5 6
4 5 12

Sample Output
176

HINT

5 <= N <= 10000,N-1 <= P <= 100000,0 <= Lj <= 1000,1 <= Ci <= 1,000。

题目分析
着题的难点无非是对边的权值和入口的处理
同时要求的联通处理无非也是最小生成树

这里给出其示意图:
在这里插入图片描述
这里给出解释:
因为题目里说是一趟来回并且要算上来往在哨所的休息的时间,所以对上面的就很好理解了

但是最后一点便是起点问题了,因为最小的生成树是连通图,所以起点的选择我们可以直接用最小的休息的点,毕竟点之间是互达的,继续用上面的来表示的话便是 min(x1,x2,x3,x4)

代码

#include <bits/stdc++.h>
using namespace std;
 
struct node
{
    int u;
    int v;
    int w;
}e[100009];
 
int a[10009],f[10009];
int n,m,mina = 999999999; 
 
void build()       //构图,按上面说的来
{
    cin>>n>>m;
    for (int i = 1; i <= n; i++)
        cin>>a[i],
        mina = min(mina,a[i]);
    for (int i = 1; i <= m; i++)
        cin>>e[i].u>>e[i].v>>e[i].w,
        e[i].w += e[i].w + a[e[i].u] + a[e[i].v];
} 
 //下面全是模板。。。。
bool cmp(node x,node y)
{
    return x.w < y.w;
}
 
int find(int x)
{
    if (f[x] == x) return x;
    f[x] = find(f[x]);
    return f[x];
}
 
void kruskal()
{
    long long ans = mina, cnt = 0;
     
    for (int i = 1; i <= n; i++) f[i] = i;
     
    sort(e+1,e+m+1,cmp);
     
    for (int i = 1; i <= m; i++)
        {
            int fu = find(e[i].u);
            int fv = find(e[i].v);
             
            if (fu != fv)
                {
                    f[fv] = f[fu];
                    ans += e[i].w;
                    cnt++;
                     
                    if (cnt == n-1) break;
                }
        }
     
    cout<<ans;
}
 
int main()
{
    build();
     
    kruskal();
     
    return 0;   
}
 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值