Сделать HTTP-запрос с Android


352

Я искал везде, но не смог найти ответ, есть ли способ сделать простой HTTP-запрос? Я хочу запросить страницу / скрипт PHP на одном из моих сайтов, но не хочу показывать эту страницу.

Если возможно, я даже хочу сделать это в фоновом режиме (в BroadcastReceiver)


Ответы:


477

ОБНОВИТЬ

Это очень старый ответ. Я определенно не буду рекомендовать клиент Apache больше. Вместо этого используйте либо:

Оригинальный ответ

Прежде всего, запросите разрешение на доступ к сети, добавьте в манифест следующее:

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

Тогда самый простой способ - использовать http-клиент Apache в комплекте с Android:

    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());
    }

Если вы хотите, чтобы он работал в отдельном потоке, я бы рекомендовал расширить 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..
    }
}

Затем вы можете сделать запрос по:

   new RequestTask().execute("http://stackoverflow.com");

11
Вот статья из официального блога разработчиков Android на AsyncTask: android-developers.blogspot.com/2010/07/…
Austyn Mahoney

77
для gingerbread или выше рекомендуется использовать HttpURLConnection над библиотекой apache, см. android-developers.blogspot.com/2011/09/… . Меньше нагрузки на батарею и лучшая производительность
Марти

8
responseString = out.toString () должен быть перед вызовом out.close (). На самом деле, вы, вероятно, должны иметь out.close () в блоке finally. Но в целом очень полезный ответ (+1), спасибо!
декабря

9
Начиная с Honeycomb (SDK 11), асинхронный подход - это путь. NetworkOnMainThreadException выкинут при попытке запустить запрос HTTP от основного потока.
msrxthr

2
Этот ответ довольно отлично. Но я бы посоветовал не использовать AsyncTasks для работы в сети. Они могут очень легко создавать утечки памяти (и фактически предоставленный пример действительно утечки) и не предоставляют все функции, которые можно ожидать для сетевых запросов. Подумайте об использовании RoboSpice для такого рода фоновых задач: github.com/octo-online/robospice
Snicolas

67

если у вас нет явной причины выбрать Apache HttpClient, вы должны предпочесть java.net.URLConnection. Вы можете найти множество примеров того, как использовать его в Интернете.

мы также улучшили документацию по Android со времени вашего первоначального поста: http://developer.android.com/reference/java/net/HttpURLConnection.html

и мы говорили о компромиссах на официальном блоге: http://android-developers.blogspot.com/2011/09/androids-http-clients.html


13
Почему использование Apache HttpClient не рекомендуется?
Тед

4
Мой соучастник подробно рассказал об этом в официальном блоге: android-developers.blogspot.com/2011/09/…
Эллиот Хьюз,

@ElliottHughes: я согласен на 100%. Нет никаких сомнений в том, что Apache httpclient предлагает простые методы и более абстрактное представление протокола, но нативное url-соединение java не менее полезно. Немного практического, его так же легко использовать, как httpclient, и он намного более портативен
Nitin Bansal

1
На самом деле, если вы посмотрите на видео клиентских приложений Google I / O 2010 - Android REST ( youtube.com/watch?v=xHXn3Kg2IQE 57min21sec), вы увидите, что Apache HttpClient - это наиболее рекомендуемый вариант. Я цитирую Вирджила Добьянски (Вирджила Добжански, разработчика программного обеспечения в Google, работающего в группе приложений Android): «Я бы просто посоветовал вам использовать HTTP-клиент Apache, поскольку он имеет более надежную реализацию. Тип соединения URL-адреса транзакции HTTP не самый эффективный реализация. И то, как он завершает соединения, иногда может оказать негативное влияние на сеть ».
Алан

46

Примечание. HTTP-клиент Apache, поставляемый в комплекте с Android, теперь устарел в пользу HttpURLConnection . Пожалуйста, смотрите блог разработчиков Android для более подробной информации.

