我正在尝试重构我的代码以使用Retrofit(来自Volley)进行一些Foursquare API调用,但是没有找到一个正确的示例来说明如何指定一个查询参数,该参数有两个用逗号分隔的值.

我的基本网址如下:

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解析的建议.如果这可能更适合作为一个新的/单独的问题,我会这样做.

解决方法

那么,正如我们已经发现问题所在? – > &安培;错字.但还有一件事要提到,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!");
    }
});

android – Retrofit:如何在请求中指定逗号分隔的参数?的更多相关文章

  1. android – 如何使用GSON发布大型json文件解析的进度

    亲爱的Stackoverflowers,我目前正在从我的原始资源中解析一个大的json文件.我不得不逐行更改读取使用Reader对象和gson,以逃避内存不足异常.到现在为止还挺好.现在这一切都发生在异步任务中,我希望通过使用publishProgress()让用户通知某种加载屏幕的进度.这是我现在正在阅读文件的方式,但我不知道是否我可以从GSON或Reader对象获得任何类型的进度更新.任何帮助是极大的赞赏!解决方法您必须围绕InputStream编写包装器.像这样的东西应该工作:如何使用:

  2. android – IncompatibleClassChangeError com.google.gson.annotations.SerializedName.value

    当用户从Play商店更新应用程序时,我们在Samsung设备中收到IncompatibleClassChangeError.请检查下面的日志.代码如下:LoginResponse是我的POJO课程.将返回JSON字符串,这是服务器响应.解决方法看起来像三星问题..很多人不仅与gsonlib有同样的问题,而且还有其他库.我认为你不能做太多,只等三星开发人员解决这个问题.已经这个问题是在三星开发者论坛

  3. 使用Retrofit在Android中重新创建flask api调用

    我在服务器上有一个烧瓶app和api,它使用从终端发送的以下url我试图在Android上使用改造来重新创建它.我使用的是1.7版,因为这适用于此处未显示的一些遗留代码.这是应用程序类的相关部分和api类我现在只得到一般性错误,例如这是我的第一个烧瓶应用程序,我不完全确定如何调试所以任何帮助在这里也是赞赏.我也没有访问服务器日志更新为了尝试追踪问题,我编辑了服务器上的代码.如果我只是在api中返回

  4. android – 如何运行Travis-CI和Espresso Test

    我目前设置了Travis-CI,以便在我的Android设备的每个版本上运行gradleConnectedCheck任务并执行我的所有单元测试.我已经能够成功地设置它.我现在正在尝试用Espresso构建一些功能测试,我目前遇到很多困难,设置Travis的方式使我的espresso测试可以与Travis的模拟器交互.我如何设置Travis以使其模拟器的工作方式与我在本地工作站上使用的模拟器完全相同

  5. android – 如何使用Retrofit RX实现WebSocket

    我有一个使用RxRetrofit的项目结构.我想用它来实现WebSocket通信,任何想法我怎么能做到这一点?解决方法Retrofit尚不支持Web套接字.该功能预计将在2.1版本中发布.但是,JW一直致力于分支机构.您可以从hisbranch构建Retrofit并尝试它.或者可以等到2.1发布.这是Github上的相应issue.

  6. android – Gson中的RuntimeException解析JSON:无法调用受保护的java.lang.ClassLoader()而没有args

    我假设我可以通过单独保存位置对象中的关键字段来解决这个问题,但如果可能的话,保存位置对象会很好.我看到有一个ExclusionStrategy对象可用于排除字段,但我不确定是否可以/应该使用它来排除我位置内的额外内容…

  7. Android Twitter Fabric SDK与Google GSON发生冲突

    我在将TwitterFabricSDK集成到我的应用程序时遇到了困难.我按照Twittertutorial的步骤进行操作,但是当我尝试用gradle构建我的项目时,我得到了这个错误:我尝试从我的app模块lib文件夹中删除我的静态Gson库,之后一切都很顺利.删除从gradle模块依赖项添加twittersdk的行时也一样,所以我很确定这两者之间存在某种冲突,我正在寻求解决它.任何帮助将不胜感激!

  8. android – 使用改造下载图像文件

    解决方法问题是响应中的内容类型标头包含一个虚假的字符集:Retrofit看到了这一点,并推断响应是它可以记录的文本.您应该将问题报告给服务器的管理员.如果您将问题报告给GitHub上的Retrofit问题跟踪器,我们可能会从此问题中恢复而不是崩溃.

  9. android – 在Retrofit 2中上传文件

    我尝试了以下但是在响应时我得到500错误–帮助我在上面的屏幕截图中设计请求的界面…谢谢解决方法以下代码工作:)

  10. android – 带有@multipart的Retrofit @body有问题

    图像Multipart在类类型对象中.案例1.(我做过的)服务参数:那时我的改造API情况2.(我遇到问题)@Bodyclass我正在尝试这个APIJava方面我不知道为什么,但它返回错误,如:“@Bodyparameterscannotbeusedwithformormulti-partencoding”任何帮助,将不胜感激.解决方法简单来说,我这样做了:我改变

