Вы правы, ожидается, что аннотированный параметр @RequestBody будет содержать все тело запроса и привязываться к одному объекту, поэтому вам, по сути, придется использовать свои параметры.
Если вам абсолютно нужен ваш подход, есть индивидуальная реализация, которую вы можете сделать:
Скажите, что это ваш json:
{
"str1": "test one",
"str2": "two test"
}
и вы хотите привязать его к двум параметрам здесь:
@RequestMapping(value = "/Test", method = RequestMethod.POST)
public boolean getTest(String str1, String str2)
Сначала определите настраиваемую аннотацию, скажем @JsonArg
, с путем JSON, например путем к нужной информации:
public boolean getTest(@JsonArg("/str1") String str1, @JsonArg("/str2") String str2)
Теперь напишите Custom HandlerMethodArgumentResolver, который использует JsonPath, определенный выше, для разрешения фактического аргумента:
import java.io.IOException;
import javax.servlet.http.HttpServletRequest;
import org.apache.commons.io.IOUtils;
import org.springframework.core.MethodParameter;
import org.springframework.http.server.ServletServerHttpRequest;
import org.springframework.web.bind.support.WebDataBinderFactory;
import org.springframework.web.context.request.NativeWebRequest;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.ModelAndViewContainer;
import com.jayway.jsonpath.JsonPath;
public class JsonPathArgumentResolver implements HandlerMethodArgumentResolver{
private static final String JSONBODYATTRIBUTE = "JSON_REQUEST_BODY";
@Override
public boolean supportsParameter(MethodParameter parameter) {
return parameter.hasParameterAnnotation(JsonArg.class);
}
@Override
public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
String body = getRequestBody(webRequest);
String val = JsonPath.read(body, parameter.getMethodAnnotation(JsonArg.class).value());
return val;
}
private String getRequestBody(NativeWebRequest webRequest){
HttpServletRequest servletRequest = webRequest.getNativeRequest(HttpServletRequest.class);
String jsonBody = (String) servletRequest.getAttribute(JSONBODYATTRIBUTE);
if (jsonBody==null){
try {
String body = IOUtils.toString(servletRequest.getInputStream());
servletRequest.setAttribute(JSONBODYATTRIBUTE, body);
return body;
} catch (IOException e) {
throw new RuntimeException(e);
}
}
return "";
}
}
Теперь просто зарегистрируйте это с помощью Spring MVC. Немного запутано, но это должно работать чисто.
@Target(ElementType.PARAMETER) @Retention(RetentionPolicy.RUNTIME) public @interface JsonArg { String value() default ""; }
Хотя это правда, что он
@RequestBody
должен соответствовать одному объекту, этот объект может быть aMap
, поэтому это дает вам хороший способ достичь того, чего вы пытаетесь достичь (нет необходимости писать одноразовый объект поддержки):Вы также можете привязаться к ObjectNode Джексона, если хотите получить полное дерево JSON:
источник
Map
объект для хранения любого количества объектов внутри него, но объект верхнего уровня по-прежнему должен быть только один, не может быть двух объектов верхнего уровня.Map<String, String>
: библиотеки документации API (swagger / springfox и т. Д.), Вероятно, не смогут проанализировать вашу схему запроса / ответа из вашего исходного кода.Вы можете перепутать аргумент post, используя переменные body и path для более простых типов данных:
источник
Для передачи нескольких объектов, параметров, переменных и т. Д. Вы можете сделать это динамически, используя ObjectNode из библиотеки Джексона в качестве параметра. Сделать это можно так:
Надеюсь, это поможет.
источник
@RequestParam
- параметрHTTP GET
или,POST
отправленный клиентом, отображение запроса - это сегмент URL-адреса, переменная которого:var1
&var2
являются параметрами запроса.{params}
отображение запроса. Вы можете позвонить в службу поддержки, например:http:/host/form/user
илиhttp:/host/form/firm
где используются фирма и пользовательPathvariable
.источник
Простое решение - создать класс полезной нагрузки с атрибутами str1 и str2:
И после того, как ты сможешь пройти
и тело вашего запроса:
источник
Вместо использования json вы можете сделать простую вещь.
Теперь в контроллере вам нужно сопоставить запрос ajax, как показано ниже:
Надеюсь, это тебе поможет.
источник
Я адаптировал решение Биджу:
В чем разница:
BR
источник
Параметр запроса существует как для GET, так и для POST, для Get он будет добавлен как строка запроса к URL-адресу, но для POST он находится в теле запроса
источник
Не уверен, где вы добавляете json, но если я сделаю это так с angular, он будет работать без requestBody: angluar:
Ява:
источник
Хорошо. Я предлагаю создать объект-значение (Vo), содержащий нужные вам поля. Код проще, мы не меняем работу Джексона и его еще проще понять. С уважением!
источник
Вы можете добиться того, чего хотите, используя
@RequestParam
. Для этого вам необходимо сделать следующее:required
параметра значение false, если вы хотите иметь возможность отправлять нулевое значение.Я знаю, это немного похоже на взлом, но он работает! ;)
источник
вы также можете
@RequestBody Map<String, String> params
использовать егоparams.get("key")
для получения значения параметраисточник
Вы также можете использовать MultiValue Map для хранения requestBody внутри. Вот пример для этого.
в отличие от аннотации @RequestBody при использовании карты для хранения тела запроса, которое нам нужно аннотировать с помощью @RequestParam
и отправьте пользователя в Json RequestBody
источник