/* (⊙v⊙)嗯 貌似是一个建图 拓扑+深搜的过程。至于为什么要深搜嘛。。一个月前敲得题现在全部推了重敲,于是明白了。因为题意要求如果有多个可能的解的话。
 * 就要输出字典序最小的那个。所以可以对26个英文字母从小到大尝试能否排出结果。于是出现了 深搜回溯。先选定入度为0的边框。标记为已用。将所有与它连通的
 * 边框入度减一。然后递归搜索下一个。此时开始回溯。当前边框标记为未用,所有与它连通的边框入度加1.
 * 建图的过程则是。对每一个字母的边框测量出来。然后对A里的其它字母B。(B 覆盖 A)。有关系 map[A][B] = *1。in[B]++;
 */

#include<stdio.h>
#include<iostream>
#include<string.h>
#include<algorithm>
using namespace std;
#define maxn 27

int map[maxn+5][maxn+5];    //建的图
char m[35][35];     //输入
int h, w;     // 输入size
int tot;  //有多少种字母边框
int in[maxn];  //每种字母对应的入度。
char ans[maxn];   // 存储答案
bool flag;

struct Node   // 存储每个字母对应的边框边界
{
    int lo, hi;
    int le, ri;
}node[maxn+5];

void buildmap()
{
    for (int i=0; i<h; ++i)  // 测量边界
    {
        for (int j=0; j<w; ++j)
        {
            if (m[i][j] == '.')
                continue;
            int temp = m[i][j] - 'A';
            if (in[temp] == -1)
            {
                in[temp] = 0;
                tot++;
            }
            if (i > node[temp].hi) node[temp].hi = i;
            if (i < node[temp].lo) node[temp].lo = i;
            if (j < node[temp].le) node[temp].le = j;
            if (j > node[temp].ri) node[temp].ri = j;
        }
    }

for (int k=0; k<maxn; ++k)   //建图
    {
        if (in[k] == -1)
            continue;
        for (int i=node[k].lo; i<=node[k].hi; ++i)
        {
            for (int j=node[k].le; j<=node[k].ri; ++j)
            {
                if (i>node[k].lo && i<node[k].hi && j>node[k].le && j<node[k].ri)
                    continue;
                int temp = m[i][j] - 'A';
                if (k != temp && !map[k][temp])
                {
                   map[k][temp] = 1;
                   in[temp] += 1;
                }
            }
        }
    }
}
int hh = 0;

void dfs(int num)
{
    if (num == tot)
    {
        ans[num] = '\0';
        cout << ans << endl;
        return;
    }

//觉得应该会有字典序不同的答案。所以在输出加上判断。然而。wa了。应该是递归没有理解清楚。
//    if (num == tot && flag)
//    {
//        ans[num] = '\0';
//        puts(ans);
//        flag = false;
//        return;
//    }

for (int i=0; i<maxn; ++i)
    {
        if (in[i] == 0)
        {
            ans[num] = i + 'A';
            in[i] = -1;
            for (int j=0; j<maxn; j++)
            {
               if (map[i][j])
                 in[j] -= 1;
            }
            dfs(num+1);
            in[i] = 0;
            for (int j=0; j<maxn; ++j)
            {
               if (map[i][j])
                 in[j] += 1;
            }
        }
    }
    return;
}

int main()
{
    while (scanf("%d%d", &h, &w) == 2)
    {
        tot = 0;
        memset(in, -1, sizeof(in));
        memset(map, 0, sizeof(map));
        flag = true;
        for (int i=0; i<maxn+5; ++i)
        {
            node[i].lo = 100;
            node[i].le = 100;
            node[i].hi = -1;
            node[i].ri = -1;
        }
        for (int i=0; i<h; ++i)
        {
            scanf("%s", m[i]);
        }
        buildmap();
        dfs(0);
    }
    return 0;
}

