Код фактора Clojure, устанавливающий множество различных полей в объекте Java

Как я учитываю код, устанавливающий много различных полей в объекте Java? Я хотел бы учесть

(set! (. employee name) "Chris")
(set! (. employee age) 100)
(set! (. employee salary) 5000)

в

(doseq [field '((name "Chris") (age 100) (salary 5000))]
  (set! (. employee (first field)) (second field)))

Однако это не сработает, потому что точка является макросом и пытается оценить (первое поле) буквально. Кстати, я понимаю, что установка полей не является хорошей практикой. Мне нужно взаимодействовать с устаревшим кодом.

2 ответа

Решение

Попробуйте использовать сеттеры - но если у вас нет выбора и вам действительно нужно это учесть, макрос подойдет:

(defmacro set-all [object & fields-and-values]
  (let [obj-sym (gensym)] 
    `(let [~obj-sym ~object]
        ~@(for [[field value] (partition 2 fields-and-values)]
           `(set! (. ~obj-sym ~field)
                  ~value)))))

Генсим нужен в том случае, если выражение, дающее объект, имеет какие-либо побочные эффекты.

Пример использования:

user> (def p (java.awt.Point.))
#'user/p
user> (set-all p x 4 y 5)
5
user> p
#<Point java.awt.Point[x=4,y=5]>

Вы можете попробовать:

(defmacro set-all! [obj m]
    `(do ~@(map (fn [e] `(set! (. ~obj ~(key e)) ~(val e))) m) ~obj))

Пример:

(def a (java.awt.Point.))
(set-all! a {x 300 y 100})

MACROEXPAND-всего:

(do (set! (. a x) 9)
    (set! (. a y) 7)
    a)
Другие вопросы по тегам