Złóż wniosek HTTP w Androidzie

352

Szukałem wszędzie, ale nie mogłem znaleźć odpowiedzi, czy istnieje sposób, aby złożyć proste żądanie HTTP? Chcę poprosić o stronę / skrypt PHP na jednej z moich stron, ale nie chcę pokazywać strony.

Jeśli to możliwe, chcę to zrobić nawet w tle (w BroadcastReceiver)

Mats Hofman
źródło

Odpowiedzi:

477

AKTUALIZACJA

To bardzo stara odpowiedź. Zdecydowanie nie będę już polecał klienta Apache. Zamiast tego użyj:

Oryginalna odpowiedź

Po pierwsze, poproś o pozwolenie na dostęp do sieci, dodaj następujące elementy do swojego manifestu:

<uses-permission android:name="android.permission.INTERNET" />

Najprostszym sposobem jest użycie klienta HTTP Apache w pakiecie z Androidem:

    HttpClient httpclient = new DefaultHttpClient();
    HttpResponse response = httpclient.execute(new HttpGet(URL));
    StatusLine statusLine = response.getStatusLine();
    if(statusLine.getStatusCode() == HttpStatus.SC_OK){
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        response.getEntity().writeTo(out);
        String responseString = out.toString();
        out.close();
        //..more logic
    } else{
        //Closes the connection.
        response.getEntity().getContent().close();
        throw new IOException(statusLine.getReasonPhrase());
    }

Jeśli chcesz, aby działał w osobnym wątku, polecam rozszerzenie AsyncTask:

class RequestTask extends AsyncTask<String, String, String>{

    @Override
    protected String doInBackground(String... uri) {
        HttpClient httpclient = new DefaultHttpClient();
        HttpResponse response;
        String responseString = null;
        try {
            response = httpclient.execute(new HttpGet(uri[0]));
            StatusLine statusLine = response.getStatusLine();
            if(statusLine.getStatusCode() == HttpStatus.SC_OK){
                ByteArrayOutputStream out = new ByteArrayOutputStream();
                response.getEntity().writeTo(out);
                responseString = out.toString();
                out.close();
            } else{
                //Closes the connection.
                response.getEntity().getContent().close();
                throw new IOException(statusLine.getReasonPhrase());
            }
        } catch (ClientProtocolException e) {
            //TODO Handle problems..
        } catch (IOException e) {
            //TODO Handle problems..
        }
        return responseString;
    }

    @Override
    protected void onPostExecute(String result) {
        super.onPostExecute(result);
        //Do anything with response..
    }
}

Następnie możesz złożyć wniosek:

   new RequestTask().execute("http://stackoverflow.com");
Konstantin Burov
źródło
11
Oto artykuł z oficjalnej android blogu deweloperskim na AsyncTask: android-developers.blogspot.com/2010/07/...
Austyn Mahoney
77
w przypadku pierników lub wyższych zaleca się używanie HttpURLConnection zamiast biblioteki apache, patrz android-developers.blogspot.com/2011/09/… . Jest mniej podatny na baterię i ma lepszą wydajność
Marty
8
responseString = out.toString () musi znajdować się przed wywołaniem out.close (). Właściwie powinieneś prawdopodobnie mieć out.close () w końcu bloku. Ale ogólnie bardzo pomocna odpowiedź (+1), dzięki!
dcp
9
Od wersji plastra miodu (SDK 11) podejście asynchroniczne jest właściwym rozwiązaniem. NetworkOnMainThreadException zostaje wrzucony podczas próby uruchomienia żądania HTTP z głównego wątku.
msrxthr
2
Ta odpowiedź jest całkiem doskonała. Ale odradzam używanie AsyncTasks do pracy w sieci. Mogą bardzo łatwo tworzyć wycieki pamięci (a faktycznie podany przykład przecieka) i nie zapewniają wszystkich funkcji, których można oczekiwać w przypadku żądań sieciowych. Zastanów się nad użyciem RoboSpice do tego rodzaju zadań w tle: github.com/octo-online/robospice
Snicolas,
67

chyba że masz wyraźny powód, aby wybrać klienta Apache HttpClient, powinieneś wybrać java.net.URLConnection. w sieci można znaleźć wiele przykładów tego, jak go używać.

