Игнорировать поля из объекта Java динамически при отправке в качестве JSON из Spring MVC
У меня есть класс модели, подобный этому, для спящего режима
@Entity
@Table(name = "user", catalog = "userdb")
@JsonIgnoreProperties(ignoreUnknown = true)
public class User implements java.io.Serializable {
private Integer userId;
private String userName;
private String emailId;
private String encryptedPwd;
private String createdBy;
private String updatedBy;
@Id
@GeneratedValue(strategy = IDENTITY)
@Column(name = "UserId", unique = true, nullable = false)
public Integer getUserId() {
return this.userId;
}
public void setUserId(Integer userId) {
this.userId = userId;
}
@Column(name = "UserName", length = 100)
public String getUserName() {
return this.userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
@Column(name = "EmailId", nullable = false, length = 45)
public String getEmailId() {
return this.emailId;
}
public void setEmailId(String emailId) {
this.emailId = emailId;
}
@Column(name = "EncryptedPwd", length = 100)
public String getEncryptedPwd() {
return this.encryptedPwd;
}
public void setEncryptedPwd(String encryptedPwd) {
this.encryptedPwd = encryptedPwd;
}
public void setCreatedBy(String createdBy) {
this.createdBy = createdBy;
}
@Column(name = "UpdatedBy", length = 100)
public String getUpdatedBy() {
return this.updatedBy;
}
public void setUpdatedBy(String updatedBy) {
this.updatedBy = updatedBy;
}
}
В Spring MVC-контроллере, используя DAO, я могу получить объект. и возвращается как объект JSON.
@Controller
public class UserController {
@Autowired
private UserService userService;
@RequestMapping(value = "/getUser/{userId}", method = RequestMethod.GET)
@ResponseBody
public User getUser(@PathVariable Integer userId) throws Exception {
User user = userService.get(userId);
user.setCreatedBy(null);
user.setUpdatedBy(null);
return user;
}
}
Просмотр части выполняется с помощью AngularJS, поэтому он получит JSON, как этот
{
"userId" :2,
"userName" : "john",
"emailId" : "[email protected]",
"encryptedPwd" : "Co7Fwd1fXYk=",
"createdBy" : null,
"updatedBy" : null
}
Если я не хочу устанавливать зашифрованный пароль, я установил это поле также как null.
Но я не хочу, как это, я не хочу отправлять все поля на клиентскую сторону. Если мне не нужен пароль, обновленный, созданный полями для отправки, мой результат JSON должен выглядеть как
{
"userId" :2,
"userName" : "john",
"emailId" : "[email protected]"
}
Список полей, которые я не хочу отправлять клиенту из другой таблицы базы данных. Поэтому он будет меняться в зависимости от пользователя, который зарегистрировался. Как я могу это сделать?
Надеюсь, у тебя есть мой вопрос.
Ответы
Ответ 1
Добавьте @JsonIgnoreProperties("fieldname")
к своему POJO.
Или вы можете использовать @JsonIgnore
перед именем поля, которое хотите игнорировать при десериализации JSON. Пример:
@JsonIgnore
@JsonProperty(value = "user_password")
public java.lang.String getUserPassword() {
return userPassword;
}
Пример GitHub
Ответ 2
Я знаю, что немного опаздываю на вечеринку, но на самом деле я столкнулся с этим еще несколько месяцев назад. Все доступные решения были не очень привлекательны для меня (mixins? Ugh!), Поэтому я закончил создание новой библиотеки, чтобы сделать этот процесс более чистым. Он доступен здесь, если кто-нибудь захочет попробовать: https://github.com/monitorjbl/spring-json-view.
Основное использование довольно просто, вы используете объект JsonView
в своих методах контроллера следующим образом:
import com.monitorjbl.json.JsonView;
import static com.monitorjbl.json.Match.match;
@RequestMapping(method = RequestMethod.GET, value = "/myObject")
@ResponseBody
public void getMyObjects() {
//get a list of the objects
List<MyObject> list = myObjectService.list();
//exclude expensive field
JsonView.with(list).onClass(MyObject.class, match().exclude("contains"));
}
Вы также можете использовать его вне Spring:
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import static com.monitorjbl.json.Match.match;
ObjectMapper mapper = new ObjectMapper();
SimpleModule module = new SimpleModule();
module.addSerializer(JsonView.class, new JsonViewSerializer());
mapper.registerModule(module);
mapper.writeValueAsString(JsonView.with(list)
.onClass(MyObject.class, match()
.exclude("contains"))
.onClass(MySmallObject.class, match()
.exclude("id"));
Ответ 3
Добавить @JsonInclude(JsonInclude.Include.NON_NULL)
(заставляет Jackson сериализовать нулевые значения) для класса, а также @JsonIgnore
в поле пароля.
Конечно, вы могли бы установить @JsonIgnore
на createdBy и updatedBy, если вы всегда хотите игнорировать, а не только в этом конкретном случае.
UPDATE
Если вы не хотите добавлять аннотацию к самому POJO, отличный вариант - аннотации Jackson Mixin. Ознакомьтесь с документацией
Ответ 4
Да, вы можете указать, какие поля сериализуются как ответ JSON, а какие игнорировать. Это то, что вам нужно сделать, чтобы реализовать динамическое игнорирование свойств.
1) Во-первых, вам нужно добавить @JsonFilter из com.fasterxml.jackson.annotation.JsonFilter в свой класс сущности как.
import com.fasterxml.jackson.annotation.JsonFilter;
@JsonFilter("SomeBeanFilter")
public class SomeBean {
private String field1;
private String field2;
private String field3;
// getters/setters
}
2) Затем в вашем контроллере вы должны добавить создание объекта MappingJacksonValue и установить на него фильтры, и, в конце концов, вы должны вернуть этот объект.
import java.util.Arrays;
import java.util.List;
import org.springframework.http.converter.json.MappingJacksonValue;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import com.fasterxml.jackson.databind.ser.FilterProvider;
import com.fasterxml.jackson.databind.ser.impl.SimpleBeanPropertyFilter;
import com.fasterxml.jackson.databind.ser.impl.SimpleFilterProvider;
@RestController
public class FilteringController {
// Here i want to ignore all properties except field1,field2.
@GetMapping("/ignoreProperties")
public MappingJacksonValue retrieveSomeBean() {
SomeBean someBean = new SomeBean("value1", "value2", "value3");
SimpleBeanPropertyFilter filter = SimpleBeanPropertyFilter.filterOutAllExcept("field1", "field2");
FilterProvider filters = new SimpleFilterProvider().addFilter("SomeBeanFilter", filter);
MappingJacksonValue mapping = new MappingJacksonValue(someBean);
mapping.setFilters(filters);
return mapping;
}
}
Вот что вы получите в ответ:
{
field1:"value1",
field2:"value2"
}
вместо этого:
{
field1:"value1",
field2:"value2",
field3:"value3"
}
Здесь вы можете видеть, что он игнорирует другие свойства (в данном случае field3) в ответ, за исключением свойства field1 и field2.
Надеюсь это поможет.
Ответ 5
Если бы я был вами и хотел бы сделать это, я бы не использовал свою сущность User на уровне контроллера. Вместо этого я создаю и использую UserDto (объект передачи данных) для связи с бизнес-уровнем (Service) и контроллером.
Вы можете использовать Apache BeanUtils (метод copyProperties) для копирования данных из сущности User в UserDto.
Ответ 6
Могу ли я сделать это динамически?
Создать класс представления:
public class View {
static class Public { }
static class ExtendedPublic extends Public { }
static class Internal extends ExtendedPublic { }
}
Аннотировать модель
@Document
public class User {
@Id
@JsonView(View.Public.class)
private String id;
@JsonView(View.Internal.class)
private String email;
@JsonView(View.Public.class)
private String name;
@JsonView(View.Public.class)
private Instant createdAt = Instant.now();
// getters/setters
}
Укажите класс представления в контроллере
@RequestMapping("/user/{email}")
public class UserController {
private final UserRepository userRepository;
@Autowired
UserController(UserRepository userRepository) {
this.userRepository = userRepository;
}
@RequestMapping(method = RequestMethod.GET)
@JsonView(View.Internal.class)
public @ResponseBody Optional<User> get(@PathVariable String email) {
return userRepository.findByEmail(email);
}
}
Пример данных:
{"id":"5aa2496df863482dc4da2067","name":"test","createdAt":"2018-03-10T09:35:31.050353800Z"}
Ответ 7
Мы можем сделать это, установив доступ к JsonProperty.Access.WRITE_ONLY
при объявлении свойства.
@JsonProperty( value = "password", access = JsonProperty.Access.WRITE_ONLY)
@SerializedName("password")
private String password;
Ответ 8
Я нашел решение для меня с Spring и jackson
Сначала укажите имя фильтра в сущности
@Entity
@Table(name = "SECTEUR")
@JsonFilter(ModelJsonFilters.SECTEUR_FILTER)
public class Secteur implements Serializable {
/** Serial UID */
private static final long serialVersionUID = 5697181222899184767L;
/**
* Unique ID
*/
@Id
@JsonView(View.SecteurWithoutChildrens.class)
@Column(name = "id")
@GeneratedValue(strategy = GenerationType.IDENTITY)
private long id;
@JsonView(View.SecteurWithoutChildrens.class)
@Column(name = "code", nullable = false, length = 35)
private String code;
/**
* Identifiant du secteur parent
*/
@JsonView(View.SecteurWithoutChildrens.class)
@Column(name = "id_parent")
private Long idParent;
@OneToMany(fetch = FetchType.LAZY)
@JoinColumn(name = "id_parent")
private List<Secteur> secteursEnfants = new ArrayList<>(0);
}
Затем вы можете увидеть, что константы фильтруют класс имен с использованием фильтра FilterProvider по умолчанию Spring
public class ModelJsonFilters {
public final static String SECTEUR_FILTER = "SecteurFilter";
public final static String APPLICATION_FILTER = "ApplicationFilter";
public final static String SERVICE_FILTER = "ServiceFilter";
public final static String UTILISATEUR_FILTER = "UtilisateurFilter";
public static SimpleFilterProvider getDefaultFilters() {
SimpleBeanPropertyFilter theFilter = SimpleBeanPropertyFilter.serializeAll();
return new SimpleFilterProvider().setDefaultFilter(theFilter);
}
}
Spring Конфигурация:
@EnableWebMvc
@Configuration
@ComponentScan(basePackages = "fr.sodebo")
public class ApiRootConfiguration extends WebMvcConfigurerAdapter {
@Autowired
private EntityManagerFactory entityManagerFactory;
/**
* config qui permet d'éviter les "Lazy loading Error" au moment de la
* conversion json par jackson pour les retours des services REST<br>
* on permet à jackson d'acceder à sessionFactory pour charger ce dont il a
* besoin
*/
@Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
super.configureMessageConverters(converters);
MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
ObjectMapper mapper = new ObjectMapper();
// config d'hibernate pour la conversion json
mapper.registerModule(getConfiguredHibernateModule());//
// inscrit les filtres json
subscribeFiltersInMapper(mapper);
// config du comportement de json views
mapper.configure(MapperFeature.DEFAULT_VIEW_INCLUSION, false);
converter.setObjectMapper(mapper);
converters.add(converter);
}
/**
* config d'hibernate pour la conversion json
*
* @return Hibernate5Module
*/
private Hibernate5Module getConfiguredHibernateModule() {
SessionFactory sessionFactory = entityManagerFactory.unwrap(SessionFactory.class);
Hibernate5Module module = new Hibernate5Module(sessionFactory);
module.configure(Hibernate5Module.Feature.FORCE_LAZY_LOADING, true);
return module;
}
/**
* inscrit les filtres json
*
* @param mapper
*/
private void subscribeFiltersInMapper(ObjectMapper mapper) {
mapper.setFilterProvider(ModelJsonFilters.getDefaultFilters());
}
}
В конце концов я могу указать конкретный фильтр в restConstoller, когда мне нужно....
@RequestMapping(value = "/{id}/droits/", method = RequestMethod.GET)
public MappingJacksonValue getListDroits(@PathVariable long id) {
LOGGER.debug("Get all droits of user with id {}", id);
List<Droit> droits = utilisateurService.findDroitsDeUtilisateur(id);
MappingJacksonValue value;
UtilisateurWithSecteurs utilisateurWithSecteurs = droitsUtilisateur.fillLists(droits).get(id);
value = new MappingJacksonValue(utilisateurWithSecteurs);
FilterProvider filters = ModelJsonFilters.getDefaultFilters().addFilter(ModelJsonFilters.SECTEUR_FILTER, SimpleBeanPropertyFilter.serializeAllExcept("secteursEnfants")).addFilter(ModelJsonFilters.APPLICATION_FILTER,
SimpleBeanPropertyFilter.serializeAllExcept("services"));
value.setFilters(filters);
return value;
}
Ответ 9
Я создал JsonUtil, который может быть использован для игнорирования полей во время выполнения при предоставлении ответа.
Пример использования: Первый аргумент должен быть любым классом POJO (Student), а ignoreFields - это поля, разделенные запятыми, которые вы хотите игнорировать в ответе.
Student st = new Student();
createJsonIgnoreFields(st,"firstname,age");
import java.util.logging.Logger;
import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.map.ObjectWriter;
import org.codehaus.jackson.map.ser.FilterProvider;
import org.codehaus.jackson.map.ser.impl.SimpleBeanPropertyFilter;
import org.codehaus.jackson.map.ser.impl.SimpleFilterProvider;
public class JsonUtil {
public static String createJsonIgnoreFields(Object object, String ignoreFields) {
try {
ObjectMapper mapper = new ObjectMapper();
mapper.getSerializationConfig().addMixInAnnotations(Object.class, JsonPropertyFilterMixIn.class);
String[] ignoreFieldsArray = ignoreFields.split(",");
FilterProvider filters = new SimpleFilterProvider()
.addFilter("filter properties by field names",
SimpleBeanPropertyFilter.serializeAllExcept(ignoreFieldsArray));
ObjectWriter writer = mapper.writer().withFilters(filters);
return writer.writeValueAsString(object);
} catch (Exception e) {
//handle exception here
}
return "";
}
public static String createJson(Object object) {
try {
ObjectMapper mapper = new ObjectMapper();
ObjectWriter writer = mapper.writer().withDefaultPrettyPrinter();
return writer.writeValueAsString(object);
}catch (Exception e) {
//handle exception here
}
return "";
}
}
Ответ 10
Я решил использовать только @JsonIgnore
как предложил @kryger. Таким образом, ваш геттер станет:
@JsonIgnore
public String getEncryptedPwd() {
return this.encryptedPwd;
}
Вы можете установить @JsonIgnore
для поля, установщика или получателя, как описано здесь.
И, если вы хотите защитить зашифрованный пароль только на стороне сериализации (например, когда вам нужно войти в систему ваших пользователей), добавьте эту аннотацию @JsonProperty
в свое поле:
@JsonProperty(access = Access.WRITE_ONLY)
private String encryptedPwd;
Больше информации здесь.
Ответ 11
Не создавал бы класс UserJsonResponse
и заполнял бы нужные поля чистым решением?
Возвращение непосредственно JSON кажется отличным решением, когда вы хотите вернуть все модели. В противном случае он просто становится беспорядочным.
В будущем, например, вы можете захотеть иметь поле JSON, которое не соответствует полю модели, а затем вы столкнулись с большими проблемами.
Ответ 12
Это чистый инструмент для ответа выше:
@GetMapping(value = "/my-url")
public @ResponseBody
MappingJacksonValue getMyBean() {
List<MyBean> myBeans = Service.findAll();
MappingJacksonValue mappingValue = MappingFilterUtils.applyFilter(myBeans, MappingFilterUtils.JsonFilterMode.EXCLUDE_FIELD_MODE, "MyFilterName", "myBiggerObject.mySmallerObject.mySmallestObject");
return mappingValue;
}
//AND THE UTILITY CLASS
public class MappingFilterUtils {
public enum JsonFilterMode {
INCLUDE_FIELD_MODE, EXCLUDE_FIELD_MODE
}
public static MappingJacksonValue applyFilter(Object object, final JsonFilterMode mode, final String filterName, final String... fields) {
if (fields == null || fields.length == 0) {
throw new IllegalArgumentException("You should pass at least one field");
}
return applyFilter(object, mode, filterName, new HashSet<>(Arrays.asList(fields)));
}
public static MappingJacksonValue applyFilter(Object object, final JsonFilterMode mode, final String filterName, final Set<String> fields) {
if (fields == null || fields.isEmpty()) {
throw new IllegalArgumentException("You should pass at least one field");
}
SimpleBeanPropertyFilter filter = null;
switch (mode) {
case EXCLUDE_FIELD_MODE:
filter = SimpleBeanPropertyFilter.serializeAllExcept(fields);
break;
case INCLUDE_FIELD_MODE:
filter = SimpleBeanPropertyFilter.filterOutAllExcept(fields);
break;
}
FilterProvider filters = new SimpleFilterProvider().addFilter(filterName, filter);
MappingJacksonValue mapping = new MappingJacksonValue(object);
mapping.setFilters(filters);
return mapping;
}
}
Ответ 13
Поместите @JsonIgnore
на поле или его получатель или создайте пользовательский dto
@JsonIgnore
private String encryptedPwd;
или, как упоминалось выше, ceekay
аннотируйте его с помощью @JsonProperty
где атрибут доступа установлен только для записи
@JsonProperty( value = "password", access = JsonProperty.Access.WRITE_ONLY)
private String encryptedPwd;
Ответ 14
В своем классе сущности добавьте @JsonInclude(JsonInclude.Include.NON_NULL)
для решения проблемы.
это будет выглядеть
@Entity
@JsonInclude(JsonInclude.Include.NON_NULL)