6

我正在编写一个简单的应用程序,Scala它通过库使用 leveldb 数据库leveldbjni。我的build.sbt文件如下所示:

name := "Whatever"

version := "1.0"

scalaVersion := "2.10.2"

libraryDependencies ++= Seq(
    "org.iq80.leveldb" % "leveldb-api" % "0.6",
    "org.fusesource.leveldbjni" % "leveldbjni-all" % "1.7"
)

然后AnObject负责创建数据库。不幸的是,如果我运行程序,我会得到一个java.lang.UnsatisfiedLinkError, 由在引擎盖下利用的hawtjni库提出的。leveldbjni

该错误也可以从 scala 控制台轻松触发:

scala> import java.io.File
scala> import org.iq80.leveldb._
scala> import org.fusesource.leveldbjni.JniDBFactory._
scala> factory.open(new File("test"), new Options().createIfMissing(true))

java.lang.UnsatisfiedLinkError: org.fusesource.leveldbjni.internal.NativeOptions.init()V
    at org.fusesource.leveldbjni.internal.NativeOptions.init(Native Method)
    at org.fusesource.leveldbjni.internal.NativeOptions.<clinit>(NativeOptions.java:54)
    at org.fusesource.leveldbjni.JniDBFactory$OptionsResourceHolder.init(JniDBFactory.java:98)
    at org.fusesource.leveldbjni.JniDBFactory.open(JniDBFactory.java:167)
    at .<init>(<console>:15)
...
scala> System getProperty "java.io.tmpdir"
res2: String = /var/folders/1l/wj6yg_wd15sg_gcql001wchm0000gn/T/

我无法理解发生了什么,因为该库已从 jar 文件中正确提取,但由于某些原因未加载。

$ file /var/folders/1l/wj6yg_wd15sg_gcql001wchm0000gn/T/lib*
/var/folders/1l/wj6yg_wd15sg_gcql001wchm0000gn/T/libleveldbjni-1.7.jnilib:    Mach-O universal binary with 2 architectures
/var/folders/1l/wj6yg_wd15sg_gcql001wchm0000gn/T/libleveldbjni-1.7.jnilib (for architecture x86_64):    Mach-O 64-bit dynamically linked shared library x86_64
/var/folders/1l/wj6yg_wd15sg_gcql001wchm0000gn/T/libleveldbjni-1.7.jnilib (for architecture i386):  Mach-O dynamically linked shared library i386

我认为这个问题可能与 sbt 使用的类加载器有关,但我不确定,因为我对 scala 比较陌生。

更新

仍然没有找到罪魁祸首或罪魁祸首。无论如何,实际上已找到并正确加载了该库,因为我可以执行以下命令:

scalac> import org.fusesource.leveldbjni.internal.NativeDB
scalac> NativeDB.LIBRARY.load()

该错误在某种程度上是由于init()根据hawtjni 文档负责将所有注释为常量字段的静态字段设置为具有常量值的函数。仍然可以通过键入以下内容触发异常:

scalac> import org.fusesource.leveldbjni.internal.NativeOptions
scalac> new NativeOptions()
java.lang.UnsatisfiedLinkError: org.fusesource.leveldbjni.internal.NativeOptions.init()V
    at org.fusesource.leveldbjni.internal.NativeOptions.init(Native Method)
    at org.fusesource.leveldbjni.internal.NativeOptions.<clinit>(NativeOptions.java:54)
    at .<init>(<console>:9)
4

1 回答 1

2

显然这是这个sbt issue page中记录的已知问题。根据eventsourced 文档,我已经实现了一个自定义run-nobootcp命令,该命令在不将 Scala 库添加到引导类路径的情况下执行代码。

这应该可以解决问题。

于 2013-07-20T14:57:21.603 回答