Volley Request에서 사용자 지정 헤더를 설정하는 방법


104

Volley 요청에 대해 사용자 정의 헤더를 어떻게 설정할 수 있습니까? 현재 POST 요청에 대한 본문 내용을 설정하는 방법이 있습니다. 간단한 GET 요청이 있지만 사용자 지정 헤더도 함께 전달해야합니다. JsonRequest 클래스가 어떻게 지원하는지 모르겠습니다. 전혀 가능합니까?


수락 된 답변을 변경하십시오. 현재 수락 된 답변이 올바르지 않습니다.
Esteban

답변:


119

public Map<String, String> getHeaders()정의 된Request 을 재정 의하여 원하는 HTTP 헤더를 반환하는 것 같습니다 .


4
@ JuanJoséMeleroGómez 링크가 끊어졌습니다. 404 찾을 수 없음
Milon

다음은 다른 예 (페이지 끝에있는 스 니펫)입니다. developer.android.com/training/volley/request-custom.html . 클래스 는 생성자에서 전달한 헤더 매개 변수 로 Map 을 반환하는 GsonRequest메서드 getHeaders()를 재정의 합니다.
Juan José Melero Gómez

157

getParams ()에 대한 대답은 POST 본문 데이터를 설정하는 것이지만 제목의 질문은 User-Agent와 같은 HTTP 헤더를 설정하는 방법을 물었습니다. CommonsWare가 말했듯이 getHeaders ()를 재정의합니다. 다음은 User-Agent를 'Nintendo Gameboy'로 설정하고 Accept-Language를 'fr'로 설정하는 샘플 코드입니다.

public void requestWithSomeHttpHeaders() {
    RequestQueue queue = Volley.newRequestQueue(this);
    String url = "http://www.somewebsite.com";
    StringRequest getRequest = new StringRequest(Request.Method.GET, url, 
        new Response.Listener<String>() 
        {
            @Override
            public void onResponse(String response) {
                // response
                Log.d("Response", response);
            }
        }, 
        new Response.ErrorListener() 
        {
            @Override
            public void onErrorResponse(VolleyError error) {
                // TODO Auto-generated method stub
                Log.d("ERROR","error => "+error.toString());
            }
        }
    ) {     
        @Override
        public Map<String, String> getHeaders() throws AuthFailureError { 
                Map<String, String>  params = new HashMap<String, String>();  
                params.put("User-Agent", "Nintendo Gameboy");  
                params.put("Accept-Language", "fr");

                return params;  
        }
    };
    queue.add(getRequest);

}

이 답변은 JSONObjectRequest에서도 작동합니까? JSONObjectRequest postRequest = new JSONObjectRequest ......이 작업을 수행하고 있고 getHeaders ()가 호출되지 않기 때문에 ... ??? 익명의 클래스를 만들고 메서드를 재정의하고 있음을 이해합니다. StringRequest 대신 JSONObjectRequest 만 사용하고 getHeaders ()가 호출되지 않습니다.
JDOaktown

getHeaders () 메소드에도 "Cookie"를 추가 할 수 있습니까? 게시 요청에서도 작동합니까?
Arnab 얀 배너 지

나머지 POST 데이터는 어디로 보내야합니까?
Fernando Torres

30

필요한 것은 URL에 정보를 추가하는 대신 데이터를 게시하는 것입니다.

public Request post(String url, String username, String password, 
      Listener listener, ErrorListener errorListener) {
  JSONObject params = new JSONObject();
  params.put("user", username);
  params.put("pass", password);
  Request req = new Request(
     Method.POST,
     url,
     params.toString(),
     listener,
     errorListener
  );

  return req;
}

수행하려는 작업이 요청의 헤더를 편집하는 것이라면 수행 할 작업입니다.

// could be any class that implements Map
Map<String, String> mHeaders = new ArrayMap<String, String>();
mHeaders.put("user", USER);
mHeaders.put("pass", PASSWORD);
Request req = new Request(url, postBody, listener, errorListener) {
  public Map<String, String> getHeaders() {
    return mHeaders;
  }
}

