[android] 매개 변수를 사용하여 개조 및 GET

Retrofit을 사용하여 Google GeoCode API에 요청을 보내려고합니다. 서비스 인터페이스는 다음과 같습니다.

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

서비스에 전화 할 때 :

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) {
    }
});

다음 스택 추적을받습니다.

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

StackOverflow 질문을 살펴 보았습니다. Retrofit : @GET 명령의 여러 쿼리 매개 변수? 그러나 그것은 적용 가능하지 않은 것 같습니다.

http://square.github.io/retrofit/ 에서 코드를 거의 그대로 가져 왔 으므로 문제를 이해하는 데 약간의 손실이 있습니다.

생각?



답변

AFAIK {...}는 쿼리 매개 변수 내부가 아닌 경로로만 사용할 수 있습니다. 대신 이것을 시도하십시오.

public interface FooService {

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

전달할 매개 변수의 양을 알 수없는 경우 다음과 같이 사용할 수 있습니다.

public interface FooService {

    @GET("/maps/api/geocode/json")
    @FormUrlEncoded
    void getPositionByZip(@FieldMap Map<String, String> params, Callback<String> cb);
}


답변

@QueryMap 대신 나를 위해 일했다 FieldMap

GET 매개 변수가 많은 경우 URL에 전달하는 또 다른 방법은 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.
   });
}
}

호출되는 URL은 http://www.example.com/thing/?key1=val1&key2=val2입니다.


답변

또한 빌드 할 복잡한 URL 매개 변수가있는 경우 수동으로 빌드해야한다는 점을 명확히하고 싶었습니다. 즉, 쿼리가 example.com/?latlng=-37,147인 경우 lat 및 lng 값을 개별적으로 제공하는 대신 latlng 문자열을 외부에서 빌드 한 다음 매개 변수로 제공해야합니다. 예 :

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

(가) 주 encoded=true, 그렇지 않으면 개조 문자열 매개 변수에 쉼표를 인코딩하는 것이 필요하다. 용법:

String latlng = location.getLatitude() + "," + location.getLongitude();
service.getLocation(latlng);


답변

Kotlin 에서 완전한 작업 예제 , API 키를 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)

                }
            }

        })


답변