3

在 Kotlin 中实现 atwitter4j.StatusListner时,我得到以下IllegalAccessError和相关的堆栈跟踪:

Exception in thread "main" java.lang.IllegalAccessError: tried to access class twitter4j.StreamListener from class rxkotlin.rxextensions.TwitterExampleKt$observe$1
    at rxkotlin.rxextensions.TwitterExampleKt$observe$1.subscribe(TwitterExample.kt:50)
    at io.reactivex.internal.operators.observable.ObservableCreate.subscribeActual(ObservableCreate.java:40)
    at io.reactivex.Observable.subscribe(Observable.java:10700)
    at io.reactivex.Observable.subscribe(Observable.java:10686)
    at io.reactivex.Observable.subscribe(Observable.java:10615)
    at rxkotlin.rxextensions.TwitterExampleKt.main(TwitterExample.kt:8)

由以下代码产生:

val twitterStream = TwitterStreamFactory().instance
// See https://stackoverflow.com/questions/37672023/how-to-create-an-instance-of-anonymous-interface-in-kotlin/37672334
twitterStream.addListener(object : StatusListener {
    override fun onStatus(status: Status?) {
        if (emitter.isDisposed) {
            twitterStream.shutdown()
        } else {
            emitter.onNext(status)
        }
    }

    override fun onException(e: Exception?) {
        if (emitter.isDisposed) {
            twitterStream.shutdown()
        } else {
            emitter.onError(e)
        }
    }

    // Other overrides.
})
emitter.setCancellable { twitterStream::shutdown }

如果我不使用 Rx,它会使异常更简单一些:

twitterStream.addListener(object: twitter4j.StatusListener {
    override fun onStatus(status: Status) { println("Status: {$status}") }
    override fun onException(ex: Exception) { println("Error callback: $ex") }
    // Other overrides.
})

Exception in thread "main" java.lang.IllegalAccessError: tried to access class twitter4j.StreamListener from class rxkotlin.rxextensions.TwitterExampleKt
at rxkotlin.rxextensions.TwitterExampleKt.main(TwitterExample.kt:14)

但是,如果我实现 Java 包装函数,则不会引发错误并且行为符合预期:

包装 -

public class Twitter4JHelper {
  public static void addStatusListner(TwitterStream stream, StatusListener listner) {
    stream.addListener(listner);
  }
}

修改后的实施 -

val twitterStream = TwitterStreamFactory().instance
val listner = object: StatusListener {
    override fun onStatus(status: Status?) {
        if (emitter.isDisposed) {
            twitterStream.shutdown()
        } else {
            emitter.onNext(status)
        }
    }

    override fun onException(e: Exception?) {
        if (emitter.isDisposed) {
            twitterStream.shutdown()
        } else {
            emitter.onError(e)
        }
    }

    // Other overrides.
}

Twitter4JHelper.addStatusListner(twitterStream, listner)
emitter.setCancellable { twitterStream::shutdown }

这个修改后的解决方案来自一篇博客文章,我认为它试图解释原因,但谷歌翻译不是我的朋友。是什么原因造成的IllegalAccessError?是否有一个纯粹基于 Kotlin 的解决方案,还是我必须接受这种解决方法?

4

2 回答 2

1

我通过使用 java 实用程序类解决了这个问题:

public class T4JCompat {

  public static void addStatusListener(TwitterStream stream, StatusListener listener) {
    stream.addListener(listener);
  }

  public static void removeStatusListener(TwitterStream stream, StatusListener listener) {
    stream.removeListener(listener);
  }

} 

您可以从 Kotlin 调用这些方法,并且事情会按预期工作。

于 2017-09-24T22:45:21.673 回答
1

是的,那是行不通的。

addListener方法需要一个StreamListener参数并且StreamListener是非公共的(包私有)。为此,我肯定会针对 Kotlin 编译器提出一个错误。

Kotlin 编译器生成的代码是:

TwitterStream twitterStream = (new TwitterStreamFactory()).getInstance();
twitterStream.addListener((StreamListener)(new StatusListener() {
         // ..overrides ...
      }));

StatusListener已经实现StreamListener了,所以我不明白为什么需要演员表。

于 2017-06-14T11:30:01.043 回答