Android中的PATCH动词(OkHttp,Volley,Retrofit ...)

4

我知道类似的问题已经被提出了几次,但是我似乎找不到解决简单问题的方法:PATCH动词。

所以我想问一下,在Android中使用OkHttp、Volley或Retrofit解决过PATCH问题的人,请分享你的方法(如果你用了不同的方法或库,请也分享一下)。

我已经花费了三天的时间查找,但是我找不到任何解决这个问题的方法。我只是尝试通过RESTful API对由.NET环境构建和运行的表进行部分更新。

我目前正在使用Volley,但每次运行应用程序时,都会出现异常错误,显示类似于“PATCH is not an allowed...”的信息,然后给出其他可以使用的动词数组,例如“POST、PUT、GET...”,这并没有什么帮助。

我已尝试像这样使用OkHttp

  JSONObject infoRequestBody = new JSONObject();


    try {


        infoRequestBody.put("Salutation", "MRrr.");
        infoRequestBody.put("FirstName", "Work Now!!");
        infoRequestBody.put("LastName", "Test from my mobile again!! Okhttp");
        infoRequestBody.put("MiddleName", "From the Mobile");
        infoRequestBody.put("NickName", null);
        infoRequestBody.put("BirthDate", "1978-11-23T00:00:00.000Z");
        infoRequestBody.put("CompanyName", null);
        // infoRequestBody.put("RegDate", "2015-01-18T23:39:13.873Z");
        infoRequestBody.put("Gender", "Male");
        infoRequestBody.put("IsPregnant", null);
        infoRequestBody.put("IsNursing", null);
        infoRequestBody.put("WorkPhone", null);
        infoRequestBody.put("WorkPhoneExt", null);
        infoRequestBody.put("PhoneNumber", null);
        infoRequestBody.put("Address", "My android app working!!");
        infoRequestBody.put("Address2", null);
        infoRequestBody.put("City", null);
        infoRequestBody.put("State", "CA");
        infoRequestBody.put("ZipCode", null);
        infoRequestBody.put("EmailAddress", "email@me.com");
        infoRequestBody.put("Occupation", null);
        infoRequestBody.put("CountryofResidence", "United States");
        infoRequestBody.put("CountryCode", "US");
        infoRequestBody.put("Ethnicity", "Asian");
        infoRequestBody.put("ModifiedDate", "2015-01-18T23:39:13.873Z");
        infoRequestBody.put("ModifiedBy", null);
        infoRequestBody.put("AcceptedTerms", true);
        infoRequestBody.put("AcceptedPrivacyPolicyCheckbox", false);
        infoRequestBody.put("AcceptedUnder18ConsentCheckbox", false);
    } catch (JSONException e1) {
        e1.printStackTrace();
    }
  RequestBody body = com.squareup.okhttp.RequestBody.create(JSON, infoRequestBody.toString());


    com.squareup.okhttp.Request.Builder request = new com.squareup.okhttp.Request.Builder()
             .url("APIUrls.PATIENT_INFO_URL")
            .addHeader("Content-Type", "application/json")
            .addHeader("Accept", "application/json")
            .addHeader("X-ZUMO-APPLICATION", APIUrls.ZUMO_KEY)
            .patch(body);



    client.newCall(request).enqueue(new Callback() {




        @Override
        public void onFailure(com.squareup.okhttp.Request request, IOException e) {


        }

        @Override
        public void onResponse(com.squareup.okhttp.Response response) throws IOException {

            if (!response.isSuccessful()) throw new IOException("Unexpected code" + response);

            Headers responseHeaders = response.headers();
            for (int i = 0; i < responseHeaders.size(); i++) {
                System.out.println(responseHeaders.name(i) + ": " + responseHeaders.value(i));
            }


            System.out.println(response.body().string());

        }
    });

