数据结构与算法--最短路径之Bellman算法、SPFA算法

数据结构与算法--最短路径之Bellman算法、SPFA算法

除了Floyd算法,另外一个使用广泛且可以处理负权边的是Bellman-Ford算法。

Bellman-Ford算法

假设某个图有V个顶点E条边。

该算法主要流程是:

  • 初始化。到起点s的距离distTo[s]设置为0,其余顶点的dist[]设置为正无穷;
  • 以任意次序放松图中的所有E条边,重复V轮;
  • V轮放松结束后,判断是否存在负权回路。如果存在,最短路径没有意义。

根据流程可以给出代码,如下

package Chap7;

import java.util.LinkedList;
import java.util.List;

public class BellmanFord {
    private boolean hasNegativeCycle;
    private double distTo[];
    private DiEdge[] edgeTo;

    public boolean hasNegativeCycle() {
        return hasNegativeCycle;
    }

    private void relax(EdgeWeightedDiGraph<?> graph, int v) {
        for (DiEdge edge : graph.adj(v)) {
            int w = edge.to();
            if (distTo[v] + edge.weight() < distTo[w]) {
                distTo[w] = distTo[v] + edge.weight();
                edgeTo[w] = edge;
            }
        }
    }

    public BellmanFord(EdgeWeightedDiGraph<?> graph, int s) {
        distTo = new double[graph.vertexNum()];
        edgeTo = new DiEdge[graph.vertexNum()];
        for (int i = 0; i < graph.vertexNum(); i++) {
            distTo[i] = Double.POSITIVE_INFINITY; // 1.0 / 0.0为INFINITY
        }

        distTo[s] = 0.0;
        // 以上都是初始化

        for (int pass = 0; pass < graph.vertexNum(); pass++) {
            for (int v = 0; v < graph.vertexNum(); v++) {
                relax(graph, v);
            }
        }

        // 上面即使有负权回路也不会陷入死循环,因为给定了循环范围,算法必然终止。
        // 进行V轮边的松弛后,如果没有负权回路,那么所有的distTo[v] + edge.weight() >= distTo[w]
        // 如果对于图中任意边,仍然存在distTo[v] + edge.weight() < distTo[w],则存在负权回路
        for (int v = 0; v < graph.vertexNum(); v++) {
            for (DiEdge edge : graph.adj(v)) {
                int w = edge.to();
                if (distTo[v] + edge.weight() < distTo[w]) {
                    hasNegativeCycle = true;
                }
            }
        }
    }

    public double distTo(int v) {
        return distTo[v];
    }

    public boolean hasPathTo(int v) {
        return distTo[v] != Double.POSITIVE_INFINITY;
    }

    public Iterable<DiEdge> pathTo(int v) {
        if (hasPathTo(v)) {
            LinkedList<DiEdge> path = new LinkedList<>();
            for (DiEdge edge = edgeTo[v]; edge != null; edge = edgeTo[edge.from()]) {
                path.push(edge);
            }
            return path;
        }
        return null;
    }

    public static void main(String[] args) {
        List<String> vertexInfo = List.of("v0", "v1", "v2", "v3", "v4", "v5", "v6", "v7");
        int[][] edges = {{4, 5}, {5, 4}, {4, 7}, {5, 7}, {7, 5}, {5, 1}, {0, 4}, {0, 2},
                {7, 3}, {1, 3}, {2, 7}, {6, 2}, {3, 6}, {6, 0}, {6, 4}};

        double[] weight = {0.35, 0.35, 0.37, 0.28, 0.28, 0.32, 0.38, 0.26, 0.39, 0.29,
                0.34, 0.40, 0.52, 0.58, 0.93};
        EdgeWeightedDiGraph<String> graph = new EdgeWeightedDiGraph<>(vertexInfo, edges, weight);
        BellmanFord[] all = new BellmanFord[graph.vertexNum()];

        for (int i = 0; i < all.length; i++) {
            all[i] = new BellmanFord(graph, i);
        }

        for (int s = 0; s < all.length; s++) {
            for (int i = 0; i < graph.vertexNum(); i++) {
                System.out.print(s + " to " + i + ": ");
                System.out.print("(" + all[s].distTo(i) + ") ");
                System.out.println(all[s].pathTo(i));
            }
            System.out.println();
        }
    }
}

