httpurlConnection.getInputStream (), приводящий к фатальной ошибке - PullRequest
0 голосов
/ 16 марта 2019

Я разрабатываю приложение для Android для бронирования.Это мой первый вопрос по stackoverflow.Так что извините меня, если я не буду точно следовать правилам.Я застрял с этой фатальной ошибкой:

FATAL EXCEPTION: main


 Process: com.connectandroiddb.myapplication, PID: 26440
    android.os.NetworkOnMainThreadException
        at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1147)
        at java.net.InetAddress.lookupHostByName(InetAddress.java:418)
        at java.net.InetAddress.getAllByNameImpl(InetAddress.java:252)
        at java.net.InetAddress.getAllByName(InetAddress.java:215)
        at com.android.okhttp.HostResolver$1.getAllByName(HostResolver.java:29)
        at com.android.okhttp.internal.http.RouteSelector.resetNextInetSocketAddress(RouteSelector.java:232)
        at com.android.okhttp.internal.http.RouteSelector.next(RouteSelector.java:124)
        at com.android.okhttp.internal.http.HttpEngine.connect(HttpEngine.java:272)
        at com.android.okhttp.internal.http.HttpEngine.sendRequest(HttpEngine.java:211)
        at com.android.okhttp.internal.http.HttpURLConnectionImpl.execute(HttpURLConnectionImpl.java:382)
        at com.android.okhttp.internal.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.java:332)
        at com.android.okhttp.internal.http.HttpURLConnectionImpl.getInputStream(HttpURLConnectionImpl.java:199)
        at com.example.myapplication.JSONParser.makeHttpRequest(JSONParser.java:65)
        at com.example.myapplication.MainActivity$GetProductDetails$1.run(MainActivity.java:144)
        at android.os.Handler.handleCallback(Handler.java:739)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:135)
        at android.app.ActivityThread.main(ActivityThread.java:5254)
        at java.lang.reflect.Method.invoke(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:372)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)

Я позаимствовал код у: https://androidhive.info/2012/05/how-to-connect-android-with-php-mysql/ Соответствующий код ниже: я вызываю GetProductDetails (), как показано ниже в

public void sendMessage(View view) {
.....
new GetProductDetails().execute(); 
......
}

, который находится в MainActivity.java

/**
     * Background Async Task to Get complete product details
     * */
class GetProductDetails extends AsyncTask<String, String, String> { 
        protected void onPreExecute() {
            super.onPreExecute();
        }
        /**
         * Getting product details in background thread
         * */
        protected String doInBackground(String... params) {

            // updating UI from Background Thread
            runOnUiThread(new Runnable() {
                public void run() {

                    JSONObject json = jsonParser.makeHttpRequest(
                            url_get_product_details, "GET");// Line 144

                }
            });
                     return null;
        }

                /**
         * After completing background task Dismiss the progress dialog
         * **/
                protected void onPostExecute(String file_url) {
                    }
    } 

Это соответствующие части из JSONParser:

private String readStream(InputStream is) {
        try {
            ByteArrayOutputStream bo = new ByteArrayOutputStream();
            int i = is.read();
            while(i != -1) {
                bo.write(i);
                i = is.read();
            }
            return bo.toString();
        } catch (IOException e) {
            return "";
        }
    }

        public JSONObject makeHttpRequest(String url, String method)
    {
 HttpURLConnection urlConnection = null;

        // Making HTTP request
        try {

            // check for request method
            if(method == "POST"){
                // request method is POST  CHECK


            }else if(method == "GET"){
                // request method is GET

            URL u = new URL("http://example.com/");
            urlConnection = (HttpURLConnection) u.openConnection();
            InputStream in = new BufferedInputStream(urlConnection.getInputStream()); //JSONParser.java:65
            json=readStream(in);
            }    
        }   
        catch (MalformedURLException ex) {
            Logger.getLogger(getClass().getName()).log(Level.SEVERE, null, ex);
        } 
        catch (IOException ex) {
            Logger.getLogger(getClass().getName()).log(Level.SEVERE, null, ex);
        }
        finally {
            if (urlConnection != null) {
                try {
                    urlConnection.disconnect();
                } catch (Exception ex) {
                    Logger.getLogger(getClass().getName()).log(Level.SEVERE, null, ex);
                }       
        } 
        }
}

Я добавил интернет-разрешение в AndroidManifest.xml.Пожалуйста помоги!!!.Заранее спасибо,

1 Ответ

0 голосов
/ 16 марта 2019

Ваша проблема в этой части кода:

protected String doInBackground(String... params) {

    // updating UI from Background Thread
    runOnUiThread(new Runnable() {
        public void run() {

            JSONObject json = jsonParser.makeHttpRequest(url_get_product_details, "GET");   
        }
    });
    return null;
}

Как видите, в методе doInBackground вы выполняете что-то в потоке пользовательского интерфейса. Android запускает все пользовательские интерфейсы в одном и том же потоке пользовательского интерфейса, и вы не можете выполнять в этом потоке никакую долгосрочную задачу (например, сетевой запрос), поскольку пользовательский интерфейс останется заблокированным.

Вот почему люди используют асинхронные задачи или простые потоки. Если в методе doInBackground вы выполняете runOnUiThread, вы нарушаете это правило, и возникает исключение NetworkOnMainThreadException.

Прочтите учебное пособие здесь о том, как передавать информацию между фоном и пользовательским интерфейсом, используя AsyncTask. Совет, вам нужно использовать onPostExecute

...