Сделать HTTP-запрос с Android
Я искал везде, но не смог найти ответ, есть ли способ сделать простой HTTP-запрос? Я хочу запросить страницу / скрипт PHP на одном из моих сайтов, но не хочу показывать эту страницу.
Если возможно, я даже хочу сделать это в фоновом режиме (в BroadcastReceiver)
11 ответов
ОБНОВИТЬ
Это очень старый ответ. Я определенно не буду рекомендовать клиент 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://stackru.com");
Если у вас нет явной причины выбрать 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
Примечание. 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-запросах.
Самый простой способ - использовать 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.
Используйте залп, как предложено выше. Добавьте следующее в 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-клиент, и это очень просто.
Поскольку ни в одном из ответов не описан способ выполнения запросов с помощью 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();
Явным преимуществом этой библиотеки является то, что она абстрагирует нас от некоторых низкоуровневых деталей, предоставляя более дружественные и безопасные способы взаимодействия с ними. Синтаксис также упрощен и позволяет писать хороший код.
private String getToServer(String service) throws IOException {
HttpGet httpget = new HttpGet(service);
ResponseHandler<String> responseHandler = new BasicResponseHandler();
return new DefaultHttpClient().execute(httpget, responseHandler);
}
С уважением
С резьбой:
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);
}
}
Посмотрите на эту удивительную новую библиотеку, которая доступна через 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();
Я сделал это для веб-службы, чтобы запросить 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;
}
Это новый код для 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
}
Для меня самый простой способ - использовать библиотеку 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