6

现在我必须在排序之前将 hastable 复制到列表中:

(defun good-red ()
  (let ((tab (make-hash-table)) (res '()))
    (dotimes (i 33) (setf (gethash (+ i 1) tab) 0))
    (with-open-file (stream "test.txt")
        (loop for line = (read-line stream nil)
             until (null line)
             do
                (setq nums (butlast (str2lst (substring line 6))))
                (dolist (n nums) (incf (gethash n tab)))
                ))
    **(maphash #'(lambda (k v) (push (cons k v) res)) tab)**
    (setq sort-res (sort res #'< :key #'cdr))
    (reverse (nthcdr (- 33 18) (mapcar #'car sort-res))) ))

顺便说一句,获取列表的前 N ​​个元素的更好方法是什么?

4

2 回答 2

12

Vatine 的回答在技术上是正确的,但对于有人提出这个问题的直接问题可能不是很有帮助。使用哈希表保存计数器集合,然后按分数选择前 N 个项目的常见情况可以这样完成:

;; convert the hash table into an association list
(defun hash-table-alist (table)
  "Returns an association list containing the keys and values of hash table TABLE."
  (let ((alist nil))
    (maphash (lambda (k v)
               (push (cons k v) alist))
             table)
    alist))

(defun hash-table-top-n-values (table n)
  "Returns the top N entries from hash table TABLE. Values are expected to be numeric."
  (subseq (sort (hash-table-alist table) #'> :key #'cdr) 0 n))

第一个函数将哈希表的内容作为列表中的一系列cons对返回,该列表称为关联列表(键/值对的典型列表表示)。大多数 Lisp 爱好者手头已经有了这个功能的变体,因为它是一种非常常见的操作。这个版本来自Alexandria库,在 CL 社区中使用非常广泛。

第二个函数使用 SUBSEQ 从返回的列表中获取前 N 个项目,该列表通过使用每对的 CDR 作为键对第一个函数返回的 alist 进行排序来获取。将 :key 更改为 #'car 将按哈希键排序,将 #'> 更改为 #'< 将反转排序顺序。

于 2012-06-06T19:17:38.227 回答
3

哈希表本质上是无序的。如果要对其进行排序,则需要使用内容初始化某种有序的数据结构。

如果你想获取一个序列的前 N ​​个元素,总有 SUBSEQ。

于 2011-09-22T12:59:28.250 回答