图-1

概念

略。
是比树更一般的结构。可以考虑转化为树(半线性),再转化为线性结构。

存储

  • 邻接矩阵:浪费空间,增删顶点慢,遍历邻居慢,但循秩访问使得其它静态操作快。
  • 邻接表:省空间,对顶点操作快,遍历邻居快。
  • 关联矩阵:略了。

图遍历

不仅仅能把图染一遍色,还顺带把边分成树边和跨边,画出一棵搜索树来。这里我在遍历过程中都略过了对边的分类。

为了便于后面玩,先画一棵丑丑的树,顶点给了个结构,边就直接表示在二维数组graph里了。以及一个把status重置的函数。

const int N = 8;
struct vertice{
    int data;
    int status;
    int dTime;
    int fTime;
}vertices[N];
int graph[N][N] = {{0, 0, 1, 0, 1, 0, 0, 0}, {0, 0, 0, 0, 0, 0, 0, 0}, 
{0, 1, 0, 0, 0, 0, 0, 0}, {0, 1, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 1, 1, 0}, {0, 0, 0, 0, 0, 1, 0, 0},
{0, 1, 0, 0, 0, 1, 0, 0}, {1, 0, 1, 1, 0, 0, 0, 0}};

void clearStatus(){
    for(int i = 0; i < N; i ++){
        vertices[i].status = 0;    
    }
}

BFS

首先访问到的就是进来的顶点。在搜索过程中会依次访问顶点的邻居、邻居的邻居,效果正是这棵搜索树的层次遍历。在实现上,也同样是拉出一个栈来。(我说学树的时候怎么觉得层次遍历越看越顺眼呢。)

这都是写过的东西了,再顺手写一遍啦:

void BFS(int v){
    int queue[N], start = 0, end = 1;
    queue[0] = v;
    vertices[queue[0]].status = 1;
    while(end > start){
        v = queue[start ++];
        for(int i = 0; i < N; i ++){
            if(graph[v][i] && vertices[i].status == 0){
                queue[end ++] = i;
                vertices[i].status = 1; //discovered和visited一并标为1
            }
        }
        printf("%d\t", v);
    }
}

void bfs(){
    clearStatus();
    for(int i = 0; i < N; i ++){
        if(vertices[i].status == 0){
            BFS(i);
        }
    }
}

DFS

DFS就是一个很简洁的递归思路:对该顶点顺着边到达的每一个邻居都做DFS。这次相比之前写过的DFS,增加了一个clock标记dTimefTime,即发现时间和访问完毕时间。

void DFS(int v, int & clock){
    vertices[v].dTime = (++clock);
    vertices[v].status = 1;
    for(int i = 0; i < N; i ++){
        if(graph[v][i]){
            if(vertices[i].status == 0){
                DFS(i, clock);
            }
        }
    }
    vertices[v].fTime = (++clock);
}

void dfs(){
    clearStatus();
    int clock = 0;
    for(int i = 0; i < N; i ++){
        if(vertices[i].status == 0){
            DFS(i, clock);
        }
    }
}

对于dTime还有其他在递归之前做的操作,都类似于对树的先序遍历;而对于fTime及其他附近的操作,都类似于后序遍历。

还有一个很神奇的引理:若在dfs遍历树中顶点uv的直系祖先,则[u.dTime, u.fTime]包含[v.dTime, v.fTime];如果没有直系关系,则这两个区间不相交。我还没有看到除了判断辈份关系外的其它用途,但据说很有用……

回顾

前面做过的题中,milk3, castle都是用图遍历算法实现的。不过当时只知道自己在搜索,好像和图论这一套联系不起来。

castle

给个地图,想知道城堡有多少个房间,每个房间有多大。另外,农夫约翰想要把一面单独的墙(指两个单位间的墙)拆掉以形成一个更大的房间。 你的工作就是帮农夫约翰做以上的准备,算出房间数与房间的大小。

当时我记得看了很久的floodfill,现在看来基于dfs或者bfs都是可以的,不过就是换个连通域的时候标记一下就好了。其实每个格子都是个顶点,与临近格子之间没有墙的话就相当于有边,就这样搜索邻居就是了。

重写得跟之前几乎长得一样,还调了一堆bug才完事,不好意思挂上来了。

milk3

