用Android发出HTTP请求


Answers:


477

更新

这是一个非常古老的答案。我绝对不会再推荐Apache的客户端。而是使用以下任一方法:

原始答案

首先,请求访问网络的权限,在清单中添加以下内容:

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

然后,最简单的方法是使用与Android捆绑在一起的Apache http客户端:

    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
这里是从的AsyncTask官方Android开发者博客的一篇文章:android-developers.blogspot.com/2010/07/...
Austyn马奥尼

77
对于姜饼或更高版本,其实际建议在apache库上使用HttpURLConnection,请参阅android-developers.blogspot.com/2011/09/…。它减少了对电池的负担,并具有更好的性能
-Marty

8
responseString = out.toString()必须在out.close()调用之前。实际上,您可能应该在finally块中包含out.close()。但是总体而言,非常有帮助的答案(+1),谢谢!
dcp 2012年

9
从Honeycomb(SDK 11)开始,异步方法是必经之路。一个 NetworkOnMainThreadException当您尝试运行在主线程的HTTP请求被抛出。
msrxthr

2
这个答案非常好。但是我建议不要使用AsyncTasks进行网络连接。它们很容易造成内存泄漏(实际上所提供的示例确实会泄漏),并且没有提供人们可以期望的所有网络请求功能。考虑将RoboSpice用于此类后台任务:github.com/octo-online/robospice
Snicolas 2012年

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?
Ted

4
我的同谋在官方博客上对此进行了详细介绍:android-developers.blogspot.com/2011/09/…–
Elliott Hughes

@ElliottHughes:我同意100%。无可否认,Apache httpclient提供了简便的方法和更抽象的协议视图,但是Java的本机urlconnection丝毫没有减少用处。带着几分动手,其易于使用的的HttpClient的,并且是方式更便携
尼廷·邦萨尔

1
实际上,如果您看一下Google I / O 2010视频-Android REST客户端应用程序(youtube.com/watch?v=xHXn3Kg2IQE 57min21sec),您会发现Apache HttpClient是最推荐的视频。我引用了Virgil Dobjanschi(谷歌的软件工程师,负责Android应用程序组)“我只建议您使用HTTP Apache客户端,因为它具有更强大的实现。HTTP事务的URL连接类型不是最有效的实施。而且它终止连接的方式有时会对网络产生不利影响。”
艾伦

46

注意:现在不建议使用与Android捆绑在一起的Apache HTTP Client,而推荐使用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
@Semmix如何?这个问题要求“一个简单的HTTP”请求,而我的代码正是这样做的。
kevinc

1
我知道您的第一个代码块是从Android文档中复制粘贴的,但实际上是那个示例/文档垃圾。readStream甚至没有定义。
尤金(Eugene K)

@EugeneK是的,但这可能是回答此问题的最简单方法。在Android中正确执行HTTP请求将涉及解释Retrofit和OkHttp。我认为,这不仅会使初学者感到困惑,还不只是发出一个从技术上讲会发出简单HTTP请求的代码段,即使该请求的构造不佳也是如此。
kevinc

12

最简单的方法是使用名为Volley的Android库

排球有以下好处:

自动安排网络请求。多个并发网络连接。具有标准HTTP缓存一致性的透明磁盘和内存响应缓存。支持请求优先级。取消请求API。您可以取消单个请求,也可以将请求的阻止或范围设置为取消。易于定制,例如重试和退避。强大的排序功能使您可以轻松地使用从网络异步获取的数据正确填充UI。调试和跟踪工具。

您可以发送一个简单的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

我使用Gson lib使Web服务能够在URL上重新查询:

客户:

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
似乎像其他所有图书馆一样……
Nick Gallimore

3

按照上面的建议使用Volley。将以下内容添加到build.gradle中(模块:app)

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

这是android中HTTP Get / POST请求的新代码。HTTPClient被贬低,可能无法使用,因为在我的情况下

首先在build.gradle中添加两个依赖项:

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

然后在ASyncTaskin 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
 }

该代码可能已被弃用,并且Android Platform API 28中不再支持apache。在这种情况下,您可以在清单或模块级Gradle文件中启用apache legacy属性。但是,建议使用OKHttp,Volley或Retrofit网络库。
Rahul Raina

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执行请求的方法,OkHttp是当今在Android和Java上非常流行的HTTP客户端,因此,我将提供一个简单的示例:

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

该库的明显优势在于,它使我们从一些底层细节中抽象出来,从而提供了更友好,更安全的方式与它们进行交互。语法也得到了简化,并允许编写漂亮的代码。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.