3
(defmethod carpet-append ((this carpet) (rect image-rectangle))
  (destructuring-bind (rect-width . rect-height)
      (rectangle-size rect)
    (destructuring-bind (bitmap-width . bitmap-height)
        (carpet-size this)
      (if this
          (iter:iter
            (iter:with min-area = (* (+ bitmap-width rect-width)
                                     (+ bitmap-height rect-height)))
            (iter:with min-pos = nil)
            (iter:for pos in (awailable-positions this))
            (iter:for test-area = (try-fit rect pos (carpet-bitmap this)))
            (when (and test-area (< test-area min-area))
              (setf min-pos pos))
            (iter:finally
             (let ((new-carpet
                    (make-carpet
                     :bitmap (make-array
                              (list (+ (car min-pos) rect-width)
                                    (+ (cdr min-pos) rect-height))
                              :element-type 'bit)
                     :rectangles (cons rect (carpet-rectangles this)))))
               (copy-bitmap-state this new-carpet)
               (setf (rectangle-position rect) min-pos)
               (place-image new-carpet rect)
               (return new-carpet))))
          (make-carpet
           :bitmap (make-array
                    (list rect-width rect-height)
                    :element-type 'bit)
           :rectangles (list rect))))))

image-rectangle并且carpet是结构。

当这个方法被这样调用时:

(carpet-append
 nil
 #s(image-rectangle
    :position (0 . 0)
    :size (48 . 76)
    :file "/home/wvxvw/projects/spritesheet/test-images/test-0.png"))

我得到一个:

#<SIMPLE-ERROR "~@<There is no applicable method for the generic function ~2I~_~S~
 ~I~_when called with arguments ~2I~_~S.~:>"

它是这样的吗?也许有一些方法可以让它接受nil作为一个论点?我将如何指定只有类型nilcarpet适用的参数?

4

1 回答 1

8

如果您有一个带有类的 arglistcarpetimage-rectangle,则参数最好是这些类或其子类。NIL当你的论点被声明为 class 时,你不能通过carpet

因此(if this没有意义。如果您通过了一个地毯对象,而您不能通过其他任何东西,那么测试this将始终为真。

如果要为NIL对象和矩形编写方法,则可以使用NULL类。

(defmethod carpet-append ((this null) (rect image-rectangle))
   ...)

由于 CLOS 没有 OR 或 AND 之类的类组合符,因此您必须为每种情况编写一个方法。

于 2013-01-04T17:15:29.073 回答