WebView方法在同一线程上出错


84

我有一个android程序(在webview中为Java + html)。我可以从javascript调用Java代码。但是另一种方式停止了工作(在Eclipse中更新之后)。

所以这就是我想要做的

  • 制作网络视图(有效)
  • 在JavaScript中调用AndroidFunction.test(); (工作)
  • java test()函数调用webView.loadUrl(“ javascript:helloBack()”); (!不再工作了)

我试图让它与MainActivity中的WebView一起使用,但是没有起作用。

MainActivity.java

public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        final WebView webView = (WebView)findViewById(R.id.webView);
        webView.getSettings().setJavaScriptEnabled(true);
        webView.setWebChromeClient(new WebChromeClient());
        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true);

        javascr = new Javascript(this, webView);
        webView.addJavascriptInterface(javascr, "AndroidFunction");
        webView.loadUrl("file:///android_asset/www/index.html");

        ....
}

Java脚本

public class Javascript {   
    Context cont;
    WebView webView;

    Javascript(Context c, WebView w) {
        cont = c;
        webView = w;
    }

    // function called in the javascript by AndroidFunction.test();
    public void test() {
          // Breaking point!!!
        webView.loadUrl("javascript:helloBack()");
    }

错误:

03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.handleMessage(SystemMessageHandler.java:27)
03-24 11:47:50.103: W/WebView(21026):   java.lang.Throwable: A WebView method was called on thread 'JavaBridge'. All WebView methods must be called on the same thread. (Expected Looper Looper{41ab68f8} called on Looper{41bb70a8}, FYI main Looper is Looper{41ab68f8})

03-24 11:47:50.103: W/WebView(21026):   at android.webkit.WebView.checkThread(WebView.java:2063)
03-24 11:47:50.103: W/WebView(21026):   at android.webkit.WebView.loadUrl(WebView.java:794)
03-24 11:47:50.103: W/WebView(21026):   at com.example.hellobt.Javascript.test(Javascript.java:24)

03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.nativeDoRunLoopOnce(Native Method)
03-24 11:47:50.103: W/WebView(21026):   at com.android.org.chromium.base.SystemMessageHandler.handleMessage(SystemMessageHandler.java:27)
03-24 11:47:50.103: W/WebView(21026):   at android.os.Handler.dispatchMessage(Handler.java:102)

03-24 11:47:50.103: W/WebView(21026):   at android.os.Looper.loop(Looper.java:137)
03-24 11:47:50.103: W/WebView(21026):   at android.os.HandlerThread.run(HandlerThread.java:61)

谢谢你的回答。我像这样在Javascript文件中编辑了该函数:

private void test(final String s) {
        webView.post(new Runnable() {
            public void run() {
                webView.loadUrl("javascript:" + s + ";");
            }
        });
        System.out.println("javscript done..");
    }

在您的编辑中,您提到您是根据答案之一来更改代码的,这听起来还是没有解决您的问题,但是答案是可以接受的。我建议您明确说明这是否解决了问题(或者只是撤消了编辑)。
汤姆

Answers:


212

JavaScript方法在后台(即非UI)线程上执行。您需要在UI线程上调用所有与Android View相关的方法。您可以通过以下方式实现所需的功能:

mWebView.post(new Runnable() {
    @Override
    public void run() {
        mWebView.loadUrl(...).
    }
});

它将发布任务以在UI线程上运行。


您将代码段(mWebView)添加到原始代码的位置是什么?
Bowie

在test()函数中,请参阅我上面的问题。
Johan Hoeksma 2014年

如果需要返回值怎么办?例如返回webview.getUrl?stackoverflow.com/questions/29748782/... @JohanHoeksma
苏雷什苏贝迪

您知道发生此问题的方式和时间吗?在我的Nexus 5X(Android 7.1.1)中,可以,但是某些设备崩溃了。
艾莉·邹

如何传递变量?我试图在mWebView.loadUrl(“ javascript:test('” + MyData +“');”);中添加变量 但价值永远不会传递给javascript!
user1788736

14

就我而言,WebView中未显示任何内容,因此我更喜欢另一种方式:

runOnUiThread(new Runnable() {
    @Override
    public void run() {
        final WebView webView = (WebView) findViewById(R.id.map);
        webView.loadDataWithBaseURL(...);
    }
});


2

Java版本:必须使用Runnable接口和Post to Handler

webView.post(new Runnable() {
          @Override
          public void run() {
             webView.loadUrl("file:///android_asset/www/index.html");
          }
       });

Kotlin版本:

webView.post(new Runnable {
   webView.loadUrl("file:///android_asset/www/index.html")
})
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.