Próbuję wysłać żądanie do Google GeoCode API przy użyciu funkcji Retrofit. Interfejs serwisowy wygląda następująco:
public interface FooService {
@GET("/maps/api/geocode/json?address={zipcode}&sensor=false")
void getPositionByZip(@Path("zipcode") int zipcode, Callback<String> cb);
}
Kiedy dzwonię do serwisu:
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) {
}
});
Otrzymuję następujący stacktrace:
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
Rzuciłem okiem na pytanie StackOverflow: Retrofit: wiele parametrów zapytania w poleceniu @GET? ale nie wydawało się to mieć zastosowania.
Wziąłem kod prawie dosłownie stąd: http://square.github.io/retrofit/, więc trochę mi brakuje, aby zrozumieć problem.
Myśli?
void getPositionByZip(@Query("address") String address, @Query("number") String number, Callback<String> cb);
@FormUrlEncoded
nie działa z@GET
adnotacjami@QueryMap
pracował dla mnie zamiastFieldMap
Jeśli masz kilka parametrów GET, innym sposobem przekazania ich do adresu URL jest plik
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. }); } }
Wywołany adres URL będzie miał postać http://www.example.com/thing/?key1=val1&key2=val2
źródło
Chciałem również wyjaśnić, że jeśli masz złożone parametry adresu URL do zbudowania, musisz je utworzyć ręcznie. tzn. jeśli twoje zapytanie to
example.com/?latlng=-37,147
, zamiast podawać wartości lat i lng osobno, będziesz musiał zbudować łańcuch latlng zewnętrznie, a następnie podać go jako parametr, tj .:public interface LocationService { @GET("/example/") void getLocation(@Query(value="latlng", encoded=true) String latlng); }
Zauważ, że
encoded=true
jest to konieczne, w przeciwnym razie retrofit zakoduje przecinek w parametrze ciągu. Stosowanie:String latlng = location.getLatitude() + "," + location.getLongitude(); service.getLocation(latlng);
źródło
Kompletny przykład roboczy w Kotlinie , zamieniłem klucze API na 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) } } })
źródło