在V轮放松完成后,如果没有负权回路存在,那么对于任何v -> w必然有distTo[v] + edge.weight() >= distTo[w],说明所有dist[w]已经是最短路径了;如果V轮后还存在distTo[v] + edge.weight() < distTo[w],说明distTo[w]无法收敛到最小值——陷入死循环了,我们围着那个环绕圈子,可以使得路径越来越短!这就是遇到了负权回路。

上面的例子没有负权回路存在,我们特意制造一个,看看结果。

public static void main(String[] args) {
    List<String> vertexInfo = List.of("v0", "v1", "v2", "v3");
    int[][] edges = {{0, 1}, {1, 2}, {2, 0}, {0, 3}, {2, 3}};

    double[] weight = {-9 , 5, 2, 4, 6};
    EdgeWeightedDiGraph<String> graph = new EdgeWeightedDiGraph<>(vertexInfo, edges, weight);
    BellmanFord bellmanFord = new BellmanFord(graph, 0);
    if (bellmanFord.hasNegativeCycle()) {
        System.out.println("路径中存在负权环!");
    }
}

0 -> 1-> 2 -> 0是一个负权回路。这里也注意下,如果图中有边的权值为负,在求最短的路径的时候要先判断有没有负权回路存在再进行后续计算。

SPFA算法--Bellman-Ford算法的优化

其实,根据经验我们很容易知道在任意一轮中许多边都不会放松成功。我们下次需要放松的顶点,只需是上次dist[w]值发生改变的那些w顶点。为此用一个队列保存这些顶点,用一个onPQ[]的布尔数组,来判断某个顶点是否已经在队列中。基于队列优化的Bellm-Ford算法又称为SPFA算法(Shortest Path Faster Algorithm)。

SPFA算法的思路是:每次放松一条边v -> w,如果放松成功(即distTo[w]的值被更新),且w没有在队列中则将其入列。然后队列的顶点出列并放松它,直到队列为空或者找到负权回路,算法终止。

这些数据结构可以保证:

  • 队列中不会出现重复的顶点;
  • 在某一轮中,改变了dist[w]和edge[w]的所有w将会在下一轮处理。

如果不存在从起点s可达的负权回路,那么算法终止的条件将是队列为空;如果存在呢?队列永远不会空(又在兜圈子了)!这意味着程序永不会结束,为此,我们必须判断从s可达的路径中是否存在负权回路。如果存在,应该立即停止算法,因为负权回路使得最短路径的研究毫无意义。而且此时经V轮放松后的edgeTo[]中必然会形成一个环,且权值和为负数。但很可能在全部V轮结束前就可以从edgeTo[]中找到负权回路,所以在放松边的过程中,可以隔若干轮就检查一下edgeTo[]中的路径是否成负权回路。

由于不是V轮结束后才检查是否存在负权回路,而是一边放松,一边检查,所以像上面那样用distTo[v] + edge.weight() < distTo[w]的方法来判断已经不适用了,因为放松尚未完成,上式成立很正常(说明需要更新最短路径了)。于是我们采用一种更通用的方法:先判断是否存在有向环,再判断该环的权值和是不是负数。

寻找有向负权环

判断有向环的实现并不复杂,核心思想其实是DFS(深度优先搜索)。

package Chap7;

import java.util.LinkedList;

public class NegativeDiCycle {
    private boolean[] marked;
    private DiEdge[] edgeTo;
    private boolean[] onStack;
    private LinkedList<DiEdge> cycle;