44
나는 이것이 질문이 전혀 요청한 것이 아니기 때문에 반대 투표했습니다. 이는 User-Agent와 같은 사용자 지정 HTTP 헤더를 설정하는 것이 아니라 POST 콘텐츠를 설정하기위한 것입니다. Zh. Atanasov와 CommonsWare가 getHeaders정답입니다.
georgiecasey

4
사용자가 요청한 내용이 아니기 때문에이 답변을 거절했습니다.
Dharmendra Pratap Singh 2014

1
이것은 헤더가 아닌 콘텐츠 매개 변수를 추가하기위한 것입니다. 정말 허용 대답 안
후안 코르테스

1
제 질문 좀 봐주 시겠어요? 이것과 비슷하지만 제대로 가져 오지 못했습니다. stackoverflow.com/a/37948318
X09

18

솔루션을 볼 수 있습니다 . 쿠키를 가져오고 설정하는 방법을 보여 주지만 쿠키는 요청 / 응답의 헤더 중 하나 일뿐입니다. Volley의 * Request 클래스 중 하나를 재정의하고 필요한 헤더를getHeaders()


링크 된 소스는 다음과 같습니다.

public class StringRequest extends com.android.volley.toolbox.StringRequest {

private final Map<String, String> _params;

/**
 * @param method
 * @param url
 * @param params
 *            A {@link HashMap} to post with the request. Null is allowed
 *            and indicates no parameters will be posted along with request.
 * @param listener
 * @param errorListener
 */
public StringRequest(int method, String url, Map<String, String> params, Listener<String> listener,
        ErrorListener errorListener) {
    super(method, url, listener, errorListener);

    _params = params;
}

@Override
protected Map<String, String> getParams() {
    return _params;
}

/* (non-Javadoc)
 * @see com.android.volley.toolbox.StringRequest#parseNetworkResponse(com.android.volley.NetworkResponse)
 */
@Override
protected Response<String> parseNetworkResponse(NetworkResponse response) {
    // since we don't know which of the two underlying network vehicles
    // will Volley use, we have to handle and store session cookies manually
    MyApp.get().checkSessionCookie(response.headers);

    return super.parseNetworkResponse(response);
}

/* (non-Javadoc)
 * @see com.android.volley.Request#getHeaders()
 */
@Override
public Map<String, String> getHeaders() throws AuthFailureError {
    Map<String, String> headers = super.getHeaders();

    if (headers == null
            || headers.equals(Collections.emptyMap())) {
        headers = new HashMap<String, String>();
    }

    MyApp.get().addSessionCookie(headers);

    return headers;
}

}

그리고 MyApp 클래스 :

public class MyApp extends Application {
    private static final String SET_COOKIE_KEY = "Set-Cookie";
    private static final String COOKIE_KEY = "Cookie";
    private static final String SESSION_COOKIE = "sessionid";

    private static MyApp _instance;
    private RequestQueue _requestQueue;
    private SharedPreferences _preferences;

    public static MyApp get() {
        return _instance;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        _instance = this;
            _preferences = PreferenceManager.getDefaultSharedPreferences(this);
        _requestQueue = Volley.newRequestQueue(this);
    }

    public RequestQueue getRequestQueue() {
        return _requestQueue;
    }


    /**
     * Checks the response headers for session cookie and saves it
     * if it finds it.
     * @param headers Response Headers.
     */
    public final void checkSessionCookie(Map<String, String> headers) {
        if (headers.containsKey(SET_COOKIE_KEY)
                && headers.get(SET_COOKIE_KEY).startsWith(SESSION_COOKIE)) {
                String cookie = headers.get(SET_COOKIE_KEY);
                if (cookie.length() > 0) {
                    String[] splitCookie = cookie.split(";");
                    String[] splitSessionId = splitCookie[0].split("=");
                    cookie = splitSessionId[1];
                    Editor prefEditor = _preferences.edit();
                    prefEditor.putString(SESSION_COOKIE, cookie);
                    prefEditor.commit();
                }
            }
    }

