POJ 1128-DAG拓扑排序

题目链接POJ1128

题意

给定一张边框覆盖图,求边框从上到下的关系

分析

如果在A的边框上出现了字母B,就说明B在A的上方

如果边框A在边框B的下方,就添加从A到B的一条有向边(题目要求从下到上输出)

那么所求的是所得有向无环图的拓扑排序

题目还要求按照字典序输出所有可能的顺序,用深度优先搜索

代码

#include <cstdio>
#include <cstring>
#include <vector>
#include <cmath>
#include <iostream>
#include <algorithm>
#define MAX 1000007
#define MAXN 50
#define MAXM 1007
#define INF  0x3f3f3f3f
#define NINF 0xc0c0c0c0
#define MOD 1000000007
using namespace std;
typedef long long LL;
int head[MAXN],tot,n,m,indegree[MAXN],realn;

struct Edge{
    int to;
    int next;
}edge[MAXM];

void addEdge(int u,int v){
    edge[tot].to=v;
    indegree[v]++;
    edge[tot].next=head[u];
    head[u]=tot++;
}

int visit[MAXN],Map[40][40],flag[40][40],df[MAXN];
struct Border{
    int x1,y1,x2,y2;
}border[MAXN];

int Queue[MAXN],iq;

void init(){
    tot=iq=realn=0;
    memset(head,-1,sizeof head);
    memset(visit,0,sizeof visit);
    memset(Map,0,sizeof Map);
    memset(flag,0,sizeof flag);
    memset(indegree,0,sizeof indegree);
    memset(df,0,sizeof df);
    for(int i=0;i<MAXN;i++){
        border[i].x1=border[i].y1=40;
        border[i].x2=border[i].y2=-1;
    }
}

//如果字母t在字母k的四条边框上,则连接k到t的有向边(从下到上的方向) 
void addE(int k,int x1,int y1,int x2,int y2){
    for(int i=x1;i<=x2;i++){
        for(int j=y1;j<=y2;j++){
            int t=Map[i][j];
            if(k!=t && !flag[k][t]){
                flag[k][t]=1;
                addEdge(k,t);
            }
        }
    }
}

//深度优先遍历出所有的拓扑排序 
void dfs(int now,int k){
    //cout<<now<<" "<<k<<endl;
    if(k==realn){
        for(int i=1;i<=realn;i++){
            printf("%c", Queue[i]+'A'-1);
        }
        printf("\n");
    }
    else{
        //删掉与now关联的边 
        for(int i=head[now];i!=-1;i=edge[i].next){
            indegree[edge[i].to]--;
        }
        //从剩余的点中选择入度为0的点,从小到大遍历 
        for(int i=1;i<MAXN;i++){
            if(!df[i]&&visit[i]&&indegree[i]==0){
                df[i]=1;
                Queue[k+1]=i;
                dfs(i,k+1);
                df[i]=0;
            }
        }
        //加回与now关联的边 
        for(int i=head[now];i!=-1;i=edge[i].next){
            indegree[edge[i].to]++;
        }
    }
}

int main(){
    char ch;
    while(scanf("%d%d",&n,&m)==2){
        init();
        for(int i=0;i<n;i++){
            for(int j=0;j<m;j++){
                scanf(" %c",&ch);
                if(ch!='.'){
                    Map[i][j]=ch-'A'+1;
                }
            }
        }
        for(int i=0;i<n;i++){
            for(int j=0;j<m;j++){
                int t=Map[i][j];
                if(t){
                    visit[t]=1;
                    border[t].x1=min(border[t].x1,i);
                    border[t].y1=min(border[t].y1,j);
                    border[t].x2=max(border[t].x2,i);
                    border[t].y2=max(border[t].y2,j);
                }
            }
        }
        for(int k=1;k<30;k++){
            if(visit[k]){
                addE(k,border[k].x1,border[k].y1,border[k].x1,border[k].y2);
                addE(k,border[k].x2,border[k].y1,border[k].x2,border[k].y2);
                addE(k,border[k].x1,border[k].y1,border[k].x2,border[k].y1);
                addE(k,border[k].x1,border[k].y2,border[k].x2,border[k].y2); 
            }
        }
        //cout<<tot<<endl;
        for(int i=1;i<=30;i++){
            if(visit[i]){
                realn++;
                addEdge(0,i);
            }
        }
        //cout<<realn<<endl;
        dfs(0,0);
    }
    return 0;
}
/*
6 6
BBBCCC
BAAAAC
BABCAC
.ADDA.
.AAAA.
..DD..
*/
View Code

 

posted @ 2017-06-17 17:02  水明  阅读(282)  评论(0编辑  收藏  举报