    public NegativeDiCycle(EdgeWeightedDiGraph<?> graph) {
        marked = new boolean[graph.vertexNum()];
        edgeTo = new DiEdge[graph.vertexNum()];
        onStack = new boolean[graph.vertexNum()];
        // 有向图可能不是强连通的,所以需要从每个顶点出发,寻找负权环
        for (int i = 0; i < graph.vertexNum(); i++) {
            dfs(graph, i);
        }
    }

    private void dfs(EdgeWeightedDiGraph<?> graph, int v) {
        // 模拟系统递归使用的栈,方法开始进栈;方法结束出栈
        onStack[v] = true;
        marked[v] = true;
        for (DiEdge edge : graph.adj(v)) {
            // 如果已经存在负权回路,终止递归方法
            if (this.hasNegativeDiCycle()) {
                return;
            }

            int w = edge.to();
            if (!marked[w]) {
                edgeTo[w] = edge;
                dfs(graph, w);
                // v -> w的路径,且w在栈中,说明形成有向环
            } else if (onStack[w]) {
                cycle = new LinkedList<>();

                DiEdge e = edgeTo[v];
                while (e.from() != w) {
                    cycle.push(e);
                    e = edgeTo[e.from()];
                }
                // 为避免空指针,离w最近的那条在循环外入栈
                cycle.push(e);
                // 把导致成环的边加入
                cycle.push(edge);
            }
        }
        onStack[v] = false;
    }

    public boolean hasNegativeDiCycle() {
        if (cycle != null) {
            double cycleWeight = cycle.stream().mapToDouble(DiEdge::weight).sum();
            if (cycleWeight < 0) {
                return true;
            }
        }
        return false;
    }

    public Iterable<DiEdge> cycle() {
        if (hasNegativeDiCycle()) {
            return cycle;
        }
        return null;
    }

}

使用DFS的原因主要是为了利用递归产生的由系统维护的栈(每次方法调用就相当于入栈,最先调用的最后才返回),而递归方法dfs的调用顺序正好反映了顶点的访问顺序,如先调用dfs(s), 接着dfs(w), 然后dfs(x),再递归调用dfs(v),那么这是一条s -> w -> x -> v的路径。我们使用了一个onStack[]布尔数组来模拟方法调用的进出栈情况——进入方法体说明方法被调用,进栈;方法执行完毕,该返回到上一层方法调用中了,出栈。onStack[]其实就是一条路径,onStack[v] = true说明顶点v位于从起点s可达的onStack[]这条路径中。

该实现最为关键的就是理解:当我们在v处发现某条v -> w的边,而恰好其w位于onStack[]中,就找到了一个环。我们知道onStack[]表示的是s -> w -> x -> v的路径,现在v -> w 刚好补全w -> x -> v成为环!如下图所示

好,寻找到有向环后,再判断环内所有边的权值是不是负数就好了。该实现不仅能判断,还能找出到底是哪些边造成了环。关键是以下几行

DiEdge e = edgeTo[v];
while (e.from() != w) {
    cycle.push(e);
    e = edgeTo[e.from()];
}
// 为避免空指针,离w最近的那条在循环外入栈
cycle.push(e);
// 把导致成环的边加入
cycle.push(edge);

对照着上图,最先x -> v入栈,到w -> x时候,发现e.from()就是w,不存入。出了while循环,将这条w -> x入栈,最后别忘了把导致成环的那条边入栈。有人可能会说了为何要这么麻烦,循环外单独push了两次,这是因为edgeTo[]中有的值是null(比如起点s),如果不在合适的地方终止循环,将在e = edgeTo[e.from()]该语句执行后,在e.from() != w处引起空指针异常!

SPFA算法的实现

可以判断负权回路的是否存在了,据此实现SPFA算法。

package Chap7;

import java.util.*;

public class SPFA {
    private double[] distTo;
    private DiEdge[] edgeTo;
    private Queue<Integer> queue;
    private boolean[] onPQ;  // 顶点是否在queue中
    private int cost; // 记录放松了边的次数
    private Iterable<DiEdge> cycle; // 找到的负权回路

