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 投票
...