我的基本网址如下:
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之后显式放置,因为当通过浏览器放置请求时,它工作正常.
任何解决API的限制的解决方案?
第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解析的建议.如果这可能更适合作为一个新的/单独的问题,我会这样做.
解决方法
在您的情况下,您可以像这样定义自定义类:
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!");
}
});