J'essaie d'envoyer une demande à l'API Google GeoCode à l'aide de Retrofit. L'interface de service ressemble à ceci:
public interface FooService {
@GET("/maps/api/geocode/json?address={zipcode}&sensor=false")
void getPositionByZip(@Path("zipcode") int zipcode, Callback<String> cb);
}
Quand j'appelle le service:
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) {
}
});
Je reçois le stacktrace suivant:
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
J'ai jeté un coup d'oeil à la question de StackOverflow: Retrofit: paramètres de requête multiples dans la commande @GET? mais cela ne semblait pas applicable.
J'ai pris le code à peu près ici: http://square.github.io/retrofit/ donc je suis un peu perdant pour comprendre le problème.
Pensées?
Autant que je sache, {...}
ne peut être utilisé que comme chemin, pas dans un paramètre de requête. Essayez ceci à la place:
public interface FooService {
@GET("/maps/api/geocode/json?sensor=false")
void getPositionByZip(@Query("address") String address, Callback<String> cb);
}
Si vous avez une quantité inconnue de paramètres à transmettre, vous pouvez utiliser l'une des méthodes suivantes:
public interface FooService {
@GET("/maps/api/geocode/json")
@FormUrlEncoded
void getPositionByZip(@FieldMap Map<String, String> params, Callback<String> cb);
}
@QueryMap
a travaillé pour moi au lieu de FieldMap
Si vous avez plusieurs paramètres GET, une autre façon de les transférer dans votre URL est une 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.
});
}
}
L'URL appelée sera http://www.example.com/thing/?key1=val1&key2=val2
Exemple de travail complet dans Kotlin , j'ai remplacé mes clés d'API par 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)
}
}
})
Je voulais aussi préciser que si vous avez des paramètres d'URL complexes à construire, vous devrez les construire manuellement. Par exemple, si votre requête est example.com/?latlng=-37,147
, au lieu de fournir les valeurs lat et lng individuellement, vous devrez construire la chaîne latlng en externe, puis le fournir en tant que paramètre, à savoir:
public interface LocationService {
@GET("/example/")
void getLocation(@Query(value="latlng", encoded=true) String latlng);
}
Notez que encoded=true
est nécessaire, sinon retrofit encodera la virgule dans le paramètre de chaîne. Usage:
String latlng = location.getLatitude() + "," + location.getLongitude();
service.getLocation(latlng);