POJ 1128 拓扑排序 + 深搜的更多相关文章

  1. poj 3687(拓扑排序)

    http://poj.org/problem?id=3687 题意:有一些球他们都有各自的重量,而且每个球的重量都不相同,现在,要给这些球贴标签.如果这些球没有限定条件说是哪个比哪个轻的话,那么默认的 ...

  2. POJ 3249 拓扑排序+DP

    貌似是道水题.TLE了几次.把所有的输入输出改成scanf 和 printf ,有吧队列改成了数组模拟.然后就AC 了.2333333.... Description: MR.DOG 在找工作的过程中 ...

  3. poj 3249 拓扑排序 and 动态规划

    思路:我们首先来一遍拓扑排序,将点按先后顺序排列于一维数组中,然后扫描一遍数组,将每个点的出边所连接的点进行更新,即可得到最优解. #include<iostream> #include& ...

  4. poj 2585 拓扑排序

    这题主要在于建图.对9个2*2的小块,第i块如果出现了不等于i的数字,那么一定是在i之后被brought的.可以从i到该数字建一条边. 图建好后,进行一次拓扑排序,判段是否存在环.若存在环,那么就是B ...

  5. Sorting It All Out POJ - 1094 拓扑排序

    题意:给N个字母,和M个偏序关系 求一个可确定的全序,可确定是指没有其他的可能例如A>B D>B 那么有ADB DAB两种,这就是不可确定的其中,M个偏序关系可以看做是一个一个按时间给出的 ...

  6. poj 3984 -- 迷宫问题 深搜

    迷宫问题 Description 定义一个二维数组: int maze[5][5] = { 0, 1, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, ...

  7. nyoj 349 (poj 1094) (拓扑排序)

    Sorting It All Out 时间限制:3000 ms  |  内存限制:65535 KB 难度:3   描述 An ascending sorted sequence of distinct ...

  8. POJ 1128 Frame Stacking 拓扑排序+暴搜

    这道题输出特别坑.... 题目的意思也不太好理解.. 就解释一下输出吧.. 它让你 从下往上输出. 如果有多种情况,按照字典序从小往大输出... 就是这个多种情况是怎么产生的呢. 下面给一组样例. 很 ...

  9. Poj(3687),拓扑排序,

    题目链接:http://poj.org/problem?id=3687 题意:n个重量为1~n的球,给定一些编号间的重量比较关系,现在给每个球编号,在符合条件的前提下使得编号小的球重量小.(先保证1号 ...

随机推荐

  1. 20145331魏澍琛《网络对抗》Exp4 恶意代码分析

    20145331魏澍琛<网络对抗>Exp4 恶意代码分析 基础问题回答 1.如果在工作中怀疑一台主机上有恶意代码,但只是猜想,所有想监控下系统一天天的到底在干些什么.请设计下你想监控的操作 ...

  2. Wireshark 捕捉本地数据 --WinPcap切换NPcap

    Wireshark默认匹配安装的是WinPcap,但是WinPcap有个缺点,不能抓取本地回环数据 NPcap是在WinPcap的基础上进行优化开发的,可以抓取本地数据 如果已安装WinPcap的请卸 ...

  3. vue.js组件(component)

    简介: 组件(Component)是 Vue.js 最强大的功能之一. 组件可以扩展 HTML 元素,封装可重用的代码. 组件系统让我们可以用独立可复用的小组件来构建大型应用,几乎任意类型的应用的界面 ...

  4. Unity3D学习笔记(二十五):文件操作

    文件是什么? 存储在硬盘上的最后的节点. 文件夹是什么? 文件的上级单位称为文件夹. 文件夹的基本结构? 文件夹是层级化结构的,对于同级的文件夹不可以重名,父文件夹和子文件夹可以同名> IO:I ...

  5. [Err] 1111 - Invalid use of group function

    本文为博主原创,未经允许不得转载: 初衷,本想通过group by sql语句查询出不同id下总数在一定范围内的数据,所以产生如下的sql,及错误sql AND STATDATE < ' GRO ...

  6. Codeforces Beta Round #94 div 2 B

    B. Students and Shoelaces time limit per test 2 seconds memory limit per test 256 megabytes input st ...

  7. MongoDB(课时20 游标)

    3.5 游标(重点) 所谓游标就是指数据可以一行行的进行操作,非常类似于ResultSet数据处理.在MongoDB里对游标的控制使用find()函数就可以返回游标.对于返回的游标如果想进行操作,使用 ...

  8. HDU 1540 Tunnel Warfare

    HDU 1540 思路1: 树状数组+二分 代码: #include<bits/stdc++.h> using namespace std; #define ll long long #d ...

  9. Unity生成屏幕快照

    public static Texture2D CaptureCamera(Camera camera, Rect rect) { RenderTexture rt = ); RenderTextur ...

  10. cmd 常用操作

    2017-08-24 16:05:28 cd : change directory(目录) 功能:显示当前目录 类型:内部命令 格式:CD[盘符:][路径名][子目录名] 说明:CD命令不能改变当前所 ...