poprawiliśmy również dokumentację Androida od czasu twojego oryginalnego postu: http://developer.android.com/reference/java/net/HttpURLConnection.html

i rozmawialiśmy o kompromisach na oficjalnym blogu: http://android-developers.blogspot.com/2011/09/androids-http-clients.html

Elliott Hughes
źródło
13
Dlaczego używanie Apache HttpClient nie jest zalecane?
Ted
@ElliottHughes: Zgadzam się w 100%. Nie można zaprzeczyć, że httpclient Apache oferuje łatwe metody i bardziej abstrakcyjny widok protokołu, ale natywne urlconnection Java jest w żaden sposób mniej użyteczne. Przy odrobinie praktyczności jest równie łatwy w użyciu jak httpclient i jest znacznie bardziej przenośny
Nitin Bansal
1
Właściwie, jeśli spojrzysz na wideo Google I / O 2010 - aplikacje klienckie REST dla Androida ( youtube.com/watch?v=xHXn3Kg2IQE 57min21sec), zobaczysz, że Apache HttpClient jest najbardziej zalecany. Cytuję Virgila Dobjanschi (inżyniera oprogramowania w Google, który działa na Android Application Group) „Po prostu radzę używać klienta HTTP Apache, ponieważ ma on bardziej niezawodną implementację. Transakcja HTTP typu połączenia URL nie jest najbardziej wydajna wdrożenie. A sposób, w jaki kończy połączenia, może czasami mieć negatywny wpływ na sieć. ”
Alan
46

Uwaga: klient HTTP Apache w pakiecie z Androidem jest teraz przestarzały na korzyść HttpURLConnection . Więcej informacji można znaleźć na blogu programistów Androida .

Dodaj <uses-permission android:name="android.permission.INTERNET" />do swojego manifestu.

Następnie pobierz stronę internetową w następujący sposób:

URL url = new URL("http://www.android.com/");
HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
try {
     InputStream in = new BufferedInputStream(urlConnection.getInputStream());
     readStream(in);
}
finally {
     urlConnection.disconnect();
}

Sugeruję również uruchomienie go w osobnym wątku:

class RequestTask extends AsyncTask<String, String, String>{

@Override
protected String doInBackground(String... uri) {
    String responseString = null;
    try {
        URL url = new URL(myurl);
        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        if(conn.getResponseCode() == HttpsURLConnection.HTTP_OK){
            // Do normal input or output stream reading
        }
        else {
            response = "FAILED"; // See documentation for more info on response handling
        }
    } catch (ClientProtocolException e) {
        //TODO Handle problems..
    } catch (IOException e) {
        //TODO Handle problems..
    }
    return responseString;
}

@Override
protected void onPostExecute(String result) {
    super.onPostExecute(result);
    //Do anything with response..
}
}

Więcej informacji na temat obsługi odpowiedzi i żądań POST można znaleźć w dokumentacji .

kevinc
źródło
1
@Semmix Jak to się dzieje? Pytanie dotyczyło „prostego żądania HTTP”, a mój kod właśnie to robi.
kevinc
1
Rozumiem, że twój pierwszy blok kodu jest kopiowany z dokumentów Android, ale człowiek to śmieci z próbki / dokumentu. readStreamnie jest nawet zdefiniowany.
Eugene K
@EugeneK Są, ale jest to prawdopodobnie najprostszy sposób na odpowiedź na to pytanie. Prawidłowe wykonanie żądania HTTP w Androidzie wymagałoby wyjaśnienia Retrofit i OkHttp. Wydaje mi się, że wprowadzałoby to w błąd bardziej niż zwykłe wydanie fragmentu kodu, który technicznie wykona proste żądanie HTTP, nawet jeśli jest źle skonstruowane.
kevinc
12

Najprostszym sposobem jest użycie biblioteki Android o nazwie Volley

Volley oferuje następujące korzyści:

Automatyczne planowanie żądań sieciowych. Wiele jednoczesnych połączeń sieciowych . Przejrzyste buforowanie odpowiedzi dysku i pamięci ze standardową spójnością pamięci podręcznej HTTP. Obsługa priorytetów żądań. Interfejs API żądania anulowania. Możesz anulować pojedyncze żądanie lub ustawić bloki lub zakresy żądań do anulowania. Łatwość dostosowywania, na przykład w celu ponownej próby i wycofania. Silne porządkowanie, które ułatwia prawidłowe wypełnienie interfejsu użytkownika danymi pobieranymi asynchronicznie z sieci. Narzędzia do debugowania i śledzenia.

