2

我在 Android 设备上运行了这个 Scala 代码:

// create Map
val myMap1 = new HashMap[Int, String]()
myMap1.put(1, "a")

// write it to file
val outStream = context.openFileOutput("test.txt", Context.MODE_PRIVATE)
val ostream = new ObjectOutputStream(outStream)
ostream.writeObject(myMap1)
ostream.close

// read from file
val inStream = context.openFileInput("test.txt")
val istream = new ObjectInputStream(inStream)
val myMap2 = (istream.readObject).asInstanceOf[HashMap[Int, String]]
istream.close

// java.lang.NullPointerException accessing myMap2
if (myMap2.contains(1)) { println("yes") } else { println("no") }

我创建一个 mutable.HashMap 并将其写入文件,读取它,然后 HashMap 为空。为什么是myMap2null 并且没有任何内容?下面是调试会话的屏幕截图。

Eclipse IDE 的屏幕截图 - 调试会话

完整的堆栈跟踪:

java.lang.NullPointerException
  at scala.collection.mutable.HashTable$class.index(HashTable.scala:353)
  at scala.collection.mutable.HashMap.index(HashMap.scala:39)
  at scala.collection.mutable.HashTable$class.findEntry(HashTable.scala:130)
  at scala.collection.mutable.HashMap.findEntry(HashMap.scala:39)
  at scala.collection.mutable.HashMap.contains(HashMap.scala:60)
  at com.test.mytest.bean.MyItem$.read(MyItem.scala:74)
  at com.test.mytest.bean.MyItem$.add(MyItem.scala:93)
  at com.test.mytest.frag.MyFragment.onClick(MyFragment.scala:114)
  at android.view.View.performClick(View.java:4084)
  at android.view.View$PerformClick.run(View.java:16966)
  at android.os.Handler.handleCallback(Handler.java:615)
  at android.os.Handler.dispatchMessage(Handler.java:92)
  at android.os.Looper.loop(Looper.java:137)
  at android.app.ActivityThread.main(ActivityThread.java:4745)
  at java.lang.reflect.Method.invokeNative(Native Method)
  at java.lang.reflect.Method.invoke(Method.java:511)
  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
  at dalvik.system.NativeStart.main(Native Method)
4

1 回答 1

3

原因是它HashTable包含一个table声明为“瞬态”的受保护字段:

@transient protected var table: Array[HashEntry[A, Entry]]

所以它永远不会写入文件,并且在反序列化时保持其默认null值。

但是,HashMap定义readObject并明确writeObject处理序列化(我没有花时间深入研究整个代码,但最肯定的是这应该处理table字段的读/写等),所以它似乎无论如何都应该工作。

现在,让我在这里做一个有根据的猜测:你在 Android 上,所以很肯定你正在使用像 proguard 这样的工具来删除所有不需要的代码(并通过重命名来缩小你确实需要的代码)。那么问题是readObjectandwriteObject是私有的,因此根据您的 proguard 配置,proguard 可能会假定这些方法永远不会被访问,从而完全删除它们。因此,在运行时,只使用标准序列化,并且应该处理的自定义序列化table永远不会发生。

您应该做的是修改您的 proguard 配置以保留readObjectwriteObject方法。标准的 proguard 文档甚至在这里有一个示例(搜索“readObject”): http: //proguard.sourceforge.net/index.html#manual/examples.html

于 2013-03-07T17:25:03.847 回答