123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135 |
- ;;; Binary heap (borrowed from Chickadee)
- ;;; Copyright (C) 2017 David Thompson <dthompson2@worcester.edu>
- ;;;
- ;;; Licensed under the Apache License, Version 2.0 (the "License");
- ;;; you may not use this file except in compliance with the License.
- ;;; You may obtain a copy of the License at
- ;;;
- ;;; http://www.apache.org/licenses/LICENSE-2.0
- ;;;
- ;;; Unless required by applicable law or agreed to in writing, software
- ;;; distributed under the License is distributed on an "AS IS" BASIS,
- ;;; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- ;;; See the License for the specific language governing permissions and
- ;;; limitations under the License.
- ;;; Commentary:
- ;;;
- ;;; Binary min-heap for use as a priority queue.
- ;;;
- ;;; Code:
- (define-module (hoot binary-heap)
- #:use-module (ice-9 format)
- #:use-module (rnrs base)
- #:use-module (srfi srfi-9)
- #:use-module (srfi srfi-9 gnu)
- #:export (make-heap
- heap?
- heap-empty?
- heap-size
- heap-min
- heap-insert!
- heap-remove!
- heap-clear!))
- (define-record-type <heap>
- (%make-heap vector size <)
- heap?
- (vector heap-vector set-heap-vector!)
- (size heap-size set-heap-size!)
- (< heap-<))
- (define (display-heap heap port)
- (format port "#<heap size: ~d>" (heap-size heap)))
- (set-record-type-printer! <heap> display-heap)
- (define* (make-heap #:optional (< <))
- "Return a new heap that uses the @var{<} procedure to determine order."
- (%make-heap (make-vector 32 #f) 0 <))
- (define (heap-empty? heap)
- "Return @code{#t} if @var{heap} is empty."
- (zero? (heap-size heap)))
- (define (heap-capacity heap)
- (1- (vector-length (heap-vector heap))))
- (define (heap-full? heap)
- (= (heap-size heap) (heap-capacity heap)))
- (define (double-heap-capacity! heap)
- (let* ((old-vec (heap-vector heap))
- (new-vec (make-vector (* (vector-length old-vec) 2) #f)))
- (vector-copy! new-vec 0 old-vec)
- (set-heap-vector! heap new-vec)))
- (define (heap-min heap)
- "Return the minimum element of @var{heap}."
- (if (zero? (heap-size heap))
- (error "empty heap" heap)
- (vector-ref (heap-vector heap) 1)))
- (define (heap-set! heap i item)
- (vector-set! (heap-vector heap) i item))
- (define (heap-ref heap i)
- (vector-ref (heap-vector heap) i))
- (define (heap-insert! heap item)
- "Add @var{item} to @var{heap}."
- (when (heap-full? heap)
- (double-heap-capacity! heap))
- (let ((hole (1+ (heap-size heap)))
- (< (heap-< heap)))
- (set-heap-size! heap hole)
- (let loop ((hole hole))
- (let* ((parent-hole (div hole 2))
- (parent-item (heap-ref heap parent-hole)))
- (if (and (> hole 1) (< item parent-item))
- (begin
- (heap-set! heap hole parent-item)
- (loop parent-hole))
- (heap-set! heap hole item))))))
- (define (heap-remove! heap)
- "Remove the minimum element of @var{heap}."
- (let ((size (1- (heap-size heap)))
- (< (heap-< heap)))
- (define (finish hole)
- (heap-set! heap (heap-size heap) #f)
- (heap-set! heap 0 #f)
- (set-heap-size! heap size)
- *unspecified*)
- (define (leaf? hole)
- (> (* hole 2) size))
- (define (smallest-child hole)
- (let ((left-child (* hole 2))
- (right-child (1+ (* hole 2))))
- (if (or (= left-child size)
- (< (heap-ref heap left-child) (heap-ref heap right-child)))
- left-child
- right-child)))
- (heap-set! heap 1 (heap-ref heap (heap-size heap)))
- (let loop ((hole 1))
- (if (leaf? hole)
- (finish hole)
- (let ((child (smallest-child hole)))
- (if (< (heap-ref heap hole) (heap-ref heap child))
- (finish hole)
- (begin
- (heap-set! heap 0 (heap-ref heap hole))
- (heap-set! heap hole (heap-ref heap child))
- (heap-set! heap child (heap-ref heap 0))
- (loop child))))))))
- (define (heap-clear! heap)
- "Remove all elements from @var{heap}."
- (vector-fill! (heap-vector heap) #f)
- (set-heap-size! heap 0))
|