동일한 스레드 오류에 대한 WebView 메서드


84

나는 안드로이드 프로그램 (웹보기의 Java + html)이 있습니다. 자바 스크립트에서 자바 코드로 호출 할 수 있습니다. 그러나 다른 방법으로 작동이 중지되었습니다 (일식에서 업데이트 한 후).

그래서 이것이 제가하려는 것입니다.

  • 웹뷰 만들기 (작동)
  • 자바 스크립트에서 AndroidFunction.test (); (일함)
  • 자바 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");

        ....
}

Javascript.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..");
    }

편집에서 답변 중 하나를 기반으로 코드를 변경했다고 언급하여 여전히 문제가 해결되지 않은 것처럼 들리지만 허용되는 답변이 있습니다. 이것이 문제가 해결되었는지 여부를 명시 적으로 지정하는 것이 좋습니다 (또는 편집을 취소 할 수 있음).
Tom

답변:


212

JavaScript 메서드는 백그라운드 (즉, UI가 아닌) 스레드에서 실행됩니다. UI 스레드에서 모든 Android View 관련 메서드를 호출해야합니다. 다음을 통해 필요한 것을 얻을 수 있습니다.

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

UI 스레드에서 실행할 작업을 게시합니다.


원본 코드에 스 니펫 (mWebView)을 어디에 추가 했습니까?
Bowie

test () 함수에서 위의 질문을 참조하십시오.
Johan Hoeksma 2014-08-26

값을 반환해야하는 경우 어떻게합니까? 예를 들어 webview.getUrl을 반환합니까? stackoverflow.com/questions/29748782/... @JohanHoeksma
수레 쉬 Subedi

이 문제가 언제 어떻게 발생했는지 알고 있습니까? 내 Nexus 5X (Android 7.1.1)에서는 괜찮지 만 일부 기기가 충돌했습니다.
Ellie Zou

변수를 전달하는 방법? mWebView.loadUrl ( "javascript : test ( '"+ MyData + "');");에 변수를 추가하려고했습니다. 그러나 값은 절대로 자바 스크립트에 전달되지 않습니다!
user1788736 dec.

14

제 경우에는 WebView에 아무것도 표시되지 않았으므로 다른 방법을 선호합니다.

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

3

이것은 post 메소드를 사용하여 올 수 있습니다. 아래 코드를 살펴보십시오.

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

2

자바 버전 : 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")
})
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.