我正在尝试实现自定义集合,所以我正在关注这篇文章。
现在,导致崩溃的“裸”代码是这样的:
object TraversableCollection extends TraversableFactory[TraversableCollection] {
implicit def canBuildFrom[T] : CanBuildFrom[Coll, T, TraversableCollection[T]] = new GenericCanBuildFrom[T]
def newBuilder[T] = new ListBuffer[T].mapResult(x => new TraversableCollection(x:_*))
}
class TraversableCollection[T](seq : T*) extends Traversable[T]
with GenericTraversableTemplate[T, TraversableCollection]
with TraversableLike[T, TraversableCollection[T]] {
override def companion = TraversableCollection
def foreach[U](f: T => U) = seq.foreach(f)
}
我的整个实现并不复杂,它只是一个包含一个不可变的集合,Vector
它可以在项目上前后移动,在内部存储索引值并公开“当前”项目。所以本质上 - 一个双向迭代器。
无论如何,整个实现(上面的骨架和我的完整实现)在 REPL 中工作得很好(这里是一个完整的实现):
scala> val vec = Vector("Just", "testing", "it")
vec: scala.collection.immutable.Vector[String] = Vector(Just, testing, it)
scala> val t = new TraversableCollection(vec:_*)
t: TraversableCollection[String] = (Just, testing, it)
scala> t.current
res6: String = Just
scala> t.moveForward(2)
scala> t.current
res8: String = it
scala> t.moveBack(1)
scala> t.current
res10: String = testing
很简单,我需要它。但问题是:在 Android 应用程序中,当我到达创建集合的点时,PathClassLoader
开始崩溃,然后到达:
this: NoClassDefFoundError
detailMessage: "scala.collection.generic.TraversableFactory"
并在日志中死去:
05-16 12:14:35.056: E/AndroidRuntime(575): java.lang.VerifyError: my.package.data.TraversableCollection
为什么?我能做些什么吗?
现在我的临时解决方案是删除GenericTraversableTemplate
and TraversableLike
,并留下Traversable
特征。不完美,但现在可以工作。
设置:
来自 ADT 的 Eclipse,Scala IDE 插件 + AndroidProguardScala 插件
斯卡拉 2.10.0-2012-12-05
[编辑] Proguard 相关
根据我对 Proguard 的非常非常有限的了解,它似乎应该被禁用,因为路径被注释掉了project.properties
:
# To enable ProGuard to shrink and obfuscate your code, uncomment this (available properties: sdk.dir, user.home):
#proguard.config=${sdk.dir}/tools/proguard/proguard-android.txt:proguard-project.txt
我尝试注释和取消注释该行,并将其添加到 proguard-project.txt:
-keepnames scala.collection.generic.TraversableFactory
无论如何,我仍然有同样的错误。
[编辑 2] 仍然与 Proguard 相关
这是 AndroidProguardScala 生成的默认配置(在应用程序崩溃时使用):
-injars "F:\adt-bundle-windows-x86_64\eclipse\configuration\org.eclipse.osgi\bundles\373\1\.cp\lib\scala-library.jar"(!META-INF/MANIFEST.MF)
-injars "F:\adt-bundle-windows-x86_64\eclipse\configuration\org.eclipse.osgi\bundles\373\1\.cp\lib\scala-actors.jar"(!META-INF/MANIFEST.MF)
-injars "F:\adt-bundle-windows-x86_64\eclipse\configuration\org.eclipse.osgi\bundles\372\1\.cp\lib\scala-reflect.jar"(!META-INF/MANIFEST.MF)
-outjars "F:\Projects\AndroidScalaProject\proguard_cache\jartender_cache_7212076580182750151.jar"
-injars "F:\Projects\AndroidScalaProject\bin\classes"
-libraryjars "F:\adt-bundle-windows-x86_64\sdk\platforms\android-17\android.jar"
-libraryjars "F:\Projects\AndroidScalaProject\libs\android-support-v4.jar"
-keep public class * extends android.**
-dontwarn **$$anonfun$*
-dontwarn
-dontoptimize
-dontobfuscate
-dontskipnonpubliclibraryclasses
-dontskipnonpubliclibraryclassmembers
-ignorewarnings
-forceprocessing
-keepattributes Exceptions,InnerClasses,Signature,Deprecated,
SourceFile,LineNumberTable,*Annotation*,EnclosingMethod
-keep public class scala.ScalaObject { *; }
-keep public class scala.Function0, scala.Function1
# Fix for https://issues.scala-lang.org/browse/SI-5397
-keep class scala.collection.SeqLike {
public protected *;
}
# AndroidProguardScala version: 0.0.47.201302150529
-keep class ...
-keep class ... # and a looot of that for anon-functions and classes in the project.
# Here's one that's interesting:
-keep class my.project.data.TraversableCollection {*;}