    public boolean hasNegativeCycle() {
        return cycle != null;
    }

    private void findNegativeCycle() {

        EdgeWeightedDiGraph<String> g = new EdgeWeightedDiGraph<>(edgeTo.length);
        for (int v = 0; v < edgeTo.length; v++) {
            if (edgeTo[v] != null) {
                g.addDiEdge(edgeTo[v]);
            }
        }
        NegativeDiCycle cycleFinder = new NegativeDiCycle(g);
        if (cycleFinder.hasNegativeDiCycle()) {
            cycle = cycleFinder.cycle();
        }
    }

    private void relax(EdgeWeightedDiGraph<?> graph, int v) {
        for (DiEdge edge : graph.adj(v)) {
            int w = edge.to();
            if (distTo[v] + edge.weight() < distTo[w]) {
                distTo[w] = distTo[v] + edge.weight();
                edgeTo[w] = edge;
                if (!onPQ[w]) {
                    queue.offer(w);
                    onPQ[w] = true;
                }
            }
            // 每调放松一条边cost自增;每放松了graph.vertexNum条边,就检查是否有负权回路
            if (cost++ % graph.vertexNum() == 0) {
                findNegativeCycle();
            }
        }
    }

    public SPFA(EdgeWeightedDiGraph<?> graph, int s) {
        distTo = new double[graph.vertexNum()];
        edgeTo = new DiEdge[graph.vertexNum()];
        queue = new LinkedList<>();
        onPQ = new boolean[graph.vertexNum()];

        for (int i = 0; i < graph.vertexNum(); i++) {
            distTo[i] = Double.POSITIVE_INFINITY; // 1.0 / 0.0为INFINITY
        }

        distTo[s] = 0.0;
        // 以上都是初始化
        queue.offer(s);
        onPQ[s] = true;

        while (!queue.isEmpty() && !hasNegativeCycle()) {
            int v = queue.poll();
            onPQ[v] = false;
            relax(graph, v);
        }
    }

    public Iterable<DiEdge> cycle() {
        return cycle;
    }

    public double distTo(int v) {
        return distTo[v];
    }

    public boolean hasPathTo(int v) {
        return distTo[v] != Double.POSITIVE_INFINITY;
    }

    public Iterable<DiEdge> pathTo(int v) {
        if (hasPathTo(v)) {
            LinkedList<DiEdge> path = new LinkedList<>();
            for (DiEdge edge = edgeTo[v]; edge != null; edge = edgeTo[edge.from()]) {
                path.push(edge);
            }
            return path;
        }
        return null;
    }

    public static void main(String[] args) {
        List<String> vertexInfo = List.of("v0", "v1", "v2", "v3");
        int[][] edges = {{0, 1}, {1, 2}, {2, 0}, {0, 3}, {2, 3}};

        double[] weight = {-9 , 5, 2, 4, 6};
        EdgeWeightedDiGraph<String> graph = new EdgeWeightedDiGraph<>(vertexInfo, edges, weight);

        SPFA spfa = new SPFA(graph, 0);
        if (spfa.hasNegativeCycle()) {
            System.out.print("存在负权环:");
            System.out.println(spfa.cycle());
        }
    }
}

程序将输出找到的负权回路,打印[(2->0 2.0), (0->1 -9.0), (1->2 5.0)]。对于不存在负权回路的图,SPFA当然也能正确处理。这里就不测试了。

代码中特别注意一点,我们之前有提到需要隔若干次就检查是否存在负权回路,所以用到一个int型的cost变量记录放松边的次数,每放松了V条边就检查一次。因为可能在第V次放松之后,edgeTo[]数组中就存在负权回路了。findNegativeCycle方法就是将edgeTo[]转化成了有向图送给NegativeDiCycle类,检测是否存在负权回路。


by @sunhaiyu

2017.9.26

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

推荐阅读更多精彩内容