从一小段代码看 Clojure 和 Java 解决问题的差异

来源:互联网 发布:网达软件千股千评 编辑:程序博客网 时间:2024/05/14 02:29

首先声明一点,这篇短文不是要挑起语言之间的关于孰优孰劣的论战,只是希望通过一个小需求,让大家能够对比体会一下函数式编程和面向对象编程的差异(包括理念和语法上的)。


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 代码,这样两边的系统就对接上了。程序猿的语言世界真的好复杂!

0 0
原创粉丝点击