2016-05-05 1 views
0

Я делаю приложение для android, используя scala (макронная каркас, если это имеет значение). Я получаю проблемы при десериализации JSON с использованием библиотеки Genson.Проблема с Genson с android - scala.Array не найден

Вот минимальный код ошибки

case class Model(a: Int, b: String) 

class MainActivity extends FragmentActivity with Contexts[FragmentActivity] with IdGeneration with AkkaActivity { 
.... 
    override def onCreate(savedInstanceState: Bundle) = { 
     super.onCreate(savedInstanceState); 
     val m = com.owlike.genson.defaultGenson.fromJson[Model]("""{"a":1, "b":"hello"}""") 
     .... 
    } 

} 

я получил следующую ошибку работает этот код на эмуляторе

05-05 10:55:29.996 5439-5439/? I/art: Not late-enabling -Xcheck:jni (already on) 
05-05 10:55:30.033 5439-5439/com.myapp.dfp W/System: ClassLoader referenced unknown path: /data/app/com.myapp.dfp-2/lib/x86 
05-05 10:55:30.316 5439-5439/com.myapp.dfp D/AndroidRuntime: Shutting down VM 
05-05 10:55:30.316 5439-5439/com.myapp.dfp E/AndroidRuntime: FATAL EXCEPTION: main 
Process: com.myapp.dfp, PID: 5439 
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.myapp.dfp/com.myapp.dfp.MainActivity}: scala.reflect.internal.MissingRequirementError: class scala.Array in JavaMirror with dalvik.system.PathClassLoader[DexPathList[[zip file "/system/framework/android.test.runner.jar", zip file "/data/app/com.myapp.dfp-2/base.apk"],nativeLibraryDirectories=[/data/app/com.myapp.dfp-2/lib/x86, /vendor/lib, /system/lib]]] of type class dalvik.system.PathClassLoader with classpath [<unknown>] and parent being [email protected] of type class java.lang.BootClassLoader with classpath [<unknown>] and parent being primordial classloader with boot classpath [/system/framework/core-libart.jar:/system/framework/conscrypt.jar:/system/framework/okhttp.jar:/system/framework/core-junit.jar:/system/framework/bouncycastle.jar:/system/framework/ext.jar:/system/framework/framework.jar:/system/framework/telephony-common.jar:/system/framework/voip-common.jar:/system/framework/ims-common.jar:/system/framework/apache-xml.jar:/system/framework/org.apache.http.legacy.boot.jar] not found. 
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2416) 
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476) 
    at android.app.ActivityThread.-wrap11(ActivityThread.java) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344) 
    at android.os.Handler.dispatchMessage(Handler.java:102) 
    at android.os.Looper.loop(Looper.java:148) 
    at android.app.ActivityThread.main(ActivityThread.java:5417) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616) 
    Caused by: scala.reflect.internal.MissingRequirementError: class scala.Array in JavaMirror with dalvik.system.PathClassLoader[DexPathList[[zip file "/system/framework/android.test.runner.jar", zip file "/data/app/com.myapp.dfp-2/base.apk"],nativeLibraryDirectories=[/data/app/com.myapp.dfp-2/lib/x86, /vendor/lib, /system/lib]]] of type class dalvik.system.PathClassLoader with classpath [<unknown>] and parent being [email protected] of type class java.lang.BootClassLoader with classpath [<unknown>] and parent being primordial classloader with boot classpath [/system/framework/core-libart.jar:/system/framework/conscrypt.jar:/system/framework/okhttp.jar:/system/framework/core-junit.jar:/system/framework/bouncycastle.jar:/system/framework/ext.jar:/system/framework/framework.jar:/system/framework/telephony-common.jar:/system/framework/voip-common.jar:/system/framework/ims-common.jar:/system/framework/apache-xml.jar:/system/framework/org.apache.http.legacy.boot.jar] not found. 
    at scala.reflect.internal.MissingRequirementError$.signal(MissingRequirementError.scala:17) 
    at scala.reflect.internal.MissingRequirementError$.notFound(MissingRequirementError.scala:18) 
    at scala.reflect.internal.Mirrors$RootsBase.getModuleOrClass(Mirrors.scala:53) 
    at scala.reflect.internal.Mirrors$RootsBase.getModuleOrClass(Mirrors.scala:66) 
    at scala.reflect.internal.Mirrors$RootsBase.getClassByName(Mirrors.scala:102) 
    at scala.reflect.internal.Mirrors$RootsBase.getRequiredClass(Mirrors.scala:105) 
    at scala.reflect.internal.Definitions$DefinitionsClass.ArrayClass$lzycompute(Definitions.scala:440) 
    at scala.reflect.internal.Definitions$DefinitionsClass.ArrayClass(Definitions.scala:440) 
    at scala.reflect.internal.Definitions$DefinitionsClass.arrayType(Definitions.scala:827) 
    at scala.reflect.internal.Definitions$DefinitionsClass$$anonfun$JavaRepeatedParamClass$1.apply(Definitions.scala:374) 
    at scala.reflect.internal.Definitions$DefinitionsClass$$anonfun$JavaRepeatedParamClass$1.apply(Definitions.scala:374) 
    at scala.reflect.internal.Definitions$DefinitionsClass.specialPolyClass(Definitions.scala:1268) 
    at scala.reflect.internal.Definitions$DefinitionsClass.JavaRepeatedPa 