农民约翰有三个容量分别是A,B,C升的桶,A,B,C分别是三个从1到20的整数, 最初,A和B桶都是空的,而C桶是装满牛奶的。有时,农民把牛奶从一个桶倒到 另一个桶中,直到被灌桶装满或原桶空了。当然每一次灌注都是完全的。由于节约, 牛奶不会有丢失。
写一个程序去帮助农民找出当A桶是空的时候,C桶中牛奶所剩量的所有可能性。

每个状态是一个顶点,能通过一步倒牛奶能达到的状态就是跟它邻接的。虽然我当时根本不觉得这是个图论(手动跟智商再见)。

dijkstra算法

我先写了一个野版本,现在看来是我不能证明正确的,而且很可能确实是错误的,就不放了(≧▽≦)

我又乖乖地看了一下比较标准的写法,也就是在选择下一个处理的顶点上,这里将所有顶点分为两个集合:最短路径已知的集合S,和最短路径未知(可能有个值,但肯定还不能确定它是最优值)的集合V - S。这里要从V - S中取出到原点最近的顶点u,就可以将它加入S(确认这个点目前已经得到了最短路径),然后对它的邻居们做一遍更新。

这是代码,在循环之前,要将原点的邻居们的最短路径填入,并且将原点加入集合S(在这里我把status置1,表示在集合S中)。

int dijkstra(int from, int to){
    int dist[N];
    int u, minDist;

    //status初始化
    clearStatus();
    vertices[from].status = 1;

    //dist初始化
    for(int i = 0; i < N; i ++) dist[i] = 1<<30;
    dist[from] = 0;
    for(int i = 0; i < N; i ++){ 
        if(graph[from][i]) dist[i] = graph[from][i];
    }
    
    while(1){
        //找到最短路未知的里面距离最小的
        minDist = 1<<30;
        for(int i = 0; i < N; i ++){ 
            if(vertices[i].status == 0 && dist[i] < minDist){
                minDist = dist[i];
                u = i;
            }
        }
        if(minDist == 1<<30) break;

        vertices[u].status = 1;
        for(int i = 0; i < N; i ++){ 
            if(graph[u][i]){
                int newDist = dist[u] + graph[u][i];
                if(newDist < dist[i]) dist[i] = newDist;
            }
        }
    }

    return dist[to];
}

我之前的写法很可能是错在“选择下一个顶点”不正确。毕竟,很难一眼就看懂为什么这么写是对的,尤其是为什么要选V - S中距原点最近的顶点u,而且还拍着胸脯说:来吧,你现在手里这个就已经是最短路径了。CSDN上的这个帖子的证明让我看懂了。下面我再重现一下小剧场:

  • 顶点u:你怎么敢说我现在手里就是最短路径?
  • 回复:你现在手里这条,一定是 S 中某倒霉顶点 v 的最短路径,再加上 v 到你那里的距离。
  • 顶点u:那如果 S 中另一个更倒霉的 v' 的最短路径,再加上 v' -> u,要比我手里的这条要短呢?
  • 回复:怎么可能,人家 v' 在早先加入 S 的时候,就已经把它的邻居们更新过一遍了,人家不会藏着掖着更短的路径不告诉你的。
  • 顶点u:那如果我的上一个节点不必在 S 里面呢?比如 S 中某顶点 v 的最短路径,再到 (V - S) 中的 v'' ,到 v''',再到我,才是最短的呢?
  • 那(V - S) 中的 v'' 、v''' 岂不是最短路径都比你短?你怎么当这里面最短的?
  • 顶点无言以对,乖乖地走到了 S 集合中——来到了我人生导师的家里,而人生导师笑眯眯地反锁上了门:“又骗来一个”。

吐槽

昨天更新了vs code的c/c++插件,也可能是我之前没注意,也可能是这个插件读了更多头文件里的东西,这两天我写什么它都在给我提示:这个在Graph::里面已经有了。

想不到dijkstra也有了

根据教程,C++里面还可以用优先级搜索算法框架,实现bfs,dfs,dijkstra等等算法。确实,这些算法的差异“就在于每一步迭代中对新顶点的选取策略不同”。bfs是优先取更早被发现的顶点,dfs则是取更晚被发现的顶点,dijk则是选距离最近的顶点。虽然不会写,但是看它们priorityUpdater的代码:
bfs

dijk

只差一个边权被换成了1而已。“BFS实际上完全等效于,在所有边的权重统一为1个单位的图中,采用Dijkstra算法构造最短路径树。 从这个意义上讲, BFS也可以视作Dijkstra算法的一个特例。”好神奇……

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容