7

我刚刚开始使用 Clojure,我想我会尝试的第一件事是存储和检索结构列表,就像这里的 Suart Halloway 的示例一样。

如果我在属性字符串中使用没有空格的结构实例,则我的结构散列的吐出/啜饮效果很好,如下所示:

(struct customer "Apple" "InfiniteLoop")

但如果我使用这个:

(struct customer "Apple" "Infinite Loop 1")

我收到一个错误:

Exception in thread "main" clojure.lang.LispReader$ReaderException: java.lang.ArrayIndexOutOfBoundsException: 7 (test-storing.clj:19)
    at clojure.lang.Compiler$InvokeExpr.eval(Compiler.java:2719)
    at clojure.lang.Compiler$DefExpr.eval(Compiler.java:298)
    at clojure.lang.Compiler.eval(Compiler.java:4537)
    at clojure.lang.Compiler.load(Compiler.java:4857)
    at clojure.lang.Compiler.loadFile(Compiler.java:4824)
    at clojure.main$load_script__5833.invoke(main.clj:206)
    at clojure.main$init_opt__5836.invoke(main.clj:211)
    at clojure.main$initialize__5846.invoke(main.clj:239)
    at clojure.main$null_opt__5868.invoke(main.clj:264)
    at clojure.main$legacy_script__5883.invoke(main.clj:295)
    at clojure.lang.Var.invoke(Var.java:346)
    at clojure.main.legacy_script(main.java:34)
    at clojure.lang.Script.main(Script.java:20)
Caused by: clojure.lang.LispReader$ReaderException: java.lang.ArrayIndexOutOfBoundsException: 7
    at clojure.lang.LispReader.read(LispReader.java:180)
    at clojure.core$read__4168.invoke(core.clj:2083)
    at clojure.core$read__4168.invoke(core.clj:2081)
    at clojure.core$read__4168.invoke(core.clj:2079)
    at clojure.core$read__4168.invoke(core.clj:2077)
    at chap_03$load_db__54.invoke(chap_03.clj:71)
    at clojure.lang.AFn.applyToHelper(AFn.java:173)
    at clojure.lang.AFn.applyTo(AFn.java:164)
    at clojure.lang.Compiler$InvokeExpr.eval(Compiler.java:2714)
    ... 12 more
Caused by: java.lang.ArrayIndexOutOfBoundsException: 7
    at clojure.lang.PersistentArrayMap$Seq.first(PersistentArrayMap.java:216)
    at clojure.lang.APersistentMap.hashCode(APersistentMap.java:101)
    at clojure.lang.Util.hash(Util.java:55)
    at clojure.lang.PersistentHashMap.entryAt(PersistentHashMap.java:134)
    at clojure.lang.PersistentHashMap.containsKey(PersistentHashMap.java:130)
    at clojure.lang.APersistentSet.contains(APersistentSet.java:33)
    at clojure.lang.PersistentHashSet.cons(PersistentHashSet.java:59)
    at clojure.lang.PersistentHashSet.create(PersistentHashSet.java:34)
    at clojure.lang.LispReader$SetReader.invoke(LispReader.java:974)
    at clojure.lang.LispReader$DispatchReader.invoke(LispReader.java:540)
    at clojure.lang.LispReader.read(LispReader.java:145)
    ... 20 more

根据结构中字段的数量,我也可能只是将字符串的一部分作为属性名称而不是错误。例如:循环 1

我使用这样的存储功能:

(defn store-customer-db [customer-db filename]
  (spit filename (with-out-str (print customer-db))))

和这样的读取功能:

(defn load-db [filename]
  (with-in-str (slurp filename)(read)))

从 spit 的输出文件中,我可以看到 print 没有给字符串加上双引号,这似乎是 slurp 的问题。什么是正确的解决方案?

我的 Clojure 版本是 1.0,contrib 是几周前的快照。

4

2 回答 2

10

print并且println用于人类可读的输出。如果您想打印一些稍后会再次读取的内容,请使用prprn

user> (read-string (with-out-str (prn {"Apple" "Infinite Loop"})))
{"Apple" "Infinite Loop"}

然而:

user> (read-string (with-out-str (print {"Apple" "Infinite Loop"})))
java.lang.ArrayIndexOutOfBoundsException: 3 (NO_SOURCE_FILE:0)

它正在尝试运行此代码:

(read-string "{Apple Infinite Loop}")

它有奇数个键/值。请注意单个哈希键/值周围缺少引号。即使此读取有效(即,如果您碰巧提供了偶数个参数),它读取的也不会是一个充满字符串的哈希映射,而是符号。所以你会得到除了你输出的东西之外的东西。

user> (map class (keys (read-string (with-out-str (print {"foo bar" "baz quux"})))))
(clojure.lang.Symbol clojure.lang.Symbol)
于 2009-05-29T19:23:50.343 回答
2

对于,说:

(def hashed-hobbits {:bilbo "Takes after his Mother's family" :frodo "ring bearer"})

你只需要:

(spit "hobbitses.txt" hashed-hobbits)

并读回来:

(def there-and-back-again (read-string (slurp "hobbitses.txt")))

spit/slurp 将其全部包装在一个字符串中,但在 slurp 上使用 read-string 会将字符串解释回 clojure 代码/数据。也适用于巨魔数据结构!

于 2013-01-09T03:14:57.293 回答