Możesz wysłać żądanie http / https tak proste:

        // Instantiate the RequestQueue.
        RequestQueue queue = Volley.newRequestQueue(this);
        String url ="http://www.yourapi.com";
        JsonObjectRequest request = new JsonObjectRequest(url, null,
            new Response.Listener<JSONObject>() {
                @Override
                public void onResponse(JSONObject response) {
                    if (null != response) {
                         try {
                             //handle your response
                         } catch (JSONException e) {
                             e.printStackTrace();
                         }
                    }
                }
            }, new Response.ErrorListener() {

            @Override
            public void onErrorResponse(VolleyError error) {

            }
        });
        queue.add(request);

W takim przypadku nie musisz sam „uruchamiać się w tle” ani „używać pamięci podręcznej”, ponieważ wszystko to zostało już wykonane przez Volley.

Shao Wenbin
źródło
6
private String getToServer(String service) throws IOException {
    HttpGet httpget = new HttpGet(service);
    ResponseHandler<String> responseHandler = new BasicResponseHandler();
    return new DefaultHttpClient().execute(httpget, responseHandler);

}

pozdrowienia

Gabriel Gómez
źródło
4

Z wątkiem:

private class LoadingThread extends Thread {
    Handler handler;

    LoadingThread(Handler h) {
        handler = h;
    }
    @Override
    public void run() {
        Message m = handler.obtainMessage();
        try {
            BufferedReader in = 
                new BufferedReader(new InputStreamReader(url.openStream()));
            String page = "";
            String inLine;

            while ((inLine = in.readLine()) != null) {
                page += inLine;
            }

            in.close();
            Bundle b = new Bundle();
            b.putString("result", page);
            m.setData(b);
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }

        handler.sendMessage(m);
    }
}
Fredley
źródło
4

Zrobiłem to, aby usługa internetowa wymagała adresu URL za pomocą biblioteki Gson:

Klient:

public EstabelecimentoList getListaEstabelecimentoPorPromocao(){

        EstabelecimentoList estabelecimentoList  = new EstabelecimentoList();
        try{
            URL url = new URL("http://" +  Conexao.getSERVIDOR()+ "/cardapio.online/rest/recursos/busca_estabelecimento_promocao_android");
            HttpURLConnection con = (HttpURLConnection) url.openConnection();

            if (con.getResponseCode() != 200) {
                    throw new RuntimeException("HTTP error code : "+ con.getResponseCode());
            }

            BufferedReader br = new BufferedReader(new InputStreamReader((con.getInputStream())));
            estabelecimentoList = new Gson().fromJson(br, EstabelecimentoList.class);
            con.disconnect();

        } catch (IOException e) {
            e.printStackTrace();
        }
        return estabelecimentoList;
}
Deividson Calixto
źródło
4

Spójrz na tę niesamowitą nową bibliotekę, która jest dostępna za pośrednictwem gradle :)

build.gradle: compile 'com.apptakk.http_request:http-request:0.1.2'

Stosowanie:

new HttpRequestTask(
    new HttpRequest("http://httpbin.org/post", HttpRequest.POST, "{ \"some\": \"data\" }"),
    new HttpRequest.Handler() {
      @Override
      public void response(HttpResponse response) {
        if (response.code == 200) {
          Log.d(this.getClass().toString(), "Request successful!");
        } else {
          Log.e(this.getClass().toString(), "Request unsuccessful: " + response);
        }
      }
    }).execute();

https://github.com/erf/http-request

electronix384128
źródło
1
Wygląda jak każda inna biblioteka ...
Nick Gallimore
3

Użyj Volley'a jak wyżej. Dodaj następujące elementy do build.gradle (moduł: aplikacja)

implementation 'com.android.volley:volley:1.1.1'

Dodaj następujące do pliku AndroidManifest.xml:

<uses-permission android:name="android.permission.INTERNET" />

I dodaj następujące kod aktywności:

public void httpCall(String url) {

    RequestQueue queue = Volley.newRequestQueue(this);

    StringRequest stringRequest = new StringRequest(Request.Method.GET, url,
            new Response.Listener<String>() {
                @Override
                public void onResponse(String response) {
                    // enjoy your response
                }
            }, new Response.ErrorListener() {
                @Override
                public void onErrorResponse(VolleyError error) {
                    // enjoy your error status
                }
    });

    queue.add(stringRequest);
}

Zastępuje klienta HTTP i jest bardzo prosty.

Jarda Pavlíček
źródło
2

To jest nowy kod dla żądania HTTP Get / POST w Androidzie. HTTPClientjest osłabiony i może nie być dostępny tak jak w moim przypadku.

Najpierw dodaj dwie zależności w build.gradle:

compile 'org.apache.httpcomponents:httpcore:4.4.1'
compile 'org.apache.httpcomponents:httpclient:4.5'

Następnie wpisz ten kod ASyncTaskw doBackgroundmetodzie.

 URL url = new URL("http://localhost:8080/web/get?key=value");
 HttpURLConnection urlConnection = (HttpURLConnection)url.openConnection();
 urlConnection.setRequestMethod("GET");
 int statusCode = urlConnection.getResponseCode();
 if (statusCode ==  200) {
      InputStream it = new BufferedInputStream(urlConnection.getInputStream());
      InputStreamReader read = new InputStreamReader(it);
      BufferedReader buff = new BufferedReader(read);
      StringBuilder dta = new StringBuilder();
      String chunks ;
      while((chunks = buff.readLine()) != null)
      {
         dta.append(chunks);
      }
 }
 else
 {
     //Handle else
 }
Rahul Raina
źródło
Kod może zostać uznany za przestarzały, a apache nie jest już obsługiwany w interfejsie API platformy Android 28. W takim przypadku można włączyć wcześniejszą właściwość apache w pliku manifestu lub pliku Gradle na poziomie modułu. Zaleca się jednak korzystanie z biblioteki sieci OKHttp, Volley lub Retrofit.
Rahul Raina
1

Dla mnie najprostszym sposobem jest użycie biblioteki o nazwie Retrofit2

Musimy tylko stworzyć interfejs zawierający naszą metodę żądania, parametry, a także możemy utworzyć niestandardowy nagłówek dla każdego żądania:

    public interface MyService {

      @GET("users/{user}/repos")
      Call<List<Repo>> listRepos(@Path("user") String user);

      @GET("user")
      Call<UserDetails> getUserDetails(@Header("Authorization") String   credentials);

      @POST("users/new")
      Call<User> createUser(@Body User user);

      @FormUrlEncoded
      @POST("user/edit")
      Call<User> updateUser(@Field("first_name") String first, 
                            @Field("last_name") String last);

      @Multipart
      @PUT("user/photo")
      Call<User> updateUser(@Part("photo") RequestBody photo, 
                            @Part("description") RequestBody description);

      @Headers({
        "Accept: application/vnd.github.v3.full+json",
        "User-Agent: Retrofit-Sample-App"
      })
      @GET("users/{username}")
      Call<User> getUser(@Path("username") String username);    

    }

A najlepsze jest to, że możemy to zrobić asynchronicznie łatwo za pomocą metody kolejkowania

faruk
źródło
1

Ponieważ żadna z odpowiedzi nie opisywała sposobu wykonywania żądań za pomocą OkHttp , który jest obecnie bardzo popularnym klientem http dla Androida i Java, podam prosty przykład:

//get an instance of the client
OkHttpClient client = new OkHttpClient();

//add parameters
HttpUrl.Builder urlBuilder = HttpUrl.parse("https://www.example.com").newBuilder();
urlBuilder.addQueryParameter("query", "stack-overflow");


String url = urlBuilder.build().toString();

//build the request
Request request = new Request.Builder().url(url).build();

//execute
Response response = client.newCall(request).execute();

Niewątpliwą zaletą tej biblioteki jest to, że oddziela nas ona od pewnych szczegółów niskiego poziomu, zapewniając bardziej przyjazne i bezpieczne sposoby interakcji z nimi. Składnia jest również uproszczona i pozwala pisać niezły kod.

NiVeR
źródło