Добавить <uses-permission android:name="android.permission.INTERNET" /> в свой манифест.

Затем вы получите веб-страницу следующим образом:

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();
}

Я также предлагаю запустить его в отдельном потоке:

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..
}
}

См. Документацию для получения дополнительной информации об обработке ответов и POST-запросах.


1
@ Семмикс Как так? Вопрос задан для «простого HTTP» запроса, и мой код делает именно это.
kevinc

1
Я понял, что ваш первый блок кода - это копия, вставленная из документов Android, но человек - это образец / документ. readStreamдаже не определено.
Евгений К

@EugeneK Да, но это, наверное, самый простой способ ответить на этот вопрос. Правильное выполнение HTTP-запроса в Android потребует объяснения Retrofit и OkHttp. Я думаю, что это запутает начинающих больше, чем просто раздача фрагмента, который технически сделает простой HTTP-запрос, даже если он плохо сконструирован.
Кевинк

12

Самый простой способ - использовать Android-библиотеку Volley.

Залп предлагает следующие преимущества:

Автоматическое планирование сетевых запросов. Несколько одновременных сетевых подключений . Прозрачное кэширование отклика диска и памяти со стандартной согласованностью HTTP-кэша Поддержка приоритизации запросов. API запроса отмены. Вы можете отменить один запрос или установить блоки или объемы запросов для отмены. Простота настройки, например, для повторных попыток и откатов. Строгий порядок, позволяющий легко заполнять ваш пользовательский интерфейс данными, извлекаемыми асинхронно из сети. Инструменты отладки и трассировки.

Вы можете отправить запрос http / https так просто:

        // 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);

В этом случае вам не нужно рассматривать «запуск в фоновом режиме» или «использование кэша» самостоятельно, поскольку все это уже было сделано Volley.


6
private String getToServer(String service) throws IOException {
    HttpGet httpget = new HttpGet(service);
    ResponseHandler<String> responseHandler = new BasicResponseHandler();
    return new DefaultHttpClient().execute(httpget, responseHandler);

}

С уважением


4

С резьбой:

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);
    }
}

4

Я сделал это для веб-службы, чтобы запросить URL, используя библиотеку Gson:

Клиент:

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;
}

4

Посмотрите на эту удивительную новую библиотеку, которая доступна через Gradle :)

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

Применение:

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


1
Похоже на любую другую библиотеку ...
Ник Галлимор

3

Используйте залп, как предложено выше. Добавьте следующее в build.gradle (Модуль: приложение)

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

Добавьте следующее в AndroidManifest.xml:

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

И добавьте следующий код активности:

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);
}

Он заменяет http-клиент и очень прост.


2

Это новый код для HTTP-запроса Get / POST в Android. HTTPClientограничен и может быть недоступен, как это было в моем случае.

Сначала добавьте две зависимости в build.gradle:

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

Затем напишите этот код ASyncTaskв doBackgroundметоде.

 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
 }

Код может устареть, и apache больше не поддерживается в Android Platform API 28. В этом случае вы можете включить устаревшее свойство apache в Manifest или в файле Gradle уровня модуля. Тем не менее, рекомендуется использовать сетевую библиотеку OKHttp, Volley или Retrofit.
Рахул Райна

1

Для меня самый простой способ использовать библиотеку под названием Retrofit2.

Нам просто нужно создать интерфейс, который содержит наш метод запроса, параметры, а также мы можем сделать собственный заголовок для каждого запроса:

    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);    

    }

И самое лучшее, мы можем сделать это асинхронно легко, используя метод enqueue


1

Поскольку ни один из ответов не описывает способ выполнения запросов с помощью OkHttp , который в настоящее время является очень популярным http-клиентом для Android и Java в целом, я приведу простой пример:

//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();

Очевидным преимуществом этой библиотеки является то, что она абстрагирует нас от некоторых низкоуровневых деталей, предоставляя более дружественные и безопасные способы взаимодействия с ними. Синтаксис также упрощен и позволяет писать хороший код.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.