如何从Android调用RESTful Web服务?

问题描述 投票:48回答:11

我使用Jersey Framework和Java在Netbean IDE中编写了一个REST Web服务。

对于用户需要提供用户名和密码的每个请求,我知道此身份验证不是最佳做法(使用curl命令,如:curl -u username:password -X PUT http://localhsot:8080/user)。

现在我想从Android类调用REST Web服务。

我该怎么办?

我有一个使用DefaultHttpClientCredentialUsernameAndPassword的Android类,但是当我在Eclipse中运行它时,有时我会遇到运行时异常或SDK异常。

java android web-services rest authorization
11个回答
19
投票

这是一个示例restclient类

public class RestClient
{
    public enum RequestMethod
    {
        GET,
        POST
    }
    public int responseCode=0;
    public String message;
    public String response;
    public void Execute(RequestMethod method,String url,ArrayList<NameValuePair> headers,ArrayList<NameValuePair> params) throws Exception
    {
        switch (method)
        {
            case GET:
            {
                // add parameters
                String combinedParams = "";
                if (params!=null)
                {
                    combinedParams += "?";
                    for (NameValuePair p : params)
                    {
                        String paramString = p.getName() + "=" + URLEncoder.encode(p.getValue(),"UTF-8");
                        if (combinedParams.length() > 1)
                            combinedParams += "&" + paramString;
                        else
                            combinedParams += paramString;
                    }
                }
                HttpGet request = new HttpGet(url + combinedParams);
                // add headers
                if (headers!=null)
                {
                    headers=addCommonHeaderField(headers);
                    for (NameValuePair h : headers)
                        request.addHeader(h.getName(), h.getValue());
                }
                executeRequest(request, url);
                break;
            }
            case POST:
            {
                HttpPost request = new HttpPost(url);
                // add headers
                if (headers!=null)
                {
                    headers=addCommonHeaderField(headers);
                    for (NameValuePair h : headers)
                        request.addHeader(h.getName(), h.getValue());
                }
                if (params!=null)
                    request.setEntity(new UrlEncodedFormEntity(params, HTTP.UTF_8));
                executeRequest(request, url);
                break;
            }
        }
    }
    private ArrayList<NameValuePair> addCommonHeaderField(ArrayList<NameValuePair> _header)
    {
        _header.add(new BasicNameValuePair("Content-Type","application/x-www-form-urlencoded"));
        return _header;
    }
    private void executeRequest(HttpUriRequest request, String url)
    {
        HttpClient client = new DefaultHttpClient();
        HttpResponse httpResponse;
        try
        {
            httpResponse = client.execute(request);
            responseCode = httpResponse.getStatusLine().getStatusCode();
            message = httpResponse.getStatusLine().getReasonPhrase();
            HttpEntity entity = httpResponse.getEntity();

            if (entity != null)
            {
                InputStream instream = entity.getContent();
                response = convertStreamToString(instream);
                instream.close();
            }
        }
        catch (Exception e)
        { }
    }

    private static String convertStreamToString(InputStream is)
    {
        BufferedReader reader = new BufferedReader(new InputStreamReader(is));
        StringBuilder sb = new StringBuilder();
        String line = null;
        try
        {
            while ((line = reader.readLine()) != null)
            {
                sb.append(line + "\n");
            }
            is.close();
        }
        catch (IOException e)
        { }
        return sb.toString();
    }
}

0
投票

这是我为简单的Web服务调用创建的库,

你可以通过添加一行gradle依赖来使用它 -

compile 'com.scantity.ScHttpLibrary:ScHttpLibrary:1.0.0'

这是使用的演示。

https://github.com/vishalchhodwani1992/httpLibrary


0
投票

有很多图书馆可以做到这一点

Retrofit取代了传统的AsyncTask,它有自己的后台任务,所以你真的不用担心它。相对于AsyncTask,性能也非常好。

在这里查看,

http://square.github.io/retrofit/

可以在这里找到Retrofit的完整示例,

https://futurestud.io/blog/retrofit-getting-started-and-android-client/

Volley也高效且易于使用。你可以看看这里:

https://github.com/mcxiaoke/android-volley

网上有很多关于如何使用它的资源:

http://www.androidhive.info/2014/05/android-working-with-volley-library-1

的AsyncTask

或者您应该使用实现AsyncTask然后覆盖方法doInTheBackground() - 您可以在其中实现REST调用。

然后,您可以使用onPostExecute()让UI线程处理上一步中返回的结果。

这个答案提供了一个如何实现AsyncTask的好例子。见AsyncTask Android example

Per Ruffles的评论如下,这是使用AsyncTask进行REST调用的更相关示例:http://alvinalexander.com/android/android-asynctask-http-client-rest-example-tutorial


14
投票

最近发现第三方库 - Square Retrofit可以很好地完成这项工作。


定义REST端点

public interface GitHubService {
   @GET("/users/{user}/repos")
   List<Repo> listRepos(@Path("user") String user,Callback<List<User>> cb);
}

获得具体服务

RestAdapter restAdapter = new RestAdapter.Builder()
    .setEndpoint("https://api.github.com")
    .build();
GitHubService service = restAdapter.create(GitHubService.class);

调用REST端点

List<Repo> repos = service.listRepos("octocat",new Callback<List<User>>() { 
    @Override
    public void failure(final RetrofitError error) {
        android.util.Log.i("example", "Error, body: " + error.getBody().toString());
    }
    @Override
    public void success(List<User> users, Response response) {
        // Do something with the List of Users object returned
        // you may populate your adapter here
    }
});

该库为您处理json序列化和deserailization。您也可以自定义序列化和反序列化。

Gson gson = new GsonBuilder()
    .setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES)
    .registerTypeAdapter(Date.class, new DateTypeAdapter())
    .create();