05-05 11:00:30.355 5439-5439/com.myapp.dfp I/Process: Sending signal. PID: 5439 SIG: 9 

Десериализация отлично работает при запуске на компьютере (как приложение автономный Scala)

Я добавил -keep class com.owlike.genson.*{ *; } опции build.sbt, как было рекомендовано here, но это не поможет. Вот часть моего build.sbt:

proguardOptions in Android ++= Seq(
    "-ignorewarnings", 
    "-keepattributes Signature", 
    "-keep class scala.Dynamic", 
    "-keep class scala.collection.SeqLike {public protected *;}", 
    "-keep public class scala.PartialFunction", 
    "-keep class com.owlike.genson.*{ *; }", 
    "-keep class com.myapp.** { *; }" 
) 

Я пытался добавить -keep class scala.Array, чтобы Proguard варианты, это не помогает также.

Что еще я могу сделать?


EDITED: Я не добился успеха с помощью genson и теперь использую gson. Gson отлично работает для меня.

+0

Тогда он работал бы с java-версией Genson, если был настроен на использование полей вместо свойств (это в основном делает Gson). – eugen

+0

Я бы использовал circe или Argonaut, оба - синтаксические анализаторы json с высокой производительностью, которые вообще не используют отражение, они также являются идиоматическими scala в отличие от genson или gson. – pfn

ответ

0

Можете ли вы попытаться использовать scala.Array в своем коде без Genson. Просто чтобы убедиться, что он присутствует.

Тогда попробуйте использовать scala.Array и в то же время использовать Genson. Если это работает, это означает, что scala.Array, вероятно, исключается proguard.

Если это не работает, я подозреваю, что классный загрузчик, используемый Genson, не видит класс Array. В этом случае попробуйте это сделать:

val builder = new GensonBuilder().withClassLoader(Array.getClassLoader) 
val genson = new ScalaGenson(builder.withBundle(ScalaBundle()).create()) 
genson.fromJson... 
+0

Eugen, спасибо за помощь. У меня есть объекты Array в моем коде, поэтому он присутствует в classpath. Исключением строителя является то же самое. Я не понимаю, почему stacktrace содержит только классы scala.reflect. *. – Alexander

+0

У вас есть более подробная таблица? Что-то, что покажет, какая часть кода Генсона производит его? – eugen

+0

Нет. Я напечатал stacktrace исключения для PrintWriter, я не вижу никакого кода genson или кода активности. На самом деле вопрос сейчас не уместен, может быть, позже я поймаю проблему – Alexander