2024 年 Clojure 调查问卷中分享您的想法!

欢迎!有关如何使用本网页的更多信息,请参阅关于页面。

0
ClojureScript
重新标记

specify! 隐式依赖提供的对象可扩展,以便提供协议实现。以下是一个来自 REPL 的快速演示。

cljs.user> (def obj #js {:current nil})
[#js {:current nil}]
cljs.user> (js/Object.preventExtensions obj)
{"current" nil}
cljs.user> (specify! obj
              IDeref
              (-deref [^js this] (.-current this)))
{"current" nil}

请注意,reify 也没有对这个要求进行文档说明。这导致了一些令人困惑的情况,例如由 JavaScript 库(例如 React)返回的非可扩展对象无法扩展,用户对此几乎一无所知。

类似地,当对象不可扩展时,set! 会静默失败。

cljs.user> (set! (.-foo obj) "bar")
"bar"
cljs.user> obj
{"current" nil}

goog.object/set 能够检测到这一点并抛出错误。

cljs.user> (goog.object/set obj "foo" "bar")    
Execution error (TypeError) at (<cljs repl>:1).
Cannot add property foo, object is not extensible
:repl/exception!
经过进一步研究,我发现 `goog.object/set` 能够检测到这个错误,因为它的定义默认启用了严格模式。不幸的是,`cljs.compiler` 中 `set!` 的定义没有提供一种简单的方法来使赋值在严格模式下运行,并且启用它有很大可能会破坏现有的代码,因为严格模式会使许多操作变为错误。

1 个答案

0
...