HDU - 4514 湫湫系列故事——设计风景线(并查集判环)

题目:

随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形,那就建的越长越好。 
现在已经勘探确定了n个位置可以用来建设,在它们之间也勘探确定了m条可以设计的路线以及他们的长度。请问是否能够建成环形的风景线?如果不能,风景线最长能够达到多少? 
其中,可以兴建的路线均是双向的,他们之间的长度均大于0。

思路:

将给出的边的两个端点用并查集放在一起,如果这两个点的祖先相等说明构成了一个环。

在这个用并查集连成的连通分量里边,找到他的root(fa[i] = i),然后先用一个BFS找到他的端点,然后从这个端点开始找树上的最长距离。

代码:

#include <bits/stdc++.h>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <vector>
#include <algorithm>
#include <queue>
#define MAX 1000000000
#define inf 0x3f3f3f3f
#define FRE() freopen("in.txt","r",stdin)

using namespace std;
typedef unsigned long long ll;
const int maxn = 100005;
int n,m;
struct Edge
{
    int to,w;
};
vector<Edge> mp[maxn];
int fa[maxn],vis[maxn],d[maxn];

int _find(int x)
{
    return fa[x] == x ? x : fa[x] = _find(fa[x]);
}

bool judgeLoop(int u,int v)
{
    int x = _find(u);
    int y = _find(v);
    if(x!=y)
    {
        fa[x] = y;
        return false;
    }
    return true;//x==y说明两者已经在一个连通分量里边了,也就是有环了
}

int BFS(int s)//两个作用1.查找最远的结点2.计算最长的距离
{
    memset(vis,0,sizeof(vis));
    memset(d,0,sizeof(d));
    vis[s] = 1;
    queue<int> que;
    que.push(s);
    int ed = s,mmax=0;
    while(!que.empty())
    {
        int u = que.front();
        que.pop();
        for(int i=0; i<mp[u].size(); i++)
        {
            Edge e = mp[u][i];
            if(vis[e.to] == 0)
            {
                d[e.to] = d[u]+e.w;//获取从根节点到e.to结点的距离
                vis[e.to] = 1;
                que.push(e.to);
                if(d[e.to]>mmax)//找到这棵树的距离根节点最远的结点
                {
                    ed = e.to;
                    mmax = d[e.to];
                }
            }
        }
    }
    return ed;
}

int main()
{
    //FRE();
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        for(int i=0; i<maxn; i++) { fa[i] = i; mp[i].clear();}
        bool ok = false;
        for(int i=0; i<m; i++)
        {
            int u,v,w;
            scanf("%d%d%d",&u,&v,&w);
            mp[u].push_back(Edge{v,w});
            mp[v].push_back(Edge{u,w});
            if(judgeLoop(u,v)) { ok = true; }
        }
        if(ok)
        {
            printf("YES\n");
            continue;
        }
        int ans = 0;
        for(int i=1; i<=n; i++)
        {
            if(fa[i]==i)//找到这个树的根节点(人为设置的)
            {
                int root = BFS(i);
                int temp = BFS(root);
                ans = max(ans,d[temp]);
            }
        }
        printf("%d\n",ans);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/sykline/p/10515583.html

时间: 2024-08-03 17:10:39

HDU - 4514 湫湫系列故事——设计风景线(并查集判环)的相关文章

HDU4514湫湫系列故事——设计风景线(并查集判环+最长直径)

题目链接:传送门 题意: 先判一个图是否存在换,不存在的话输出这个图的最长路径. 代码如下: #include <iostream> #include <cstring> #include <algorithm> #include <cstdio> #include <queue> using namespace std; const int maxn = 1e6+10; int par[maxn/10],head[maxn/10]; bool

HDU 4514 湫湫系列故事——设计风景线(并查集+树形DP)

湫湫系列故事——设计风景线 Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Total Submission(s): 4669    Accepted Submission(s): 853 Problem Description 随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形

HDU - 4514 湫湫系列故事——设计风景线(并查集判环+树形DP)

题目链接:https://vjudge.net/problem/HDU-4514 随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形,那就建的越长越好.  现在已经勘探确定了n个位置可以用来建设,在它们之间也勘探确定了m条可以设计的路线以及他们的长度.请问是否能够建成环形的风景线?如果不能,风景线最长能够达到多少?  其中,可以兴建的路线均是双向的,他们之间的长度均大于0. Input 测试数据有

hdu-----(4514)湫湫系列故事——设计风景线(树形DP+并查集)

湫湫系列故事——设计风景线 Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total Submission(s): 3105    Accepted Submission(s): 562 Problem Description 随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形,

HDU - 4514 湫湫系列故事――设计风景线

Description 随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形,那就建的越长越好. 现在已经勘探确定了n个位置可以用来建设,在它们之间也勘探确定了m条可以设计的路线以及他们的长度.请问是否能够建成环形的风景线?如果不能,风景线最长能够达到多少? 其中,可以兴建的路线均是双向的,他们之间的长度均大于0. Input 测试数据有多组,每组测试数据的第一行有两个数字n, m,其含义参见题目描

hdu 4514 湫湫系列故事――设计风景线(求树的直径)

随着杭州西湖的知名度的进一步提升,园林规划专家湫湫希望设计出一条新的经典观光线路,根据老板马小腾的指示,新的风景线最好能建成环形,如果没有条件建成环形,那就建的越长越好.  现在已经勘探确定了n个位置可以用来建设,在它们之间也勘探确定了m条可以设计的路线以及他们的长度.请问是否能够建成环形的风景线?如果不能,风景线最长能够达到多少?  其中,可以兴建的路线均是双向的,他们之间的长度均大于0. Input 测试数据有多组,每组测试数据的第一行有两个数字n, m,其含义参见题目描述:  接下去m行,

HDU 4514 湫湫系列故事――设计风景线 (树形DP)

题意:略. 析:首先先判环,如果有环直接输出,用并查集就好,如果没有环,那么就是一棵树,然后最长的就是树的直径,这个题注意少开内存,容易超内存, 还有用C++交用的少一些,我用G++交的卡在32764K,限制是32768K.. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024000000") #include <cstdio> #include <string> #include <cstdlib&

HDU ACM 4514 湫湫系列故事——设计风景线-&gt;树上环的判断+树上最长路

题意:一个无向图,首先判定是否成环,然后求一条最长链. 分析:成环用并查集判断,最长链就是树形dp了. #include<iostream> #include<vector> #pragma comment(linker, "/STACK:1024000000,1024000000") using namespace std; #define N 100005 int ans; int dp[N]; //dp[i]表示i节点为根节点树的最长路 int p[N];

hdu 4502 吉哥系列故事——临时工计划

吉哥系列故事--临时工计划 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Total Submission(s): 2921    Accepted Submission(s): 1128 Problem Description 俗话说一分钱难倒英雄汉,高中几年下来,吉哥已经深深明白了这个道理,因此,新年开始存储一年的个人资金已经成了习惯,不过自从大学之后他不好意思再向大人要