Actualización y GET mediante parámetros

92

Estoy intentando enviar una solicitud a la API de Google GeoCode mediante Retrofit. La interfaz de servicio se ve así:

public interface FooService {    
    @GET("/maps/api/geocode/json?address={zipcode}&sensor=false")
    void getPositionByZip(@Path("zipcode") int zipcode, Callback<String> cb);
}

Cuando llamo al servicio:

OkHttpClient okHttpClient = new OkHttpClient();

RestAdapter restAdapter = new RestAdapter.Builder().setEndpoint(Constants.GOOGLE_GEOCODE_URL).setClient(new OkClient(okHttpClient)).build();

FooService service = restAdapter.create(FooService.class);

service.getPositionByZip(zipCode, new Callback<String>() {
    @Override public void success(String jsonResponse, Response response) {
       ...
    }
@Override public void failure(RetrofitError retrofitError) {
    }
});

Recibo el siguiente seguimiento de pila:

06-07 13:18:55.337: E/AndroidRuntime(3756): FATAL EXCEPTION: Retrofit-Idle
06-07 13:18:55.337: E/AndroidRuntime(3756): Process: com.marketplacehomes, PID: 3756
06-07 13:18:55.337: E/AndroidRuntime(3756): java.lang.IllegalArgumentException: FooService.getPositionByZip: URL query string "address={zipcode}&sensor=false" must not have replace block.
06-07 13:18:55.337: E/AndroidRuntime(3756):     at retrofit.RestMethodInfo.methodError(RestMethodInfo.java:120)
06-07 13:18:55.337: E/AndroidRuntime(3756):     at retrofit.RestMethodInfo.parsePath(RestMethodInfo.java:216)
06-07 13:18:55.337: E/AndroidRuntime(3756):     at retrofit.RestMethodInfo.parseMethodAnnotations(RestMethodInfo.java:162)
06-07 13:18:55.337: E/AndroidRuntime(3756):     at 

Eché un vistazo a la pregunta de StackOverflow: Retrofit: ¿múltiples parámetros de consulta en el comando @GET? pero no parecía aplicable.

Tomé el código casi literalmente de aquí: http://square.github.io/retrofit/, así que estoy un poco perdido para entender el problema.

Pensamientos

Perry Hoekstra
fuente

Respuestas:

154

AFAIK, {...}solo se puede usar como una ruta, no dentro de un parámetro de consulta. Prueba esto en su lugar:

public interface FooService {    

    @GET("/maps/api/geocode/json?sensor=false")
    void getPositionByZip(@Query("address") String address, Callback<String> cb);
}

Si tiene una cantidad desconocida de parámetros para pasar, puede hacer algo como esto:

public interface FooService {    

    @GET("/maps/api/geocode/json")
    @FormUrlEncoded
    void getPositionByZip(@FieldMap Map<String, String> params, Callback<String> cb);
}
Bart Kiers
fuente
¿Cómo se pueden agregar múltiples parámetros de consulta?
Gilberto Ibarra
2
@GilbertoIbarra err, agregando más:void getPositionByZip(@Query("address") String address, @Query("number") String number, Callback<String> cb);
Bart Kiers
9
FormUrlEncoded solo se puede especificar en métodos HTTP con cuerpo de solicitud (por ejemplo, @POST)
jiashie
8
Respuesta incorrecta aquí, @FormUrlEncoded no se puede usar con GET
mr.icetea
1
@FormUrlEncodedno funciona con @GETanotación
Black_Zerg
40

@QueryMap funcionó para mí en lugar de FieldMap

Si tiene varios parámetros GET, otra forma de pasarlos a su URL es a HashMap.

class YourActivity extends Activity {

private static final String BASEPATH = "http://www.example.com";

private interface API {
    @GET("/thing")
    void getMyThing(@QueryMap Map<String, String> params, new Callback<String> callback);
}

public void onCreate(Bundle savedInstanceState) {
   super.onCreate(savedInstanceState);
   setContentView(R.layout.your_layout);

   RestAdapter rest = new RestAdapter.Builder().setEndpoint(BASEPATH).build();
   API service = rest.create(API.class);

   Map<String, String> params = new HashMap<String, String>();
   params.put("key1", "val1");
   params.put("key2", "val2");
   // ... as much as you need.

   service.getMyThing(params, new Callback<String>() {
       // ... do some stuff here.
   });
}
}

La URL llamada será http://www.example.com/thing/?key1=val1&key2=val2

madhu527
fuente
4

También quería aclarar que si tiene que compilar parámetros de URL complejos, deberá compilarlos manualmente. es decir, si su consulta es example.com/?latlng=-37,147, en lugar de proporcionar los valores lat y lng individualmente, deberá construir la cadena latlng externamente, luego proporcionarla como parámetro, es decir:

public interface LocationService {    
    @GET("/example/")
    void getLocation(@Query(value="latlng", encoded=true) String latlng);
}

Tenga en cuenta que encoded=truees necesario, de lo contrario, la modificación codificará la coma en el parámetro de cadena. Uso:

String latlng = location.getLatitude() + "," + location.getLongitude();
service.getLocation(latlng);
La cosa
fuente
1

Ejemplo de trabajo completo en Kotlin , he reemplazado mis claves API con 1111 ...

        val apiService = API.getInstance().retrofit.create(MyApiEndpointInterface::class.java)
        val params = HashMap<String, String>()
        params["q"] =  "munich,de"
        params["APPID"] = "11111111111111111"

        val call = apiService.getWeather(params)

        call.enqueue(object : Callback<WeatherResponse> {
            override fun onFailure(call: Call<WeatherResponse>?, t: Throwable?) {
                Log.e("Error:::","Error "+t!!.message)
            }

            override fun onResponse(call: Call<WeatherResponse>?, response: Response<WeatherResponse>?) {
                if (response != null && response.isSuccessful && response.body() != null) {
                    Log.e("SUCCESS:::","Response "+ response.body()!!.main.temp)

                    temperature.setText(""+ response.body()!!.main.temp)

                }
            }

        })
Hitesh Sahu
fuente
1
¿Puede poner la definición del método getWeather (params) para mayor claridad?
supro_96