    /**
     * Adds session cookie to headers if exists.
     * @param headers
     */
    public final void addSessionCookie(Map<String, String> headers) {
        String sessionId = _preferences.getString(SESSION_COOKIE, "");
        if (sessionId.length() > 0) {
            StringBuilder builder = new StringBuilder();
            builder.append(SESSION_COOKIE);
            builder.append("=");
            builder.append(sessionId);
            if (headers.containsKey(COOKIE_KEY)) {
                builder.append("; ");
                builder.append(headers.get(COOKIE_KEY));
            }
            headers.put(COOKIE_KEY, builder.toString());
        }
    }

}

7

Kotlin에서

다음 과 같이 getHeaders () 메서드 를 재정의해야합니다 .

val volleyEnrollRequest = object : JsonObjectRequest(GET_POST_PARAM, TARGET_URL, PAYLOAD_BODY_IF_YOU_WISH,
            Response.Listener {
                // Success Part  
            },

            Response.ErrorListener {
                // Failure Part
            }
        ) {
            // Providing Request Headers

            override fun getHeaders(): Map<String, String> {
               // Create HashMap of your Headers as the example provided below

                val headers = HashMap<String, String>()
                headers["Content-Type"] = "application/json"
                headers["app_id"] = APP_ID
                headers["app_key"] = API_KEY

                return headers
            }
        }

1
Kotlin에서 헤더를 재정의하기 위해 찾을 수있는 유일한 리소스입니다. 감사합니다!
Mathew Sonke

@MathewSonke 나는 당신이 형제를 느낀다. Btw는 Android에서 네트워킹을 위해 Retrofit을 시도합니다.
devDeejay

6

이 문제에 대한 해결책도 찾고 있습니다. 여기에서 뭔가를 참조하십시오 : http://developer.android.com/training/volley/request.html

ImageLoader 대신 ImageRequest를 직접 사용하는 것이 좋은 생각입니까? ImageLoader는 어쨌든 내부적으로 사용하는 것 같습니다. ImageLoader의 캐시 지원 외에 중요한 것이 누락됩니까?

ImageView mImageView;
String url = "http://i.imgur.com/7spzG.png";
mImageView = (ImageView) findViewById(R.id.myImage);
...

