首先声明一点,这篇短文不是要挑起语言之间的关于孰优孰劣的论战,只是希望通过一个小需求,让大家能够对比体会一下函数式编程和面向对象编程的差异(包括理念和语法上的)。
Clojure 是什么?
Clojure 是运行在 Java 虚拟机(JVM)上的一种 Lisp 方言,她比 Common Lisp 更强调纯函数式编程,同时拥有复杂的「宏」。具体可以看 CSDN 上的一篇介绍文章 现实世界的LISP:Clojure语言初探 和知乎上的讨论 请评价一下Clojure语言的设计
要解决一个什么问题
我们在开发过程中经常会碰到这个需求,要对一个字符串进行哈希,然后当成 key 存入 redis。我们准备使用 Java 的 SHA-512 哈希算子来做第一步运算,并且把结果再做一次 Base64 编码转换。
好的,那我们接下来就看看在两种语言中如何做到这一点吧。
Clojure 的解法
程序猿的世界,代码说明一切,我们直接上代码吧,算法的入口是一个名为 hash-name 函数:
(defonce ^:private hash-key "universal_redis_hash_key")
(defn hash-name [k]
(.substring ^String (:password (digest hash-key k)) 0 4))
(defn digest
([salt passwd]
(digest "SHA-512" 512 salt passwd))
([hashalg iterations salt passwd]
(let [jhash (MessageDigest/getInstance hashalg)
new-pass (b64-encode (digester jhash salt passwd iterations))]
{:salt salt :password new-pass})))
(defn- digester [^MessageDigest hasher ^String salt ^String pw-clear iter]
{:tag String}
(letfn [(hashme [hv]
(letfn [(oneround [hv]
(do (.reset hasher)
(.digest hasher hv)))]
(nth (iterate oneround hv) iter)))]
(.reset hasher)
(.update ^MessageDigest hasher (.getBytes salt))
(.update ^MessageDigest hasher (.getBytes pw-clear))
(hashme (.digest hasher))))
(defn b64-encode [^bytes b]
{:tag String}
(Base64/encodeBase64String b))
不知道能完全看懂上面代码的人有多少。要是我们用Java写出来的话,相信很多人会拍着脑袋:原来如此!
Java 的解法
还是一样,直接上代码,这里只有一个函数 hashName:
private final static String hashKey = “universal_redis_hash_key";
public static String hashName(String salt, String pw, int iter) {
try {
MessageDigest digest = MessageDigest.getInstance("SHA-512");
digest.reset();
digest.update(salt.getBytes());
digest.update(pw.getBytes());
byte[] temp = digest.digest();
for (int i = 0; i < iter; i++) {
digest.reset();
temp = digest.digest(temp);
}
String result = Base64.encodeBase64String(temp);
return result.substring(0, 4);
} catch (Exception ex) {
ex.printStackTrace();
}
return "";
}
结论
对这两门语言的看法,或许有人会从通俗易懂上进行比较,有人会从优雅严谨性上进行批判,我没有任何倾向性,这就是一个仁者见仁智者见智的问题。
我只想补充说一下这种对比的由来。我司的程序员中 Clojure 粉较多,很多程序都是用 Clojure 写的。最近要跟其他公司的小伙伴合作搭一个服务,两边都需要采用同样的关键字哈希算法。我找到一块我们的实现片段,发了过去。但是对方不懂 Clojure,他们使用的是 Ruby;我懂 Clojure 但是不懂 Ruby,两个人都快没法交流了。最后我们一商量,大家都能懂 Java,所以我先把 Clojure 代码翻译成 Java 代码,然后他再转换成 Ruby 代码,这样两边的系统就对接上了。程序猿的语言世界真的好复杂!