动态类型语言,少了静态类型语言必须声明变量类型的累赘,但也缺失了编译时类型检查和编译时优化的好处。cljs虽然做为动态类型语言,但其提供Metadata让咱们在必要的时候可选择地补充类型提示,以便提升代码可读性和供编译器优化使用。除了上述之外,Metadata还让咱们在不影响对象本质的前提下,附加额外信息加强元编程能力。html
首先要明确一点的是,Metadata不是任何对象/值都拥有的。只有以下的对象才可附加Metadata编程
Symbol函数
Var单元测试
Collections(List,Map,Vector,Set)测试
Record优化
Typecode
meta
读取Metadata 经过meta
咱们能够获取对象的Metadata,若没有则返回nil
。
示例1:获取Var的Metadatahtm
(def a 1) (meta #'a) ;;=> {:ns cljs.user, :name a, :file "<cljs repl>", :end-column 7, :source "a", :column 1, :line 1, :end-line 1, :arglists (), :doc nil, :test nil}
上述示例1中是(meta #'a)
而不是(meta a)
,前者是获取Var的Metadata,然后者是获取值1的Metadata,显而后者是没有Metadata的。
示例2:获取Symbol的Metadata对象
(def a (with-meta 'a {:something "test"})) (meta a) ;;=> {:something "test"}
经过with-meta
咱们能够获取附加了metadata的symbol'a
(注意做为入参的symbol'a
不会受到影响)。blog
with-meta
后期追加Metadata 上面咱们已经看到with-meta
的使用示例了,下面咱们再看看具体的函数签名吧。
;; Returns an object of the same type and value as obj, with map m as its metadata. (with-meta obj m)
值得注意的是,with-meta
会的返回值才会附加上metadata,而入参obj不会附加上metadata。所以须要用绑定来保存结果,以便后续使用。
(def a (with-meta obj m))
除了with-meta
后期追加外,不少时候咱们是在定义时就已经能够明确metadata的了,那么能够两种形式定义metadata。
;; 定义Var的metadata (def ^{:dynamic true, :tag "test"} a 1) ;; 读取metadata (meta #'a) ;; 定义Map的metadata (def b ^{:something "test"} {:name 1}) ;; 读取metadata (meta b)
有时咱们只想定义一两个metadata,完整写法显然有些累赘,那么咱们就能够采用metadata reader的写法,小清新一下。
(def ^:dynamic ^"test" a 1) ;;等价于(def ^{:dynamic true, :tag "test"} a 1)
缩写是有限制,因此只能表达以下metadata
^:foo ;;=> ^{:foo true} ^"foo";;=> ^{:tag "foo"} ^foo ;;=> ^{:tag <value of foo>}
&esmp;至于其它metadata则仍是要使用完整写法处理。
:dynamic ;; Boolean, 指定Var为动态绑定 :private ;; Boolean, 指定该Symbol的访问控制为私有,默认为public :doc ;; String, 设置document string :test ;; Function,不带入参的函数,单元测试函数 :tag ;; Class,指定Symbol所指向的Var的数据类型
另外编译器会自动附加一下信息到Var上。
:file ;; String :line ;; Int :name ;; Symbol :ns ;; Symbol :macro ;; Boolean,true表示是macro :arglists ;; List<Vector>,每一个Vector表示一个函数签名
今天就写到这里,下次继续^_^
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohn... ^_^肥仔John