我试图重构我的代码以使用Retrofit(来自Volley)进行一些Foursquare API调用,但是没有找到一个正确的示例来说明如何指定一个查询参数,该参数有2个值由a分隔逗号。
我的基本网址如下:
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()方法总是被调用,所以我的代码仍然有问题。有人可以就实施此调用的正确方法提出一些建议吗?我最确定的问题是&#34; ll?&#34;参数。
更新 转换日志后,这是我从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, Response response) {
Log.d(TAG, "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)方法完全相同,除了那个中间&#34; response2&#34;对象,所以绝对这个中间/额外步骤是瓶颈。我正在寻找有关如何更好地实现Gson解析的建议。如果这可能更适合作为一个新的/单独的问题,我会这样做。
答案 0 :(得分:11)
因此,我们已经发现问题出在?
- &gt; &
错字。但还有一件事要提到,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
&amp; Gson
?对于基本相同的问题,它们是不同的工具。从单词到代码:
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!");
}
});
答案 1 :(得分:0)
只需将您的逗号分隔的参数urlencode,逗号=%2
答案 2 :(得分:0)
尝试按有效顺序添加所有Query参数(固定和变量)。 我的意思是
@GET("/explore")
Response getVenues(@Query("ll") ll, @Query("limit") limit, @Query("radius") radius, @Query("v") v, @Query("venuePhotos") venuePhotos, @Query("oauth_token") oauth_token);
并使用固定参数作为常量
的函数进行包装调用答案 3 :(得分:0)
您可以使用@EncodedQuery
:
@GET("/explore&limit=50&radius=25000&v=20140905&venuePhotos=1&oauth_token=xxyyxx")
void getVenues(@EncodedQuery("ll") LatLng ll, Callback<String> cb);
答案 4 :(得分:0)
如果您正在使用kotlin,并且有var items: List<String>
应该作为查询参数。
使用此方法组成字符串:
fun itemsString() = items.joinToString(separator = ",")
您的网址查询应类似于:
@Query(value = "items", encoded = true) String items