RestAdapter restAdapter = new RestAdapter.Builder()
    .setEndpoint("https://api.github.com")
    .setConverter(new GsonConverter(gson))
    .build();

6
投票

我用这个REST Client为我的机器人。这看起来很酷。卢克的好作品。

http://lukencode.com/2010/04/27/calling-web-services-in-android-using-httpclient/


6
投票

无论你做什么都停下来! :)

将RESTful客户端实现为SERVICE,并将密集型网络内容委托给与活动无关的组件:SERVICE。

观看这个富有洞察力的视频http://www.youtube.com/watch?v=xHXn3Kg2IQE,其中Virgil Dobjanschi正在解释他对这一挑战的态度......


1
投票

使用Spring for Android与RestTemplate https://spring.io/guides/gs/consuming-rest-android/

// The connection URL 
String url = "https://ajax.googleapis.com/ajax/" + 
    "services/search/web?v=1.0&q={query}";

// Create a new RestTemplate instance
RestTemplate restTemplate = new RestTemplate();

// Add the String message converter
restTemplate.getMessageConverters().add(new StringHttpMessageConverter());

// Make the HTTP GET request, marshaling the response to a String
String result = restTemplate.getForObject(url, String.class, "Android");

1
投票

我使用OkHttpClient来调用restful web服务。这很简单。

OkHttpClient httpClient = new OkHttpClient();
Request request = new Request.Builder()
                .url(url)
                .build();

Response response = httpClient.newCall(request).execute();
String body = response.body().string()

0
投票

什么后端?如果JAVA那么你可以使用REST与Java(JAX-RS)使用Jersey。

在Android端,您可以使用这个简单的RestClient来处理该REST服务。

对于双方的JSON < - >对象映射(Android,Java后端),您可以使用GSON。


0
投票

也许已经很晚了或者你之前已经使用过了,但是还有一个名为ksoap并且它非常棒......它还包括超时并且可以有效地解析任何基于SOAP的Web服务。我也做了一些改动以适合我的解析..查一查


0
投票

按照以下步骤在android中使用RestFul。

步骤1

创建一个android空白项目。

第2步

需要互联网访问权限。在AndroidManifest.xml文件中编写以下代码。

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

第三步:

需要在另一台服务器或同一台机器上运行的RestFul url。

第4步

创建一个将扩展AsyncTask的RestFul客户端。请参阅RestFulPost.java。

第五步:

为RestFul请求和响应创建DTO类。

restful post.Java

package javaant.com.consuming_restful.restclient;
import android.app.ProgressDialog;
import android.content.Context;
import android.os.AsyncTask;
import android.util.Log;
import com.google.gson.Gson;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
import java.util.Map;
import javaant.com.consuming_restful.util.Util;
/**
 * Created by Nirmal Dhara on 29-10-2015.
 */
public class RestFulPost extends AsyncTask<map, void,="" string=""> {
    RestFulResult restFulResult = null;
    ProgressDialog Asycdialog;
    String msg;
    String task;
    public RestFulPost(RestFulResult restFulResult, Context context, String msg,String task) {
        this.restFulResult = restFulResult;
        this.task=task;
        this.msg = msg;
        Asycdialog = new ProgressDialog(context);
    }
    @Override
    protected String doInBackground(Map... params) {
        String responseStr = null;
        Object dataMap = null;
        HttpPost httpost = new HttpPost(params[0].get("url").toString());

        try {
            dataMap = (Object) params[0].get("data");
            Gson gson = new Gson();
            Log.d("data  map", "data map------" + gson.toJson(dataMap));
            httpost.setEntity(new StringEntity(gson.toJson(dataMap)));
            httpost.setHeader("Accept", "application/json");
            httpost.setHeader("Content-type", "application/json");
            DefaultHttpClient httpclient= Util.getClient();
            HttpResponse response = httpclient.execute(httpost);
            int statusCode = response.getStatusLine().getStatusCode();
            Log.d("resonse code", "----------------" + statusCode);

            if (statusCode == 200)
                responseStr = EntityUtils.toString(response.getEntity());
            if (statusCode == 404) {
                responseStr = "{\n" +
                        "\"status\":\"fail\",\n" +
                        " \"data\":{\n" +
                        "\"ValidUser\":\"Service not available\",\n" +
                        "\"code\":\"404\"\n" +
                        "}\n" +
                        "}";
            }

        } catch (Exception e) {
            e.printStackTrace();
        }
        return responseStr;
    }
    @Override
    protected void onPreExecute() {
        Asycdialog.setMessage(msg);
        //show dialog
        Asycdialog.show();
        super.onPreExecute();
    }
    @Override
    protected void onPostExecute(String s) {
        Asycdialog.dismiss();
        restFulResult.onResfulResponse(s,task);
    }
}

有关更多详细信息和完整代码,请访问http://javaant.com/consume-a-restful-webservice-in-android/#.VwzbipN96Hs

© www.soinside.com 2019 - 2024. All rights reserved.