首页 > 其他 > 详细

[NOIP 2019]旅行

时间:2019-09-20 23:27:51      阅读:140      评论:0      收藏:0      [点我收藏+]

题目链接

题意简介:现有一个图,小Y要把它走完,每个点只去一次,路径字典序最小。

分析:这道题我认为很重要的一个点就是它的数据范围。它只有两种 m=n-1 或 m=n。我们先考虑第一种:m=n-1也就是边为节点数减一,这种说法已经很隐晦了,其实这种情况就是树啊。树的遍历且字典序最小什么的应该很多人都会做吧,就只用深搜一下,这60分还是很好拿的。(由于每个点只做一次sort所以是不会超时的!)

技术分享图片
void dfs_60(int u,int fa){
    ans[++top]=u;
    int ver[1000],k;
    k=0;
    memset(ver,0,sizeof(ver));
    for(int i=head[u];i;i=next[i]){
        if(to[i]!=fa)ver[++k]=to[i];
    }
    sort(ver+1,ver+k+1);
    for(int i=1;i<=k;++i){
        dfs_60(ver[i],u);
    }
    return ;
}
View Code

然后我们要想个办法解决一个m=n的情况,并且在前面60的基础上加一些改动以达到ac的效果。那么m=n到底是个什么呢?他叫做:基环树。用我的理解来说,就是树里有一个环(也可以说是随便将一棵树上本不相连的两个点连接)。他大概长什么样子呢(m=n):技术分享图片

 

 em,基本上是这个样子。其实啊,基环树这个东西,换句话说,只要删去环中的一条边也就变成了一棵树(多少颗不同的树取决于环的大小),那么我们可不可以删呢?在这道题当中是可以的。因为每一个点只去一次,就是说不存在从环上某个点走出去再沿着环返回这个点的这种情况,最多只走k-1条边(k为组成环的边数)。那么这道题我们从这k条边中每次取一条边出来删,按照m=n-1计算出最佳答案,最后所有的答案取最佳的就好。

那么目前只有一个问题:如何找环?

我是用的DFS,由于这道题只可能有一个环(因为m=n),所以就直接遍历,用一个栈存可能是环的点,遍历到已遍历的点的时候,就全部出栈直到那个点也弹出去。如果这个点所有子树搜完都没有找到环,那么把这个点出栈就行了。

技术分享图片
void findh(int u,int fa){
    if(flag==1)return;
    for(int i=head[u];i;i=next[i]){
        if(to[i]!=fa){
            if(vis[to[i]]){
                while(to[i]!=st[top_st]){
                    h[++hsum]=st[top_st--];
                }
                h[++hsum]=to[i];
                flag=1;
            }
            else{
                vis[to[i]]=1;
                st[++top_st]=to[i];
                findh(to[i],u);
                if(flag==1)return;
                top_st--;
                vis[to[i]]=0;
            }
        
        }
    }
}
View Code

接下来就任意删边用m=n-1弄就行了。注意一边搜索一边剪枝,不然会超时。

完整代码:

技术分享图片
#include<iostream>
#include<cstdio>
#include<fstream>
#include<cmath>
#include<algorithm>
#include<cstring>
using namespace std;
int read(){
    char ch;
    int res=0,f=1;
    ch=getchar();
    while(ch<0||ch>9){
        if(ch==-)f=-1;
        ch=getchar();
    }
    while(ch>=0&&ch<=9){
        res=res*10+(ch-0);
        ch=getchar();
    }
    return res*f;
}
const int MAXN=20005;
int n,m,tot;
int head[MAXN],next[MAXN],to[MAXN];
int ans[MAXN],top;
int sz,sy,st[MAXN],top_st,vis[MAXN];
int h[MAXN],hsum,flag;
int lans[MAXN],ltop,flag_dfs;
void add(int x,int y){
    to[++tot]=y;
    next[tot]=head[x];
    head[x]=tot;
}
bool check(int u,int v){
    if(u==sz&&v==sy)return 0;
    if(u==sy&&v==sz)return 0;
    return 1;
}
void dfs_60(int u,int fa){
    ans[++top]=u;
    if(flag_dfs==0&&ans[top]<lans[top])flag_dfs=-1;
    if(flag_dfs==0&&ans[top]>lans[top])flag_dfs=1;
    if(flag_dfs==1)return;
    if(flag_dfs==-1&&top==n){
        for(int i=1;i<=top;++i){
            lans[i]=ans[i];
        }
        flag_dfs=1;
        return;
    }
    int ver[1000],k=0;
    //memset(ver,0,sizeof(ver));
    for(int i=head[u];i;i=next[i])if(to[i]!=fa&&check(u,to[i]))ver[++k]=to[i];
    sort(ver+1,ver+k+1);
    for(int i=1;i<=k;++i)dfs_60(ver[i],u);
    return ;
}
void findh(int u,int fa){
    if(flag==1)return;
    for(int i=head[u];i;i=next[i]){
        if(to[i]!=fa){
            if(vis[to[i]]){
                while(to[i]!=st[top_st]){
                    h[++hsum]=st[top_st--];
                }
                h[++hsum]=to[i];
                flag=1;
            }
            else{
                vis[to[i]]=1;
                st[++top_st]=to[i];
                findh(to[i],u);
                if(flag==1)return;
                top_st--;
                vis[to[i]]=0;
            }
        
        }
    }
}
int main(){
    n=read();m=read();
    for(int i=1;i<=m;++i){
        int u,v;
        u=read();v=read();
        add(u,v);add(v,u);
    }
    memset(lans,127/3,sizeof(lans));
    if(n==m+1){
        dfs_60(1,0);
        for(int i=1;i<=top;++i){
            printf("%d ",ans[i]);
        }
    }
    else{
        st[++top_st]=1;
        findh(1,0);
        sz=h[1],sy=h[hsum],top=0;
        dfs_60(1,0);
        for(int i=2;i<=hsum;++i){
            sz=h[i-1],sy=h[i],top=0,flag_dfs=0;
            //memset(ans,0,sizeof(ans));
            dfs_60(1,0);
            }
        }
        for(int i=1;i<=n;++i){
            printf("%d ",lans[i]);
        }
    }
    return 0;
}
View Code

注:我注释掉了一些初始化的操作,是因为反正存答案的时候会覆盖,只用把top清零就行了

 

[NOIP 2019]旅行

原文:https://www.cnblogs.com/clockwhite/p/11559895.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!