123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- ;;; queue.el --- Queue data structure -*- lexical-binding: t; -*-
- ;; Copyright (C) 1991-1995, 2008-2009, 2012 Free Software Foundation, Inc
- ;; Author: Inge Wallin <inge@lysator.liu.se>
- ;; Toby Cubitt <toby-predictive@dr-qubit.org>
- ;; Maintainer: Toby Cubitt <toby-predictive@dr-qubit.org>
- ;; Version: 0.1
- ;; Keywords: extensions, data structures, queue
- ;; URL: http://www.dr-qubit.org/emacs.php
- ;; Repository: http://www.dr-qubit.org/git/predictive.git
- ;; This file is part of Emacs.
- ;;
- ;; GNU Emacs is free software: you can redistribute it and/or modify it under
- ;; the terms of the GNU General Public License as published by the Free
- ;; Software Foundation, either version 3 of the License, or (at your option)
- ;; any later version.
- ;;
- ;; GNU Emacs is distributed in the hope that it will be useful, but WITHOUT
- ;; ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
- ;; FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
- ;; more details.
- ;;
- ;; You should have received a copy of the GNU General Public License along
- ;; with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;;
- ;; These queues can be used both as a first-in last-out (FILO) and as a
- ;; first-in first-out (FIFO) stack, i.e. elements can be added to the front or
- ;; back of the queue, and can be removed from the front. (This type of data
- ;; structure is sometimes called an "output-restricted deque".)
- ;;
- ;; You create a queue using `make-queue', add an element to the end of the
- ;; queue using `queue-enqueue', and push an element onto the front of the
- ;; queue using `queue-prepend'. To remove the first element from a queue, use
- ;; `queue-dequeue'. A number of other queue convenience functions are also
- ;; provided, all starting with the prefix `queue-'. Functions with prefix
- ;; `queue--' are for internal use only, and should never be used outside this
- ;; package.
- ;;; Code:
- (eval-when-compile (require 'cl))
- (defstruct (queue
- ;; A tagged list is the pre-defstruct representation.
- ;; (:type list)
- :named
- (:constructor nil)
- (:constructor queue-create ())
- (:copier nil))
- head tail)
- ;;;###autoload
- (defalias 'make-queue 'queue-create
- "Create an empty queue data structure.")
- (defun queue-enqueue (queue element)
- "Append an ELEMENT to the end of the QUEUE."
- (if (queue-head queue)
- (setcdr (queue-tail queue)
- (setf (queue-tail queue) (cons element nil)))
- (setf (queue-head queue)
- (setf (queue-tail queue) (cons element nil)))))
- (defalias 'queue-append 'queue-enqueue)
- (defun queue-prepend (queue element)
- "Prepend an ELEMENT to the front of the QUEUE."
- (if (queue-head queue)
- (push element (queue-head queue))
- (setf (queue-head queue)
- (setf (queue-tail queue) (cons element nil)))))
- (defun queue-dequeue (queue)
- "Remove the first element of QUEUE and return it.
- Returns nil if the queue is empty."
- (unless (cdr (queue-head queue)) (setf (queue-tail queue) nil))
- (pop (queue-head queue)))
- (defmacro queue-empty (queue)
- "Return t if QUEUE is empty, otherwise return nil."
- (null (queue-head queue)))
- (defmacro queue-first (queue)
- "Return the first element of QUEUE or nil if it is empty,
- without removing it from the QUEUE."
- (car (queue-head queue)))
- (defun queue-nth (queue n)
- "Return the nth element of a queue, without removing it.
- If the length of the queue is less than N, return nil. The first
- element in the queue has index 0."
- (nth n (queue-head queue)))
- (defun queue-last (queue)
- "Return the last element of QUEUE, without removing it.
- Returns nil if the QUEUE is empty."
- (car (queue-tail queue)))
- (defun queue-all (queue)
- "Return a list of all elements of QUEUE or nil if it is empty.
- The oldest element in the queue is the first in the list."
- (queue-head queue))
- (defun queue-copy (queue)
- "Return a copy of QUEUE.
- The new queue contains the elements of QUEUE in the same
- order. The elements themselves are *not* copied."
- (let ((q (queue-create))
- (list (queue-head queue)))
- (when (queue-head queue)
- (setf (queue-head q) (cons (car (queue-head queue)) nil)
- (queue-tail q) (queue-head q))
- (while (setq list (cdr list))
- (setf (queue-tail q)
- (setcdr (queue-tail q) (cons (car list) nil)))))
- q))
- (defun queue-length (queue)
- "Return the number of elements in QUEUE."
- (length (queue-head queue)))
- (defun queue-clear (queue)
- "Remove all elements from QUEUE."
- (setf (queue-head queue) nil
- (queue-tail queue) nil))
- (provide 'queue)
- ;;; queue.el ends here
|