随机推荐

  1. bluetooth-lowenergy – Altbeacon库无法在Android 5.0上运行

    昨天我在Nexus4上获得了Android5.0的更新,并且altbeacon库停止了检测信标.似乎在监视和测距时,didEnterRegion和didRangeBeaconsInRegion都没有被调用.即使RadiusNetworks的Locate应用程序现在表现不同,一旦检测到信标的值,它们就不再得到更新,并且通常看起来好像信标超出了范围.我注意到的一点是,现在在logcat中出现以下行“B

  2. android – react-native动态更改响应者

    我正在使用react-native进行Android开发.我有一个视图,如果用户长按,我想显示一个可以拖动的动画视图.我可以使用PanResponder实现这一点,它工作正常.但我想要做的是当用户长按时,用户应该能够继续相同的触摸/按下并拖动新显示的Animated.View.如果您熟悉Google云端硬盘应用,则它具有类似的功能.当用户长按列表中的任何项目时,它会显示可拖动的项目.用户可以直接拖

  3. android – 是否有可能通过使用与最初使用的证书不同的证书对其进行签名来发布更新的应用程序

    是否可以通过使用与最初使用的证书不同的证书进行签名来发布Android应用程序的更新?我知道当我们尝试将这样的构建上传到市场时,它通常会给出错误消息.但有没有任何出路,比如将其标记为主要版本,指定市场中的某个地方?解决方法不,你不能这样做.证书是一种工具,可确保您是首次上传应用程序的人.所以总是备份密钥库!

  4. 如何检测Android中是否存在麦克风?

    ..所以我想在让用户访问语音输入功能之前检测麦克风是否存在.如何检测设备上是否有麦克风.谢谢.解决方法AndroidAPI参考:hasSystemFeature

  5. Android – 调用GONE然后VISIBLE使视图显示在错误的位置

    我有两个视图,A和B,视图A在视图B上方.当我以编程方式将视图A设置为GONE时,它将消失,并且它正下方的视图将转到视图A的位置.但是,当我再次将相同的视图设置为VISIBLE时,它会在视图B上显示.我不希望这样.我希望视图B回到原来的位置,这是我认为会发生的事情.我怎样才能做到这一点?编辑–代码}这里是XML:解决方法您可以尝试将两个视图放在RelativeLayout中并相对于彼此设置它们的位置.

  6. android – 获得一首歌的流派

    我如何阅读与歌曲相关的流派?我可以读这首歌,但是如何抓住这首歌的流派,它存放在哪里?解决方法检查此代码:

  7. android – 使用textShadow折叠工具栏

    我有一个折叠工具栏的问题,在展开状态我想在文本下面有一个模糊的阴影,我使用这段代码:用:我可以更改textColor,它可以工作,但阴影不起作用.我为阴影尝试了很多不同的值.是否可以为折叠文本投射阴影?

  8. android – 重用arm共享库

    我已经建立了armarm共享库.我有兴趣重用一个函数.我想调用该函数并获得返回值.有可能做这样的事吗?我没有任何头文件.我试过这个Android.mk,我把libtest.so放在/jni和/libs/armeabi,/lib/armeabi中.此时我的cpp文件编译,但现在是什么?我从objdump知道它的名字编辑:我试图用这个android.mk从hello-jni示例中添加prebuild库:它工作,但libtest.so相同的代码显示以下错误(启动时)libtest.so存在于libhello-j

  9. android – 为NumberPicker捕获键盘’Done’

    我有一个AlertDialog只有一些文本,一个NumberPicker,一个OK和一个取消.(我知道,这个对话框还没有做它应该保留暂停和恢复状态的事情.)我想在软键盘或其他IME上执行“完成”操作来关闭对话框,就像按下了“OK”一样,因为只有一个小部件可以编辑.看起来处理IME“Done”的最佳方法通常是在TextView上使用setonEditorActionListener.但我没有任何Te

  10. android – 想要在调用WebChromeClient#onCreateWindow时知道目标URL

    当我点击一个带有target=“_blank”属性的超链接时,会调用WebChromeClient#onCreateWindow,但我找不到新的窗口将打开的新方法?主页url是我唯一能知道的东西?我想根据目标网址更改应用行为.任何帮助表示赞赏,谢谢!

返回
顶部