WebView问题系列:
1.遇到的问题-基本使用
2.遇到的问题- All WebView methods must be called on the same thread
1.问题场景:
JavaScript调用Android native方法finish掉当前界面,JavaScript调用代码为:
setTimeout(
function(){
Android.close();
},50);
Android 代码为(部分):
WebSettings webSettings = webview.getSettings();
webSettings.setJavaScriptEnabled(true);
webview.addJavascriptInterface(new WebAppInterface(), "Android");
private final class WebAppInterface {
// JavaScript调用此方法关闭页面
@JavascriptInterface
public void close() {
finish();
}
}
然而,并没有达到预期效果,抛出了以下异常:
exception--->java.lang.Throwable: A WebView method was called on thread 'JavaBridge'.
All WebView methods must be called on the same thread.
(Expected Looper Looper (main, tid 2) {fe6c9a9} called on Looper (JavaBridge, tid 884) {f5c0c14},
FYI main Looper is Looper (main, tid 2) {fe6c9a9})
2.先说解决办法,直接将finish()用runOnUIThread()包裹起来或者用view.post(runnable),将finish包裹在runnable里面就行了,效果也有了,异常也不报了
疑问:那是不是所有供JavaScript调用的native方法,都需要这样包裹起来呢?异常抛出来的根本原因是什么呢?
开始没有注意业务代码,后来定位代码才发现以下内容,当前Activity复写了finish方法,而复写的里面触发了UIThread的检查才导致的以上报错
@Override
public void finish() {
JRPCTTTaskManager.getInstance().releaseTask();
Intent intent = getIntent();
intent.putExtra(INTENT_EXTRA_JRPC_NEXT_URL, ui_webview.getUrl());
setResult(RESULT_OK, intent);
super.finish();
}
**************WebView************
@ViewDebug.ExportedProperty(category = "webview")
public String getUrl() {
checkThread();//就是这里检查线程时抛出了异常
return mProvider.getUrl();
}
3.问题解答
1.区分业务场景,setResult
2.将finish方法用UIThread包裹
既然至此,就看下android源码怎么说的吧,大写的尴尬。。。
其中框中明确说了,js和java对象在webview的后台、私有线程交互的,所以要注意线程安全
网友评论