Clojure基础
最近看了一段clojure,下面是从书上摘下来的一下语言基础的精华部分
;函数的基本形式
(defn average
[numbers]
(/ (apply + numbers) (count numbers))) (average [60 80 100 400])
(read-string "42")
(read-string "(+ 1 2)")
(pr-str [1 2 3])
(read-string "[1 2 3]")
"hello there"
(class \c)
;;创建一份hashmap包含两个键值对
(def person {:name "Sandra Cruz"
:city "Protland, ME"})
(:city person)
person
(:user/location person)
(def x 1)
x
;定义变量
(def x "hello")
*ns*
String
Integer
java.util.List
java.net.Socket
filter
x
;quote阻止求值,'x是其简便形式
(quote x)
'x
;fn是依次行定义,所以只能定义的同时使用,不能分开使用,可以再使用def定义别名这样就可以多次使用了
(def strange-adder (fn adder-self-reference
([x] (adder-self-reference x 1))
([x y] (+ x y))))
(strange-adder 10)
;defn=(def (fn ))
(defn adder-self-reference1
([x] (adder-self-reference1 x 1))
([x y] (+ x y)))
(adder-self-reference1 10)
(adder-self-reference1 10 50)
;可变参函数,x是一个固定参数,剩下其余的参数都被解构到rest中
(defn concat-rest
[x & rest]
(apply str (butlast rest))) (concat-rest 0 1 2 3 4)
;fn定义后立即需要传入参数进行运算
((fn [x y] (Math/pow x y)) 2 10)
;使用#定义匿名函数
(#(Math/pow %1 %2) 2 10)
;匿名函数不隐含do,需要手工指定do
(#(do (println (str %1 \^ %2))
(Math/pow %1 %2)) 2 10)
;条件判断,第一个表达式是true的话,整个if就是第二个表达式的值,否则为第三个。任何非nil、非false的值都位true
(if (< 2 1) 2 1) (defn countdown
[x]
(if (zero? x)
:blastoff!
(do (println x)
(recur (dec x)))))
(countdown 0)
(countdown 5)
;循环,recur能够在不消耗堆栈空间的情况下把程序执行转到离本地上下文最近的loop去
(loop [x 5];x=5
(if (neg? x) ;if x<0
x ;return x
(recur (dec x)))) ;esle loop(x-1)
;高阶函数map,将函数作用在集合上,返回一个序列
(map clojure.string/lower-case ["Java" "Imperative" "Weeping"
"Clojure" "Learning" "Peace"])
(map * [1 2 3 4] [5 6 7 8])
;高阶函数reduce 把集合应用在一个函数而产生单个值(归约)
(reduce max [0 -3 10 48])
(reduce + 50 [1 2 3 4]) (reduce
(fn [m v] (assoc m v (* v v)))
{}
[1 2 3 4])
;偏函数,把函数的一部分参数传给一个函数,创建一个新函数,这个函数所需参数就是剩下的那部分参数
;Clojure使用partial提供偏函数
(def only-strings (partial filter string?))
(only-strings ["a" "b" 1 2 3])
;使用函数组合重写上面的函数,clojure使用comp来实现函数组合
;comp接受任意数量的函数,comp接收的参数与最后一个函数的参数个数相等,一次从最后一个函数往前调用,前一个函数的返回值是后一个函数的参数,如果不能作为后一个函数的参数,则会报错
;例,返回给定数字列表的所有数字总和的负数字符串形式
(defn negated-sum-str
[& numbers]
(str (- (apply + numbers))))
(negated-sum-str 10 12 3.4)
;函数组合的形式
(def negated-sum-str (comp str - +))
(negated-sum-str 10 12 3.4)
;编写高阶函数
;编写一个高阶函数。它返回某个给定数字与它的参数的和
(defn adder
[n]
(fn [x] (+ n x)))
((adder 5) 18)
;编写一个高阶函数,它接受一个函数作为参数,同时返回一个函数,返回函数的作用是返回接受的函数的返回值的2倍
(defn doubler
[f]
(fn [& args]
(* 2 (apply f args))))
((doubler +) 1 2 3)
;可以使用别名简化
(def double+ (doubler +))
(double+ 1 2 3)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;日志系统;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;定义输入转接
(defn print-logger
[Writer];接收实现java.io.Writer接口的类实例
#(binding [*out* Writer];*out*默认绑定到标准输出,这里重新绑定到我们的Writer
(println %)));把要打印的消息,用println写入到*out*(已经替换成了Writer)
;标准输出
(def *out*-logger (print-logger *out*))
(*out*-logger "hello")
;内存buffer
(def writer (java.io.StringWriter.))
(def retained-logger (print-logger writer))
(retained-logger "hello")
;打印到文件
(require 'clojure.java.io);require加载命名空间,可以起别名(require 'clojure.java.io :as jio)
(defn file-logger
[file]
#(with-open [f (clojure.java.io/writer file :append true)];flie以追加模式打开,并把这个writer局部命名为f;with-open能够保证f在with-open结束的时候被关闭
((print-logger f) %))) (def log->file (file-logger "/home//breezeli/Document/messages.log"));不指定目录的话用light table就找不到存到哪个目录去了
(log->file "hello") ;记录到多处的日志串接,高阶函数
(defn mutli-logger
[& logger-fns];任意数量参数
#(doseq [f logger-fns] (f %)));dosep遍历 (def log (mutli-logger
(print-logger *out*)
(file-logger "/home//breezeli/Document/messages.log")))
(log "hello again") ;每条日志前面加时间戳
(defn timestamped-logger
[logger]
#(logger (format "[%1$tY-%1$tm-%1$te %1$tH:%1$tM:%1$tS] %2$s"(java.util.Date.) %))) (def log-timestamped (timestamped-logger
(mutli-logger
(print-logger *out*)
(file-logger "/home//breezeli/Document/messages.log"))))
(log-timestamped "goodbye,now")
;;;;;;;;;;;;;;;END;;;;;;;;;;;;;;;;;;;
;没有副作用且调用成本很高的纯函数可以内存化加快处理速度,clojure使用memoize实现内存化
(defn prime?
[n]
(cond
(== 1 n) false
(== 2 n) true
(even? n) false
:else (->> (range 3 (inc (Math/sqrt n)) 2)
(filter #(zero? (rem n %)))
empty?)));检测给定数字是不是素数
(time (prime? 1125899906842679))
(let [m-prime? (memoize prime?)]
(time (m-prime? 1125899906842679))
(time (m-prime? 1125899906842679)))
;map、vector、set和列表是Clojure提供的基本数据解构
'(a b :name 12.5) ;;列表
['a 'b :name 12.5] ;;vector
{:name "Chas" :age 31} ;;map
#{1 2 3} ;;set
;Clojure中所有的数据结构都实现了Collection抽象
;Collection的提供一下核心的集合函数
;conj添加一个元素到集合,保证对于所有的集合类型,都会高效地把元素添加到列表的第一个位置,因为如果添加在最后的话需要遍历列表,无法保证高效
;seq获取集合的顺序视图
;into建立在conj和seq之上
;count获取集合的元素个数
;empty获取一个跟所提供集合类型一样的空集合
;=判断两个或多个集合是否相等
;sequence系列,通常都被叫做“seq”,除了支持Collection提供的函数外还支持
;seq返回给传入参数的一个序列
(seq "Clojure")
;first、rest、next提供遍历序列的方法
(first "Clojure")
(rest "Clojure")
(next "Clojure") ;如果操作的结果是空,rest始终返回一个空序列,next返回nil,这是rest和next唯一的区别
;lazy-seq创建一个内容是一个表达式结果的惰性序列
Clojure基础的更多相关文章
- Clojure基础课程2-Clojure中的数据长啥样?
本文来自网易云社区 作者:李诺 " Clojure is elegant and pragmatic; it helps me focus more on solving business ...
- clojure基础入门(一)
最近在看storm的源码,就学习分享下clojure语法. 阅读目录: 概述 变量 运算符 流程控制 总结 概述 clojure是一种运行在JVM上的Lisp方言,属于函数式编程范式,它和java可以 ...
- clojure 环境搭建
以下是clojure 基础环境搭建的几种方式 mac brew install clojure linux curl -O https://download.clojure.org/install/l ...
- 跟我一起云计算(1)——storm
概述 最近要做一个实时分析的项目,所以需要深入一下storm. 为什么storm 综合下来,有以下几点: 1. 生逢其时 MapReduce 计算模型打开了分布式计算的另一扇大门,极大的降低了实现分布 ...
- == vs === in Javascript
本文来自网易云社区 作者:魏文庆 如果你只想知道==与===的区别,请直接看总结,当然我更希望您能耐心看完全文.Javascript中用于相等比较的操作符有两个==和===.==我们通常称为" ...
- 基于Impala平台打造交互查询系统
本文来自网易云社区 原创: 蒋鸿翔 DataFunTalk 本文根据网易大数据蒋鸿翔老师DataFun Talk--"大数据从底层处理到数据驱动业务"中分享的<基于Impal ...
- 巧用Scrum与Kanban
本文来自网易云社区 文\屈鹏飞 在互联网行业的项目管理实践中,敏捷和精益一直是大家所提倡的思想,其中Scrum和Kanban方法作为即敏捷又精益的典型代表,许多PM都在研究,笔者近期也在学习和实施Sc ...
- 【转】《从入门到精通云服务器》第六讲—OpenStack基础
前五期的<从入门到精通云服务器>受到了广泛好评,收到留言,有很多读者对云计算相关的技术非常感兴趣.应观众要求,我们这期要安利一条纯技术内容.准备好瓜子.花生,随小编一起进入OpenStac ...
- Clojure学习笔记(一)——介绍、安装和语法
什么是Clojure Clojure是一种动态的.强类型的.寄居在JVM上的语言. Clojure的特性: 函数式编程基础,包括一套性能可以和典型可变数据结构媲美的持久性数据结构 由JVM提供的成熟的 ...
随机推荐
- Spring Security – security none, filters none, access permitAll
1.概述 Spring Security提供了几种将请求模式配置为不安全或允许所有访问的机制.取决于这些机制中的哪一种 - 这可能意味着根本不在该路径上运行安全过滤器链,或者运行过滤器链并允许访问 2 ...
- CocoaPods 提交自己的库
今想把自己写的一个view提交到CocoaPods时候,突然发现pull request被拒了,原来从去年开始就改用trunk了... 网上那些folk 在提交的pull request的教程都不可用 ...
- Java中的if-else语句——通过示例学习Java编程(7)
作者:CHAITANYA SINGH 来源:https://www.koofun.com/pro/kfpostsdetail?kfpostsid=18 当我们需要根据一个条件执行一组语句时,我们需 ...
- kafka系列一:单节点伪分布式集群搭建
Kafka集群搭建分为单节点的伪分布式集群和多节点的分布式集群两种,首先来看一下单节点伪分布式集群安装.单节点伪分布式集群是指集群由一台ZooKeeper服务器和一台Kafka broker服务器组成 ...
- git命令收集
$ git clone ... $ git status 查看状态 $ git commit -am "XXX" 提交信息 $ git commit -am "XXXX& ...
- BZOJ1485: [HNOI2009]有趣的数列(Catalan数,质因数分解求组合数)
题意 挺简洁的. 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所有的奇数项满足a1<a3<…<a ...
- 从零开始利用vue-cli搭建简单音乐网站(五)
上一篇文章讲到的是如何利用mongoose从数据库读取数据然后更新页面,接下来要实现的就是用户注册登录功能,这个功能涉及到的东西太多了,今天只实现了登录功能,登陆之后更新导航条界面,最后效果如下: 登 ...
- 修改输入框placeholder的默认样式
一般网页中都用到input的placeholder属性,想让这个默认样式和网页保持一致,就需要重新设定样式,百度百度: :-moz-placeholder { / color: #000; opaci ...
- Android 滑动RecyclerView时隐藏部分控件
在使用RecyclerView控件时,上下拖动控件时的时候,需要实时的隐藏与显示部分控件,已到达很好的用户体验. 原理很简单,当RecyclerView拖动至最上层时显示控件,当RecyclerV ...
- [文章泛读] The varying faces of a program transformation systems (ACM Inroads, 2012)
Beevi S. Nadera, D. Chitraprasad, and Vinod S. S. Chandra. 2012. The varying faces of a program tran ...