但是client.newCall(request).enque(new Callback()...会报错,提示request不能应用于Request.Builder

顺便说一下,我是新手,对OkHttp不太了解。

无论如何,我已经努力许久仍然无法使PATCH起作用。也许我在瞎猜,但如果有人有任何建议或思路可以让`PATCH`起作用,我将永远感激你。

感谢你的帮助。


.patch()之后,您缺少对.build()的调用。您需要一个Request而不是一个Request.Builder - Jesse Wilson
你尝试过使用像Rested这样的REST客户端,或者许多Chrome扩展程序之一来确保您的服务器支持该请求吗? - Jacob Tabak
@JacobTabak,是的,我已经试过了。所有的都可以正常运行,但在安卓上尝试做同样的事情时出现问题。 - ito
3个回答

4
如 GreyBeardedGeek 所说,您可以使用 Retrofit + OkHttp 来实现此功能。
RestAdapter restAdapter = new RestAdapter.Builder()
                .setClient(new OkClient())
                .setEndpoint(ENDPOINT)
                .build();
MyPatchService patchService = restAdapter.create(MyPatchService.class);
UserUpdate updatedUser = new UserUpdate();
updatedUser.Salutation = "Mr";
updatedUser.FirstName = "Adam";
patchService.update(updatedUser);

而且,MyPatchService 看起来像这样:

public interface MyPatchService {
    @PATCH("/some/update/endpoint")
    User update(UserUpdate updatedObject);
}

您的 UserUpdate 类可能如下所示:

public class UserUpdate {
    public String Salutation;
    public String FirstName;
    public String LastName;
}

如果这样做,将发送的 JSON 将如下所示(请注意,LastName 未序列化,因为它从未设置 - 这可以通过向 Retrofit 提供自定义 Gson 实例来配置):
{
  "Salutation": "Mr",
  "FirstName": "Adam"
}

请求/响应对象是您在网络上传输的对象的普通Java表示形式。作为补丁请求,您可以将成员变量设置为IntegerBoolean等类型,如果未设置(即如果它们为null),则不会进行序列化。

感谢@Adam和@GreyBeardedGeek的回复。那么,我作为PATCH发送的JSON对象,它会被添加到restAdapter对象中吗?就像这样.setPatch(myJsonData),还是会在接口内部传递?抱歉问这些问题:我对OkHTTP和Retrofit还不熟悉。谢谢。 - ito
你只需创建请求/响应对象,然后编写接口以期望它们 - Retrofit会为您执行所有序列化和反序列化操作。我将进一步更新示例,但我建议好好阅读一下Retrofit的魔力 - Adam S
不,它是按照预期编写的 - 在这个例子中,服务器返回完整更新的对象(在本例中为“User”对象)。我不知道这是否是接受PATCH动词请求的端点的正确行为。 - Adam S
好的。我的端点将是/tables/PatientInformation/my-table-id-here,所以我认为我需要有类似这样的东西:public interface MyPatchService { @PATCH("/tables/PatientInformation/my-table-id-here"") User PatientInformation(UserUpdate updatedObject); }这是你想说的吗?如果这太初学者了,我非常抱歉。 - ito
1
谢谢@Adam。我会的。再次感谢,如果/当我让这个东西工作了,我一定会报告的 :) - ito
显示剩余2条评论

1
好的!感谢@Adam和所有为我提供帮助的人。非常感激。所以,这就是我最终采取的步骤(我几乎没有自定义@Adam发布的内容,当然那是正确的答案!):
首先,
我按照以下方式设置了我的UserUpdate类:
public class UserUpdate {
public String Salutation;
public String FirstName;
public String LastName;
public String MiddleName;
public String NickName;

     //Setters and Getters...
  public String getSalutation() {
      return Salutation;
  }

  public void setSalutation(String salutation) {
      Salutation = salutation;
  }

  public String getFirstName() {
     return FirstName;
  }

  public void setFirstName(String firstName) {
     FirstName = firstName;
  }

  public String getLastName() {
      return LastName;
  }

  public void setLastName(String lastName) {
      LastName = lastName;
  }

public String getMiddleName() {
    return MiddleName;
}

public void setMiddleName(String middleName) {
    MiddleName = middleName;
}

public String getNickName() {
    return NickName;
}

 public void setNickName(String nickName) {
     NickName = nickName;
 }


}

然后我设置了名为PatientInfoAPI接口类

public interface PatientInfoAPI{

@PATCH("/tables/PatientInformation/{id}")
  public void update(@Body UserUpdate updatedDAta, @Path("id") String  id,Callback<UserUpdate> response);


 }

注意我正在传递要“PATCH”的表格ID,最重要的是我有@Body注释,它允许我传递数据以进行更新而不会出错(我试过没有它并且出现了错误)

最后在我的主Activity中:

 RestAdapter restAdapter = new RestAdapter.Builder()
            .setClient(new OkClient()) //very important!!!
            .setEndpoint(ENDPOINT)//base url
            .setRequestInterceptor(new RequestInterceptor() {//You may not need to pass headers, but I do, so this is how it's done.
                @Override
                public void intercept(RequestFacade request) {

                    request.addHeader("Content-Type", "application/json");
                    request.addHeader("Accept", "application/json");
                    request.addHeader("X-ZUMO-APPLICATION", APIUrls.ZUMO_KEY);

                }
            })
            .build();
    PatientInfoAPI patchService = restAdapter.create(PatientInfoAPI.class);

    UserUpdate updatedUser = new UserUpdate();
    updatedUser.setSalutation("Sr.");
    updatedUser.setFirstName("YesSirtItsWorking");
    updatedUser.setLastName("Bays");
    updatedUser.setMiddleName("ComeOn Now man!");
    updatedUser.setNickName("");

     //Passing the tableId along with the data to update
     patchService.update(updatedUser, APIUrls.TABLE_ID, new retrofit.Callback<UserUpdate>() {
        @Override
        public void success(UserUpdate userUpdates, retrofit.client.Response response) {


            Log.v("All is good", "Nickname:" +  userUpdates.getNickName().toString() + " Email: " + userUpdates.getEmailAddress().toString());



        }

        @Override
        public void failure(RetrofitError error) {

            Log.v("All is good", error.getLocalizedMessage().toString());


        }
    });

我知道有些步骤可能不是很复杂,但对于那些刚开始使用Retrofit和OkHttp,并且最重要的是一直在努力让PATCH与Android配合工作的人来说,这是解决方案。再次感谢@Adam几乎引导我完成了整个过程。@Adam是真正回答了这个问题的人 - 我只是把它们都整合起来供那些可能需要看到完整画面的人使用。谢谢,@Adam!!! 你是一个摇滚明星!


1

1
截至2016年3月,我使用Retrofit 2.0.0-beta4成功地使用PATCH方法,无需像这里的答案中提到的任何其他库。 - ahmadalibaloch

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接