我的基本网址如下:
public static final String VENUES_BASE_URL = "https://api.foursquare.com/v2/venues";
我的其余网址是这样的:
"?ll=40.7,50.2&limit=50&radius=25000&v=20140909&venuePhotos=1&oauth_token=xxyyxx";
我的界面的第一个实现:
public interface Fourquare { @GET("/explore?ll={p1},{p2}&limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx") Response getVenues(@Path("p1") String param1,@Path("p2") String param2); }
然后提出这样的请求:
RestAdapter restAdapter = new RestAdapter.Builder() .setEndpoint(ConfigConstants.VENUES_BASE_URL) .build(); Fourquare fourquare = restAdapter.create(Fourquare.class); Response myResponse = fourquare.getVenues("50","75");
但是,上面给了我以下错误:
retrofit.RetrofitError: Fourquare.getVenues: URL query string "ll={p1},{p2}&limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx" must not have replace block.
第二个实现(在查看一些使用Query参数的SO响应之后.注意:一旦我找到了ll?参数调用,我将把令牌作为参数):
@GET("/explore&limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx") void getVenues(@Query("ll") String ll,Callback<String> cb);
通过这样的实际调用:
fourquare.getVenues("50,75",new Callback<String>() { @Override public void success(String s,Response response) { Log.d(TAG,"Successful run!"); } @Override public void failure(RetrofitError error) { Log.d(TAG,"Failed run!"); } });
通过上面的实现,fail()方法总是被调用,所以我的代码仍然有问题.有人可以就实施此调用的正确方法提出一些建议吗?我最确定问题是“ll?”参数.
更新:
转向记录后,这是我从Retrofit获得的最终网址:
https://api.foursquare.com/v2/venues/explore&limit=50&radius=25000&v=20140909&venuePhotos=1&oauth_token=xxyyxx?ll=30.26%2C-97.74
看起来Foursquare服务器不喜欢url末尾的?ll参数,它必须在../v2/venues/explore之后显式放置,因为当通过浏览器放置请求时,它工作正常.
第3次实施(2014年9月17日)通过colriot建议,我能够解决我之前实施的400响应代码.我仍然遇到GSON的速度问题,所以寻找有关如何解决这个问题的建议.具体来说,与Volley相比,我的Retrofit实现需要更长的时间来显示我的结果,所以我想知道是否有更好的方法来实现回调.
Foursquare界面
public interface Fourquare { @GET("/explore?limit=50&radius=25000&v=20140909&venuePhotos=1&oauth_token=xxyyxx") void getVenues(@Query("ll") String ll,Callback<Object> cb); }
RestAdapter调用
RestAdapter restAdapter = new RestAdapter.Builder() .setEndpoint(ConfigConstants.VENUES_BASE_URL) .build(); Foursquare foursquare = restAdapter.create(Foursquare.class); foursquare.getVenues("30.26,-97.74",new Callback<Object>() { @Override public void success(Object o,"Success!"); // Parse response GsonBuilder gsonBuilder = new GsonBuilder(); Gson gson = gsonBuilder.create(); JsonParser parser = new JsonParser(); String response2 = gson.toJson(o); JsonObject data = parser.parse(response2).getAsJsonObject(); // Populate data model MetaResponse MetaResponse = gson.fromJson(data.get("Meta"),MetaResponse.class); VenuesExploreResponse myResponse = gson.fromJson(data.get("response"),VenuesExploreResponse.class); // Store results from myResponse in List } @Override public void failure(RetrofitError error) { Log.d(TAG,"Failures!"); } });
上述回调实现的当前问题是它需要比使用Volley更长(约1秒)来解析并显示结果. GsonBuilder / Gson / JsonParser块与我的Volley onResponse(String response)方法完全相同,除了那个中间的“response2”对象,所以大多数肯定这个中间/额外步骤是瓶颈.我正在寻找有关如何更好地实现Gson解析的建议.如果这可能更适合作为一个新的/单独的问题,我会这样做.
解决方法@H_502_54@
那么,正如我们已经发现问题所在? – > &安培;错字.但还有一件事要提到,Retrofit可以接受复杂的对象作为调用参数.然后将调用String.valueOf(object)将对象转换为Query / Path参数.
在您的情况下,您可以像这样定义自定义类:
class LatLng {
private double lat;
private double lng;
...
@Override public String toString() {
return String.format("%.1f,%.1f",lat,lng);
}
}
并重构您的端点方法:
@GET("/explore?limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx")
void getVenues(@Query("ll") LatLng ll,Callback<String> cb);
关于解析答案:
>永远不要在回调中创建Gson对象.它太重量了.使用您提供给RestAdapter的那个.
>为什么要混合JsonParser& GSON?对于基本相同的问题,它们是不同的工具.
>利用Retrofit的内置转换器机制;)
从单词到代码:
public class FoursquareResponse<T> {
private MetaResponse Meta;
private T response;
// getters
}
共:
@GET("/explore?limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx")
void getVenues(@Query("ll") LatLng ll,Callback<FoursquareResponse<VenuesExploreResponse>> cb);
...
foursquare.getVenues(LatLng.valueOf(30.26,-97.74),new Callback<FoursquareResponse<VenuesExploreResponse>>() {
@Override
public void success(FoursquareResponse<VenuesExploreResponse> r,Response response) {
MetaResponse MetaResponse = r.getMeta;
VenuesExploreResponse myResponse = r.getResponse();
// Store results from myResponse in List
}
@Override
public void failure(RetrofitError error) {
Log.d(TAG,"Failures!");
}
});
在您的情况下,您可以像这样定义自定义类:
class LatLng { private double lat; private double lng; ... @Override public String toString() { return String.format("%.1f,%.1f",lat,lng); } }
并重构您的端点方法:
@GET("/explore?limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx") void getVenues(@Query("ll") LatLng ll,Callback<String> cb);
关于解析答案:
>永远不要在回调中创建Gson对象.它太重量了.使用您提供给RestAdapter的那个.
>为什么要混合JsonParser& GSON?对于基本相同的问题,它们是不同的工具.
>利用Retrofit的内置转换器机制;)
从单词到代码:
public class FoursquareResponse<T> { private MetaResponse Meta; private T response; // getters }
共:
@GET("/explore?limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx") void getVenues(@Query("ll") LatLng ll,Callback<FoursquareResponse<VenuesExploreResponse>> cb); ... foursquare.getVenues(LatLng.valueOf(30.26,-97.74),new Callback<FoursquareResponse<VenuesExploreResponse>>() { @Override public void success(FoursquareResponse<VenuesExploreResponse> r,Response response) { MetaResponse MetaResponse = r.getMeta; VenuesExploreResponse myResponse = r.getResponse(); // Store results from myResponse in List } @Override public void failure(RetrofitError error) { Log.d(TAG,"Failures!"); } });