打开主菜单

Clojure/ˈklʒər/[13]Lisp编程语言Java平台上的现代、动态函数式方言。[14][15] 与其他Lisp一样,Clojure视代码为数据且拥有一套Lisp系统。[16] 其开发过程目前由社区驱动,[17] 其作者里奇·希基(Rich Hickey)则以终身仁慈独裁者(BDFL)的身份进行监督。[18]

Clojure
Clojure logo.svg
编程范型

多重编程范型:

設計者 里奇·希基(Rich Hickey)
发行时间 2007年,​12年前​(2007
穩定版本
1.10.1[8]
(2019年6月6日,​12天前​(2019-06-06
型態系統
系统平台
許可證 Eclipse公共许可证
文件扩展名
  • .clj
  • .cljs
  • .cljc
  • .edn
網站 clojure.org
啟發語言
影響語言

Clojure提倡不可变性(immutability)与持久数据结构(persistent data structures)并鼓励程序员显式地管理标识(identity)及其状态(state)。[19] 对利用不可变值(immutable values)及显式时间进展构造(explicit progression-of-time constructs)进行编程的专注旨在促进更加健壮的(尤其是多线程)程序的开发。[20][21] Clojure的类型系统是完全动态的,但人们近期也开始探索其基于渐进类型化(gradual typing)的实现。[22]

对Clojure的商业支持由Cognitect公司提供。[23] 每年都会在全球范围内举办年度Clojure会议,其中最著名的有Clojure/conj(美国东海岸)、[24] Clojure/West(美国西海岸)[25]和EuroClojure(欧洲)。[26]

目录

历史与开发过程编辑

 
里奇·希基,Clojure的创造者

里奇·希基(Rich Hickey)是Clojure语言的创造者。[14] 在Clojure之前,他开发过类似但基于.NET平台的项目——dotLisp。[27] 在那之前,他还有过三次旨在Lisp与Java之间提供互操作(interoperability)的尝试:“Common Lisp的Java外语接口”(Java foreign language interface for Common Lisp (jfli))、[28] “Lisp的外语对象接口”(Foreign Object Interface for Lisp (FOIL))[29] 以及“Lisp友好的Java Servlet接口”(Lisp-friendly interface to Java Servlets (Lisplets))。[30]

在公开发布之前,里奇·希基在Clojure的工作上花了大约两年半的时间。期间,在没有外部资金的情况下,他将其大部分时间都专门投入到了Clojure的工作上。在这段时间结束时,里奇·希基以电子邮件的形式向他在Common Lisp社区里的一些朋友宣布了该语言。

Clojure的开发过程由社区驱动[17]并且在“Clojure社区”网站上进行管理。[31] 该网站上有其规划文档及一款用于提交错误(bug)报告的问题跟踪器(issue tracker)。一般的开发讨论是在“Clojure Dev谷歌网上论坛”(Clojure Dev Google Group)上进行的。[32] 任何人都可以提交错误报告和想法,而贡献补丁(patches)则需要先签署“Clojure贡献者协议”(Clojure Contributor Agreement)。[33] JIRA错误报告(JIRA ticket)由一个筛选者(screener)团队进行处理并最终由里奇·希基批准更改。[34]

设计理念编辑

里奇·希基开发Clojure的原因是因为他想要一款适合函数式编程的现代Lisp。该语言既需要与已建立的Java平台共生又需要有适合并发性的设计。[20][21][35]

Clojure对待变化(change)的方式以标识(identity)的概念为特征。[36] 标识是指随着时间的推移而产生的一系列状态(state)。而状态则是指标识在某一特定时间点上的值(value)。需要强调的是,这里的值是不可变的(immutable)。由此引申,由于状态是不可变的值,任意数量的工作单位(workers)都可以在其上以并行(parallel)的方式实施操作。因此,并发性(concurrency)就成为一道管理状态间变化的问题[注意,这里的“变化”是指从一个状态到另外一个状态的跃迁(transition)而不是状态本身的变化(mutation)。]为此,Clojure提供了几个可变的(mutable)引用类型(reference type)。每个引用类型都有其明确定义的语义用于控制状态之间的跃迁。[19][36]

功能特点编辑

版本 发布日期 主要功能/改进
2007年10月16日[37] 首次公开发布
1.0 2009年5月4日[38] 首个稳定版[39]
1.1 2009年12月31日[38] 将来(future)[40]
1.2 2010年8月19日[38] 协议(protocol)[41]
1.3 2011年9月23日[38] 增强对原始类型(primitive type)的支持[42]
1.4 2012年4月15日[38] 读取器字面量(reader literal)
1.5 2013年3月1日[38] 归纳器(reducer)
1.6 2014年3月25[43] Java API、经过改进的哈希算法
1.7 2015年6月30日[44] 变换归纳器(transducer)、读取器条件表达式(reader conditionals)
1.8 2016年1月19日[45] 附加的字符串函数、直接连接(direct linking)、套接字服务器(socket server)
1.9 2017年12月8日[46] 集成spec、命令行工具
1.10 2018年12月17日[47] 经过改进的错误报告、Java兼容性
當前版本: 1.10.1 2019年6月6日[48] 解决Java性能回归问题并改进clojure.main的错误报告
格式:
舊版本
舊版本,仍被支援
当前版本
最新的预览版
未来版本

Clojure执行于Java平台之上,因此,与Java紧密集成并完全支持从Clojure调用Java代码。[49] 与此同时,也可以从Java调用Clojure代码。[50] Leiningen[51]是社区中普遍使用的项目自动化工具。Leiningen为Maven集成提供支持,处理项目包管理和依赖项。Leiningen的配置使用的则是Clojure语法。[51]

与其他大多数Lisp一样,Clojure的语法建立在S-表达式之上。S-表达式在被编译之前先由读取器(reader)解析为数据结构。[52] 除了列表(list)之外,Clojure的读取器还支持映射(map)、集合(set)及向量(vector)等的字面量(literal)语法。这些字面量随后会被直接编译成上述数据结构。[52] Clojure是Lisp-1且有一套与其它Lisp不兼容的数据结构,因此,Clojure不支持与Lisp的其它方言之间的代码级兼容性。[52][53]

作为一门Lisp方言,函数在Clojure中是一等公民。此外,Clojure还支持读取﹣求值﹣输出循环(REPL)以及一套宏系统。[54] Clojure的Lisp宏系统与Common Lisp的系统极为相似。唯一不同的是,Clojure的重音符[称为“语法引用”(syntax quote)]用名字空间(namespace)来限定符号(symbol)。这有助于防止意外的名字捕获(unintended name capture),因为Clojure禁止绑定(binding)到用名字空间限定的名字(namespace-qualified name)上。如果需要强制捕获宏扩展(capturing macro expansion,)那么就需要显示地完成该过程。Clojure不支持用户定义的读取器宏(reader macro,)但Clojure的读取器支持更具约束力的语法扩展(syntactic expansion)形式。[55] Clojure支持多重方法(multimethods。)[56] 对于类似接口的抽象,Clojure提供基于协议(protocol)[57]的多态性(polymorphism)以及基于记录(record)[58]的数据类型系统。 Clojure通过这些设计来提供高性能且动态的多态性以避免所谓的“表达式问题”("expression problem"。)

Clojure支持惰性序列,并鼓励不可变性(immutability)与持久数据结构(persistent data structure。)Clojure作为一门函数式编程语言将重点放在递归高阶函数上而不是基于副作用循环流程上。Clojure不支持自动尾调用优化,因为JVM还不支持该项优化,[59][60][61]但是,可以用recur关键字显式地进行该项优化。[62] 对于并行并发计算,Clojure提供软件事务内存[63] 响应式代理系统[64]及基于通道(channel)的并发编程。[65]

Clojure 1.7引入了读取器条件表达式(reader conditional)从而允许在同一名字空间中嵌入Clojure与ClojureScript代码。[44][66] 变换归纳器(transducer)的加入则提供了另一种组合变换的方法。变换归纳器可以使高阶函数(如,mapfold)更加抽象从而使之独立于其输入数据源。传统地说,这些函数一般被应用于序列(sequence)上,而变换归纳器允许这些函数被应用于通道(channel)上并让用户定义她们自己的变换归纳(transduction)模型。[67][68][69]

平台编辑

Clojure的主要平台是Java[15][49]但也存在其他目标平台上的实现。其中,最值得关注的是ClojureScript[70](可被编译成ECMAScript 3[71])和ClojureCLR[72].NET平台上的完整移植版,可与其生态系统互操作。)2013年对1,060名受访者进行的Clojure社区调查[73]发现,47%的受访者在使用Clojure的同时也使用ClojureScript。2014年,这一数字增长到了55%,[74]而到了2015年,则达到了66%(根据2,445名受访者)。[75] 人气较高的ClojureScript项目包括React实现,如Reagent[76]和Om。[77][78]

人气编辑

随着对函数式编程的兴趣的持续升温,Clojure也越来越多地受到Java平台上的软件开发人员的青睐。该语言也一度成为知名软件开发老将(如詹姆斯·高斯林[79]保罗·格雷厄姆[80]及Robert C. Martin(俗称“Uncle Bob”)[81]等)的首选或推荐语言。

在由Snyk和Java Magazine合作编写的“JVM生态系统报告2018”(据称是“Java开发人员有史以来规模最大的调查”)中,Clojure被评为用于“主要应用程序”的第二大人气编程语言(仅次于Java)。[82]

业内使用Clojure的公司有苹果公司[83][84]Atlassian[85]、Funding Circle[86]Netflix[87]、 Puppet[88]沃尔玛[89]及其他大型软件公司[90]美国国家航空航天局[91]等政府机构。Clojure也一度被用于创意计算,包括视觉艺术、音乐、游戏和诗歌。[92]

美国知名软件咨询公司ThoughtWorks在为其“技术雷达”[93]评估函数式编程语言时表达了他们对Clojure的青睐,称其为“Lisp在JVM上的简单及优雅实现”,并在2012年将其状态提升为“采用”(“ADOPT”)[94]

越来越多的非官方和/或实验性的其他平台实现也验证了该语言的人气:

  • las3r[95]:执行于ActionScript虚拟机(Adobe Flash Player平台)的Clojure子集
  • clojure-py[96]:Clojure的纯Python实现
  • rouge[97]:Clojure基于YARV的Ruby实现
  • CljPerl[98]:Clojure的Perl实现
  • Pixie:受Clojure启发并用RPython实现的Lisp方言
  • Ferret[99]:可被编译成运行于微控制器的自包含(self-contained)C++11
  • Joker[100]:用Go实现的解释器(interpreter)和linter
  • clojerl[101]:基于Erlang虚拟机的Clojure

开发工具编辑

Clojure的开发工具在近几年得到了显著的改善。以下是目前最具人气的集成开发环境(IDE)/编辑器及其Clojure插件。[102]这些工具的结合为Clojure编程提供了出色的支持。

集成开发环境/编辑器及其Clojure插件
集成开发环境/编辑器 Clojure插件
Emacs CIDER[103]
IntelliJ IDEA Clojure-Kit[104]或Cursive[105](提供免费的非商业许可证)
Vim fireplace.vim[106]
Visual Studio Code Calva[107]
Light Table[108] (不适用)

除了社区提供的开发工具之外,官方的命令行界面(CLI)工具也随着Clojure 1.9一起发布并可在GNU/Linux、macOS及Windows上使用。[109]

示例编辑

Hello, World!程序编辑

(println "Hello, World!")

REPL编程编辑

与其他Lisp一样,Clojure的标志性特征之一是基于REPL的交互式编程。[110]在以下示例中,“;;”表示一行注释的开始,而“;; =>”则表示输出:

;; 定义一个var
(def a 42)
;; => #'user/a

;; 调用名为`+`的函数
(+ a 8)
;; => 50

;; 调用名为`even?`的函数
(even? a)
;; => true

;; 定义一个函数以返回n除10之余
(defn foo [n] (rem n 10))
;; => #'user/foo

;; 调用该函数
(foo a)
;; => 2

;; 打印`rem`的文档字符串(docstring)
(doc rem)
;; =>
-------------------------
clojure.core/rem
([num div])
 remainder of dividing numerator by denominator.

;; 打印`rem`的源代码
(source rem)
;; =>
(defn rem
  "remainder of dividing numerator by denominator."
  {:added "1.0"
   :static true
   :inline (fn [x y] `(. clojure.lang.Numbers (remainder ~x ~y)))}
  [num div]
    (. clojure.lang.Numbers (remainder num div)))

运行时可用的名字编辑

与Clojure不同,其他语言的编译器会将程序中的名字编译掉使得它们在运行时不可用。而在Clojure中,可以用普通的数据结构对其运行时进行观察:

;; 定义一个var
(def a 42)
;; => #'user/a

;; 以映射(map)的形式获取在`user`名字空间中捕获的(interned)所有var
(ns-publics 'user)
;; => {a #'user/a}

;; 用`#'`(读取器宏)及其关联的、名字空间限定的符号`user/a`引用该var
#'user/a
;; => #'user/a

;; 解引用该var(获取其值)
(deref #'user/a)
;; => 42

;; 定义(并附加文档字符串)一个函数以返回n除10之余
(defn foo "返回`(rem n 10)`" [n] (rem n 10))
;; => #'user/foo

;; 获取var `#'user/foo`的元数据
(meta #'user/foo)
;; =>
{:arglists ([n]),
 :doc "返回`(rem n 10)`",
 :line 1,
 :column 1,
 :file "user.clj",
 :name foo,
 :ns #namespace[user]}

代码即数据(同像性)编辑

与其他Lisp类似,Clojure也具有同像性[又被称为“代码即数据”(code as data)]。从下面的示例中可以看到,用Clojure编写代码从而修改代码本身是非常容易的:

;; 调用一个函数 (代码)
(+ 1 1)
;; => 2

;; 引用该函数调用
;;(将代码转换成数据,此处为含一组符号的列表)
(quote (+ 1 1))
;; => (+ 1 1)

;; 获取该列表上的首个元素
;; (视代码为数据并对其进行操作)
(first (quote (+ 1 1)))
;; => +

;; 获取该列表上的最后一个元素
;; (视代码为数据并对其进行操作)
(last (quote (+ 1 1)))
;; => 1

;; 替换原列表上的符号从而获取一个新列表
;; (视代码为数据并对其进行操作)
(map (fn [form]
       (case form
         1 'one
         + 'plus))
     (quote (+ 1 1)))
;; => (plus one one)

富有表现力的数据变换操作符编辑

穿梭宏(threading macro,如,->->>等)可以在语法上表达一个数据集在一系列变换间穿梭的抽象:

(->> (range 10)
     (map inc)
     (filter even?))
;; => (2 4 6 8 10)

利用变换归纳器(transducer)也可以更有效地实现该过程:

(sequence (comp (map inc)
                (filter even?))
          (range 10))
;; => (2 4 6 8 10)

标识及其状态的线程安全管理编辑

线程安全的唯一序列号生成器(然而,和许多其他Lisp方言一样,Clojure内部使用其内置的gensym函数):

(def i (atom 0))

(defn generate-unique-id
  "每次调用会返回一个唯一数字ID。"
  []
  (swap! i inc))

编辑

java.io.Writer的一个匿名子类(不写任何内容)和一个宏(使用该类来静音其中打印的所有内容):

(def bit-bucket-writer
  (proxy [java.io.Writer] []
    (write [buf] nil)
    (close []    nil)
    (flush []    nil)))

(defmacro noprint
  "在对给定的`forms`求值的同时静音所有向`*out*`打印的内容。"
  [& forms]
  `(binding [*out* bit-bucket-writer]
     ~@forms))

(noprint
  (println "Hello, nobody!"))
;; => nil

Java互操作编辑

作为其主要设计目标之一,Clojure从一开始就将其宿主平台(host platforms)视为其不可分割的一部分。Clojure与Java之间出色的互操作即得益于此:

;; 调用一个实例方法
(.toUpperCase "apple")
;; => "APPLE"

;; 调用一个静态方法
(System/getProperty "java.vm.version")
;; => "12+33"

;; 创建`java.util.HashMap`的一个实例
;; 并加入一些键值对(key-value pairs)
(doto (java.util.HashMap.)
  (.put "apple" 1)
  (.put "banana" 2))
;; => {"banana" 2, "apple" 1}

;; 创建`java.util.ArrayList`的一个实例
;; 并用`clojure.core/map`递增(increment)其元素
(def al (doto (java.util.ArrayList.)
          (.add 1)
          (.add 2)
          (.add 3)))

(map inc al)
;; => (2 3 4)

;; 利用Java Swing显示一个消息对话框
(javax.swing.JOptionPane/showMessageDialog
  nil
  "Hello, World!")
;; => nil

软件事务内存编辑

10个线程操纵一个共享数据结构,该结构由100个向量组成,而每个向量包含10个(最初是连续的)唯一数字。每个线程随后在两个随机向量中重复选择两个随机位置并交换它们。通过使用Clojure的软件事务内存系统,对向量的所有更改都发生在事务中:

(defn run
  [nvecs nitems nthreads niters]
  (let [vec-refs
        (->> (* nvecs nitems)
             (range)
             (into [] (comp (partition-all nitems)
                            (map vec)
                            (map ref))))

        swap
        #(let [v1 (rand-int nvecs)
               v2 (rand-int nvecs)
               i1 (rand-int nitems)
               i2 (rand-int nitems)]
          (dosync
            (let [tmp (nth @(vec-refs v1) i1)]
              (alter (vec-refs v1) assoc i1 (nth @(vec-refs v2) i2))
              (alter (vec-refs v2) assoc i2 tmp))))

        report
        #(->> vec-refs
              (into [] (comp (map deref)
                             (map (fn [v] (prn v) v))
                             cat
                             (distinct)))
              (count)
              (println "Distinct:"))]

    (report)

    (->> #(dotimes [_ niters] (swap))
         (repeat nthreads)
         (apply pcalls)
         (dorun))

    (report)))

(run 100 10 10 100000)
;; =>
[0 1 2 3 4 5 6 7 8 9]
[10 11 12 13 14 15 16 17 18 19]
  ...
[990 991 992 993 994 995 996 997 998 999]
Distinct: 1000

[382 318 466 963 619 22 21 273 45 596]
[808 639 804 471 394 904 952 75 289 778]
  ...
[484 216 622 139 651 592 379 228 242 355]
Distinct: 1000
nil

参考文献编辑

  1. ^ Agents and Asynchronous Actions
  2. ^ Functional Programming
  3. ^ Concurrent Programming
  4. ^ core.async
  5. ^ core.logic
  6. ^ Macros
  7. ^ Threading Macros Guide
  8. ^ Clojure 1.10.1 release. clojure.org. 2019-06-06 [2019-06-06]. 
  9. ^ Fogus, Michael. Rich Hickey Q&A. CodeQuarterly.com. [2017-01-11]. 
  10. ^ Hickey, Rich. Clojure Bookshelf. Listmania!. Amazon.com. [2017-10-03]. 
  11. ^ Bonnaire-Sergeant, Ambrose. A Practical Optional Type System for Clojure (Thesis). The University of Western Australia. 2012. 
  12. ^ Clojure Programming (PDF). [2013-04-30]. 
  13. ^ meaning and pronunciation of Clojure. Rich Hickey. [2012-04-20]. 
  14. ^ 14.0 14.1 Krill, Paul. Clojure inventor Hickey now aims for Android. www.infoworld.com. [2018-11-08]. 
  15. ^ 15.0 15.1 Clojure – 首页. Clojure.org. [2018-10-31]. 
  16. ^ Clojure – Lisp. Clojure.org. [2015-09-15]. 
  17. ^ 17.0 17.1 Contributing FAQ - Clojure Community - Clojure Development. dev.clojure.org. [2015-09-15]. 
  18. ^ Clojure - funding. clojure.org. [2015-09-15]. 
  19. ^ 19.0 19.1 Clojure – state. Clojure.org. [2015-09-15]. 
  20. ^ 20.0 20.1 Hickey, Rich. Rationale. Clojure.org. [2008-10-17]. 
  21. ^ 21.0 21.1 Charles. Expert to Expert: Rich Hickey and Brian Beckman – Inside Clojure; Going Deep; Channel 9. Channel9.msdn.com. 2009-10-06 [2012-06-28]. 
  22. ^ clojure/core.typed. GitHub. [2015-09-15]. 
  23. ^ Clojure Programming Language: Cognitect, Clojure Consulting, Clojure Support, Functional Programming, JVM. Cognitect.com. [2015-09-15]. 
  24. ^ Clojure/conj
  25. ^ Clojure/West
  26. ^ EuroClojure
  27. ^ [ANN] dotLisp – a Lisp dialect for .NET. Google Groups. 
  28. ^ jfli, a Java foreign language interface for Common Lisp. 
  29. ^ FOIL – a Foreign Object Interface for Lisp. 
  30. ^ Lisplets – a Lisp-friendly interface to Java Servlets. 
  31. ^ Clojure Community website
  32. ^ Clojure Dev Google Group
  33. ^ Contributing FAQ – Clojure Community – Clojure Development. dev.clojure.org. [2015-09-16]. 
  34. ^ JIRA workflow – Clojure Community – Clojure Development. dev.clojure.org. [2015-09-16]. 
  35. ^ Economy Size Geek – Interview with Rich Hickey, Creator of Clojure. Linux Journal. [2015-09-15]. 
  36. ^ 36.0 36.1 Hickey, Rich. On State and Identity. Clojure.org. [2010-03-01]. 
  37. ^ Clojure: Clojure is Two!. Clojure Blog. [2015-09-16]. 
  38. ^ 38.0 38.1 38.2 38.3 38.4 38.5 Fingerhut, Andy. Clojure version history. jafingerhut.github.io. [2015-09-16]. 
  39. ^ Clojure: Clojure 1.0. Clojure Blog. [2015-09-16]. 
  40. ^ Clojure: Clojure 1.1 Release. clojure.blogspot.com. [2015-09-16]. 
  41. ^ Clojure – protocols. Clojure.org. [2015-09-16]. 
  42. ^ Clojure/clojure. GitHub. [2015-09-16]. 
  43. ^ Google Groups. groups.google.com. [2015-09-16]. 
  44. ^ 44.0 44.1 Clojure 1.9. Cognitect. [2017-12-08]. 
  45. ^ Google Groups. groups.google.com. [2016-01-25]. 
  46. ^ Google Groups. groups.google.com. [2017-12-08]. 
  47. ^ Clojure 1.10 release. clojure.org. [2018-12-17]. 
  48. ^ Clojure 1.10.1 release. clojure.org. [2019-06-06]. 
  49. ^ 49.0 49.1 Clojure – jvm_hosted. Clojure.org. [2015-09-15]. 
  50. ^ Clojure – java_interop. Clojure.org. [2015-09-15]. 
  51. ^ 51.0 51.1 Hagelberg, Phil; contributors. Leiningen. leiningen.org. [2015-09-15]. 
  52. ^ 52.0 52.1 52.2 Clojure – reader. Clojure.org. [2015-09-15]. 
  53. ^ Clojure – Lisps. Clojure.org. [2015-09-15]. 
  54. ^ Clojure – macros. Clojure.org. [2015-09-15]. 
  55. ^ Hickey, Rich. edn. GitHub. [2014-05-24]. 
  56. ^ Clojure – multimethods. Clojure.org. [2015-09-15]. 
  57. ^ Clojure – protocols. Clojure.org. [2015-09-15]. 
  58. ^ Clojure – datatypes. Clojure.org. [2015-09-15]. 
  59. ^ Goetz, Brian. Brian Goetz – Stewardship: the Sobering Parts. YouTube: ClojureTV. [2015-09-15]. 
  60. ^ Rose, John. tail calls in the VM. blogs.oracle.com. [2018-11-03]. 
  61. ^ Rose, John. Some languages need to be able to perform tail calls. JDK Bug System. [2018-11-03]. 
  62. ^ Clojure – special_forms. Clojure.org. [2015-09-15]. 
  63. ^ Clojure – Refs. Clojure.org. [2015-09-15]. 
  64. ^ Clojure – Agents. Clojure.org. [2016-07-04]. 
  65. ^ Clojure: Clojure core.async Channels. Clojure.com. [2015-09-15]. (原始内容存档于2013-07-02). 
  66. ^ Clojure – reader. Clojure.org. [2015-09-15]. 
  67. ^ "Transducers" by Rich Hickey. https://www.youtube.com/watch?v=6mTbuzafcII. Retrieved on 2015-09-15.
  68. ^ Transducers are Coming. [2015-09-15]. 
  69. ^ Rich Hickey – Inside Transducers. YouTube: Cognitect Inc. 2014-11-20 [2015-09-15]. 
  70. ^ Clojure/Clojurescript. GitHub. [2015-09-15]. 
  71. ^ ClojureScript – FAQ (for JavaScript developers). clojurescript.org. [2018-02-04]. 
  72. ^ clojure/clojure-clr. GitHub. [2012-06-28]. 
  73. ^ Emerick, Chas. Results of the 2013 State of Clojure & ClojureScript survey. Cemerick. [2015-09-17]. 
  74. ^ State of Clojure 2014 Survey Results. Cognitect Blog. [2015-09-17]. 
  75. ^ State of Clojure 2015 Survey Results. Cognitect Blog. [2016-09-08]. 
  76. ^ Reagent. GitHub. 
  77. ^ Om. GitHub. 
  78. ^ Om: Enhancing Facebook's React with Immutability. InfoQ. [2015-09-17]. 
  79. ^ James Gostling meetup with London Java Community. 2016-10-11 [2019-02-10]. 
  80. ^ Graham, Paul. Paul Graham on Twitter. 2016-05-06 [2019-02-10]. 
  81. ^ Martin, Robert. Unble Bob Martin on Twitter. 2018-11-29 [2019-02-10]. 
  82. ^ Maple, Simon; Binstock, Andrew. JVM Ecosystem Report 2018 - Synk. 2018-10-17 [2019-02-10]. 
  83. ^ Roman Liutikov at Twitter. Twitter. [2018-10-27]. 
  84. ^ Jobs at Apple. Apple Inc. [2018-10-27]. 
  85. ^ Borges, Leonardo. Realtime Collaboration with Clojure. 2015-07-07 [2019-02-10]. 
  86. ^ JUXT Blog: Clojure in London: Funding Circle. juxt.pro. [2017-02-01]. 
  87. ^ Williams, Alex. The New Stack Makers: Adrian Cockcroft on Sun, Netflix, Clojure, Go, Docker and More. 2014-08-03 [2019-02-10]. 
  88. ^ A New Era of Application Services at Puppet Labs. Puppet Labs. [2015-09-15]. 
  89. ^ Walmart Runs Clojure at Scale. Cognitect.com. [2015-09-15]. 
  90. ^ Clojure - Success Stories. Clojure.org. [2018-10-27]. 
  91. ^ nasa/Common-Metadata-Repository. GitHub. 
  92. ^ Meier, Carin. Creative computing with Clojure. O'Reilly Radar. [2015-09-17]. 
  93. ^ Frequently Asked Questions - Technology Radar - ThoughtWorks. [2019-02-10]. 
  94. ^ Clojure - Technology Radar - ThoughtWorks. [2019-02-10]. 
  95. ^ aemoncannon. Home: aemoncannon/las3r Wiki. GitHub. 2010-12-30 [2012-06-28]. 
  96. ^ drewr/clojure-py. GitHub. [2017-01-10]. 
  97. ^ rouge-lang/rouge. GitHub. [2015-12-19]. 
  98. ^ A Lisp on Perl. MetaCPAN. [2014-05-25]. 
  99. ^ Ferret. GitHub. 
  100. ^ Joker. GitHub. 
  101. ^ clojerl. GitHub. 
  102. ^ Miller, Alex. "State of Clojure 2018" Results. clojure.org. [2018-12-09]. 
  103. ^ CIDER: The Clojure Interactive Development Environment that Rocks. 
  104. ^ Clojure-Kit: Clojure and ClojureScript plugin for IntelliJ-based tools. 
  105. ^ Cursive: Provides full Clojure and ClojureScript language support. 
  106. ^ fireplace.vim: Clojure REPL Support. 
  107. ^ Calva: Clojure(Script) Interactive Programming. 
  108. ^ Light Table: The Next Generation Code Editor. 
  109. ^ Clojure - Deps and CLI Guide
  110. ^ Programming at the REPL: Introduction. clojure.org. [2018-12-04]. 

延伸阅读编辑

外部链接编辑