最近自己陷入了很长时间的学习和思考之中,突然发现好久没有更新博文了,于是便想更新一篇。

  这篇文章是我之前程序设计语言课作业中一段代码,用scheme语言实现单源最段路算法。当时的我,花了一整天时间,学习了scheme并实现了SPFA算法,那天实现之后感觉很有成就感~在这里贴出来,以飨读者。


  突然发现博客园不支持scheme语言,于是只能放弃高亮了。不得不说,scheme代码有没有高亮差别好大……

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; 题目:单源最短路,输入数据给定
;;
;; 作者:卢奇
;; 学号:5130309680
;; 邮箱:icedream@sjtu.edu.cn
;;
;; 算法:SPFA(简化版)
;;
;; 代码结构:共三大部分——
;; 开始是一些语法糖,
;; 然后是SPFA算法的实现,
;; 最后是主体部分,调用了SPFA算法并输出结果。
;;
;; 备注:代码备注共有两种——
;; 1. 代码的三大部分,各自开头有一段备注
;; 2. 代码的两个主体部分,内部穿插了一些备注
;; 其中,两个主体部分是指:代码主体部分 以及 SPFA算法的主体部分(即SPFA函数)
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (begin ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; 这里是为后面代码定义的一些语法糖。
;;
;; 为了可读性,我做了一个“下标变换”:
;; 题目图中6个点,存储为0~5,但供操作的API对外设计成1~6的假象,简化思路
;;
;; 有一维数组、二维数组、队列和逻辑运算几方面,具体如下所示:
;; 1. 根据下标该值,构造新数组:change, change2
;; 2. 根据下标赋值(+下标变换):set, set2
;; 3. 根据下标取值(+下标变换):get, get2
;; 4. 入队、出队:push, pop
;; 5. 逻辑运算(二元与、二元或):and, or
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (define (change a i x)
(if (eqv? i 0)
(cons x (cdr a))
(cons (car a) (change (cdr a) (- i 1) x))))
(define (change2 a i j x)
(if (eqv? i 0)
(cons (change (car a) j x) (cdr a))
(cons (car a) (change2 (cdr a) (- i 1) j x))))
(define-syntax set
(syntax-rules ()
([set a i x] (set! a (change a (- i 1) x)))))
(define-syntax set2
(syntax-rules ()
([set2 a i j x]
(begin
(set! a (change2 a (- i 1) (- j 1) x))
(set! a (change2 a (- j 1) (- i 1) x))))))
(define-syntax get
(syntax-rules ()
([get a i] (list-ref a (- i 1)))))
(define-syntax get2
(syntax-rules ()
([get2 a i j] (list-ref (list-ref a (- i 1)) (- j 1))))) (define-syntax push
(syntax-rules ()
([push Q x] (set! Q (append Q (list x))))))
(define-syntax pop
(syntax-rules ()
([pop Q]
(let ([x (car Q)])
(set! Q (cdr Q))
x)))) (define-syntax and
(syntax-rules ()
([and Ea Eb]
(if (eqv? Ea #t)
(if (eqv? Eb #t) #t #f)
#f))))
(define-syntax or
(syntax-rules ()
([or Ea Eb]
(if (eqv? Ea #t)
#t
(if (eqv? Eb #t) #t #f))))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; 此为SPFA算法部分(简化版)
;;
;; 其中SPFA函数是主体,其调用了update-all函数,后者又调用了update函数。
;;
;; 注:之所以称之为简化版,是因为本来SPFA的入队应该去重的,但被我给省了。
;; 不过本题中并不要求速度、也不影响正确性,写不写也就无所谓了。
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (define (update map d Q u allv)
(cond
[(not (eqv? allv null))
(let ([v (car allv)] [lastv (cdr allv)])
; (newline)
; (display "an update --") (newline)
; (display "d: ") (display d) (newline)
; (display "Q: ") (display Q) (newline)
; (display "u: ") (display u) (newline)
; (display "v: ") (display v) (newline)
; (display "allv: ") (display allv) (newline)
(cond
[(and (not (eqv? (get2 map u v) #f)) (or (eqv? (get d v) #f) (< (+ (get d u) (get2 map u v)) (get d v))))
(begin
(set d v (+ (get d u) (get2 map u v)))
(push Q v))])
(update map d Q u lastv))]
[else (list d Q)])) (define (update-all map d Q)
(if (eqv? Q null)
d
(let ([u (pop Q)])
(define tmp (update map d Q u (list 1 2 3 4 5 6)))
(set! d (car tmp))
(set! Q (cadr tmp))
(update-all map d Q)))) (define (SPFA map s)
; 初始化SPFA中的数组
(define d (make-list 6 #f))
(set d s 0)
(define Q null)
(push Q s)
; 输出初始化的数组,仅供调试
(display "d: ") (display d) (newline)
(display "Q: ") (display Q) (newline)
; 计算由s出发的单源最短路,并返回计算出的结果
(update-all map d Q)) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; 题目主函数在此
;;
;; 本题所有的IO都在这里给出了,一目了然。
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; 建图 称为map
(define map (make-list 6 (make-list 6 #f)))
(set2 map 1 2 7)
(set2 map 1 3 9)
(set2 map 1 6 14)
(set2 map 2 4 15)
(set2 map 2 3 10)
(set2 map 3 4 11)
(set2 map 3 6 2)
(set2 map 4 5 6)
(set2 map 5 6 9)
; 通过简化的SPFA算法计算最短路
(define d (SPFA map 1))
; 输出答案
(display "last-d: ") (display d) (newline)
(display "result: ") (display (get d 5)) (newline))

用scheme语言实现SPFA算法(单源最短路)的更多相关文章

  1. Dijkstra算法——单源最短路算法

    一.介绍 迪杰斯特拉(Dijkstra)算法是典型最短路径算法,用于计算一个节点到其他各个节点的最短路径. 它的主要特点是以起始点为中心向外层层扩展(广度优先搜索思想),直到扩展到终点为止. 适用于有 ...

  2. 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)

    关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...

  3. 模板C++ 03图论算法 1最短路之单源最短路(SPFA)

    3.1最短路之单源最短路(SPFA) 松弛:常听人说松弛,一直不懂,后来明白其实就是更新某点到源点最短距离. 邻接表:表示与一个点联通的所有路. 如果从一个点沿着某条路径出发,又回到了自己,而且所经过 ...

  4. 2018/1/28 每日一学 单源最短路的SPFA算法以及其他三大最短路算法比较总结

    刚刚AC的pj普及组第四题就是一种单源最短路. 我们知道当一个图存在负权边时像Dijkstra等算法便无法实现: 而Bellman-Ford算法的复杂度又过高O(V*E),SPFA算法便派上用场了. ...

  5. spfa 单源最短路究极算法

    学习博客链接:SPFA 求单源最短路的SPFA算法的全称是:Shortest Path Faster Algorithm.     SPFA算法是西南交通大学段凡丁于1994年发表的.    从名字我 ...

  6. [ACM_图论] Domino Effect (POJ1135 Dijkstra算法 SSSP 单源最短路算法 中等 模板)

    Description Did you know that you can use domino bones for other things besides playing Dominoes? Ta ...

  7. 【算法】单源最短路——Dijkstra

    对于固定起点的最短路算法,我们称之为单源最短路算法.单源最短路算法很多,最常见的就是dijkstra算法. dijkstra主要用的是一种贪心的思想,就是说如果i...s...t...j是最短路,那么 ...

  8. 【算法系列学习】Dijkstra单源最短路 [kuangbin带你飞]专题四 最短路练习 A - Til the Cows Come Home

    https://vjudge.net/contest/66569#problem/A http://blog.csdn.net/wangjian8006/article/details/7871889 ...

  9. 单源最短路_SPFA_C++

    当我们需要求一个点到其它所有点的最短路时,我们可以采用SPFA算法 代码特别好写,而且可以有环,但是不能有负权环,时间复杂度是O(α(n)n),n为边数,α(n)为n的反阿克曼函数,一般小于等于4 模 ...

随机推荐

  1. Effective Java 34 Emulate extensible enums with interfaces

    Advantage Disadvantage Enum types Clarity Safety Ease of maintenance. None extensibility Typesafe en ...

  2. Effective Java 39 Make defensive copies when needed

    Principle It is essential to make a defensive copy of each mutable parameter to the constructor. Def ...

  3. spring+mybatis多数据源切换

    在实际的公司项目中,很可能会遇到一个问题就是,一个java项目,但是项目中涉及两个数据库,这两个数据库还在不同IP的机子上. 遇到这种情况的时候,我们有两个选择 1.不走spring的aop方式,直接 ...

  4. BIG biang教你误删oracle 怎么办,

    今天无意中在网上看到了关于oracle误删除数据恢复的一条信息,发现的确很好使,下面就我的测试向大家汇报下. 1. select * from t_viradsl2 t         //查询t_v ...

  5. hibernate一对多映射实现

    Junit4方法详解 setUpBeforeClass()类初始化前调用 tearDownAfterClass()类初始化后调用 setUp()在测试方法前调用 tearDown()在测试方法后调用 ...

  6. 移动语义 && 函数调用过程中的 lvalue

    当以一个函数内的临时变量对象作为另一个函数的形参的时候,原函数内的临时对象即 rvalue,就会成为此函数内的 lvalue. 这样会重新导致效率低下,因为造成了大量复制操作. <utility ...

  7. JavaScript 题目破解过程与解析

    题目来源 https://www.hackthissite.org/missions/javascript/ HackThisSite JavaScript mission 1-7 1 我先尝试输入  ...

  8. LessonFifth Redis的持久化功能

    #验证redis的快照和AOF功能 1.先验证RDB快照功能,由于AOF优先级高,先关闭,然后测试,截图如下                 2.设置打开AOF 然后进行实验,截图如下:       ...

  9. [转]Cordova + Ionic in Visual Studio - 101 Tutorial [Part I]

    本文转自:http://binarylies.ghost.io/cordova-ionic-in-visual-studio/ Hi everyone, I thought about lending ...

  10. NOIP2014提高组 DAY1 -SilverN

    T1  生活大爆炸版石头剪刀布 题目描述 石头剪刀布是常见的猜拳游戏:石头胜剪刀,剪刀胜布,布胜石头.如果两个人出拳一样,则不分胜负.在<生活大爆炸>第二季第8 集中出现了一种石头剪刀布的 ...