// Retrieves an image specified by the URL, displays it in the UI.
mRequestQueue = Volley.newRequestQueue(context);;
ImageRequest request = new ImageRequest(url,
  new Response.Listener() {
      @Override
      public void onResponse(Bitmap bitmap) {
          mImageView.setImageBitmap(bitmap);
      }
  }, 0, 0, null,
  new Response.ErrorListener() {
      public void onErrorResponse(VolleyError error) {
          mImageView.setImageResource(R.drawable.image_load_error);
      }
  })   {
    @Override
    public Map<String, String> getHeaders() throws AuthFailureError {
        Map<String, String> params = new Map<String, String>();
        params.put("User-Agent", "one");
        params.put("header22", "two");

        return params;
    };
mRequestQueue.add(request);

단순히 '-1'을 제공하는 대신 질문이 잘못되었다고 생각하거나 요청한 솔루션에 문제가 있다고 생각하는 이유를 지적 할 수 있다면 훨씬 감사하고 도움이 될 것입니다.
lannyf 2014

1
지도는 추상적입니다. HashMap이어야합니다
superuserdo

4

이 시도

{
    @Override
       public Map<String, String> getHeaders() throws AuthFailureError {
           String bearer = "Bearer ".concat(token);
            Map<String, String> headersSys = super.getHeaders();
            Map<String, String> headers = new HashMap<String, String>();
            headersSys.remove("Authorization");
            headers.put("Authorization", bearer);
            headers.putAll(headersSys);
            return headers;
       }
};

4

다음과 같이 StringRequest를 확장하고 getHeaders () 메서드를 재정의하는 사용자 정의 Request 클래스를 만들 수 있습니다.

public class CustomVolleyRequest extends StringRequest {

    public CustomVolleyRequest(int method, String url,
                           Response.Listener<String> listener,
                           Response.ErrorListener errorListener) {
        super(method, url, listener, errorListener);
    }

    @Override
    public Map<String, String> getHeaders() throws AuthFailureError {
        Map<String, String> headers = new HashMap<>();
        headers.put("key1","value1");
        headers.put("key2","value2");
        return headers;
    }
}

1
public class CustomJsonObjectRequest  extends JsonObjectRequest
{
    public CustomJsonObjectRequest(int method, String url, JSONObject jsonRequest,Response.Listener listener, Response.ErrorListener errorListener)
    {
        super(method, url, jsonRequest, listener, errorListener);
    }


@Override
public Map getHeaders() throws AuthFailureError {
    Map headers = new HashMap();
    headers.put("AppId", "xyz");

    return headers;
}

}

1

또한 다음과 관련하여 찾은 내용을 공유하고 싶습니다 Content-Type.

@Override
public Map<String, String> getHeaders() throws AuthFailureError {
Map<String, String> params = new HashMap<String, String>();
.
.
.
return params;
}

나는 추가해야했다 :

@Override
public String getBodyContentType() {
return /*(for exmaple)*/ "application/json";
}

이유를 묻지 말고, Content-Type설정을 제대로 할 수없는 다른 사람들에게 도움이 될 것이라고 생각 했습니다.


0

다음은 github 샘플에서 헤더를 설정하는 것입니다.

StringRequest myReq = new StringRequest(Method.POST,
                       "http://ave.bolyartech.com/params.php",
                        createMyReqSuccessListener(),
                        createMyReqErrorListener()) {

 protected Map<String, String> getParams() throws 
         com.android.volley.AuthFailureError {
                        Map<String, String> params = new HashMap<String, String>();
                        params.put("param1", num1);
                        params.put("param2", num2);
                        return params;
                    };
                };
                queue.add(myReq);

0

이 시도

 public void VolleyPostReqWithResponseListenerwithHeaders(String URL,final Map<String, String> params,final Map<String, String> headers,Response.Listener<String> responseListener) {


    String url = URL;

    Log.i("url:", ":" + url);
    StringRequest mStringRequest = new StringRequest(Request.Method.POST,
            url, responseListener, new Response.ErrorListener() {
        @Override
        public void onErrorResponse(VolleyError error) {
            // error


            //Log.d("Error.Response", error.getLocalizedMessage());
        }
    }){
        @Override
        protected Map<String, String> getParams() {
            return params;
        }

        @Override
        public Map<String, String> getHeaders() throws AuthFailureError {
            return headers;
        }
    };



    mStringRequest.setRetryPolicy(new DefaultRetryPolicy(
            60000,
            DefaultRetryPolicy.DEFAULT_MAX_RETRIES,
            DefaultRetryPolicy.DEFAULT_BACKOFF_MULT));

    mStringRequest.setShouldCache(true);



    //  dialog.show();
    SingletonRequestQueue.getInstance(context).addToRequestQueue(mStringRequest);
}

@Override public Map <String, String> getHeaders () throws AuthFailureError {return headers; }};
Osama Ibrahim

헤더는 어디에 있습니까?
Osama Ibrahim

당신이지도 <문자열, 문자열> 헤더를 호출 할 때 signutare에, 당신은 그것을 설정할 수 있습니다
Yigit하기 Yuksel에게

0

그것은 내 코드입니다. 잊지 마세요 = 개체 : 넣지 않으면 작동하지 않습니다.

val queue = Volley.newRequestQueue(this)
        val url = "http://35.237.133.137:8080/lamarrullaWS/rest/lamarrullaAPI"
        // Request a string response from the provided URL.
        val jsonObjectRequest = object: JsonObjectRequest(Request.Method.GET, url, null,
                Response.Listener { response ->
                    txtPrueba.text = "Response: %s".format(response.toString())
                },
                Response.ErrorListener { txtPrueba.text = "That didn't work!" }
        )
        {
            @Throws(AuthFailureError::class)
            override fun getHeaders(): Map<String, String> {
                val headers = HashMap<String, String>()
                headers.put("Content-Type", "application/json")
                return headers
            }
        }
        queue.add(jsonObjectRequest)
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.