序列化一个对象时没有找到序列化程序

时间:2018-03-19 06:49:14

标签: java json spring objectmapper

我试图将对象作为JSON返回。 使用/user/id端点,我想根据他的ID显示用户。 调用此controllerMethod时,我得到以下异常:

InvalidDefinitionException: No serializer found for class org.hibernate.proxy.pojo.javassist.JavassistLazyInitializer and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.sample.scrumboard.models.User_$$_jvsta02_1["handler"])

我的contollerClass看起来像这样:

@RestController
@RequestMapping(path="/user")
@JsonIgnoreProperties(ignoreUnknown = true)
public class UserRestController {

    private UserRepository repository;

    @Autowired
    public UserRestController(UserRepository repository){
        this.repository = repository;
    }

    @GetMapping(value = "/list")
    public List<User> getUsers(){
        return repository.findAll();
    }

    @GetMapping(value = "/{id}")
    public @ResponseBody User getUserById(@PathVariable Long id, User user){
            user = repository.getOne(id);
            return user;
    }
}

我检查了al字段是否有公共getter并尝试了@JSONIgnoreProperties的各种选项,但我找不到它。 将所有用户显示为JSONlist确实可以使用/user/list工作JSONlist。所以问题只在于尝试显示一个对象而不是对象列表时。 它从存储库中找到了用户,但它无法序列化该对象并将其放入屏幕。

User类本身如下所示:

@Entity
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "userId", nullable = false, updatable = false)
    private Long id;

    @NotNull
    @Size(min=2, max=20)
    private String firstName;

    @NotNull
    @Size(min=2, max=30)
    private String lastName;

    @NotNull
    @Size(min=2, max=20)
    private String userName;

    @NotNull
    @Size(min=2, max=30)
    private String passWord;

    @NotNull
    @Email
    private String email;

    //the mappedBy element must be used to specify the relationship field or property of the entity that is the owner of the relationship
    @OneToMany(mappedBy = "owner", cascade = CascadeType.ALL, fetch = FetchType.LAZY)
    @JsonIgnore
    private List<UserStory> userStoryList;

    public User() {
    }

    public User(String firstName, String lastName, String userName, String passWord, String email) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.userName = userName;
        this.passWord = passWord;
        this.email = email;
    }

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", firstName='" + firstName + '\'' +
                ", lastName='" + lastName + '\'' +
                ", userName='" + userName + '\'' +
                ", passWord='" + passWord + '\'' +
                ", email='" + email + '\'' +
                '}';
    }

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getFirstName() {
        return firstName;
    }

    public void setFirstName(String firstName) {
        this.firstName = firstName;
    }

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    public String getUserName() {
        return userName;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public String getPassWord() {
        return passWord;
    }

    public void setPassWord(String passWord) {
        this.passWord = passWord;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public List<UserStory> getUserStoryList() {
        return userStoryList;
    }

    public void setUserStoryList(List<UserStory> userStoryList) {
        this.userStoryList = userStoryList;
    }
}

如何显示从/user/id返回的用户?

解决方案?

如下所述,我使用Dto和ModelMapper使其工作。

我添加了

@Bean
public ModelMapper modelMapper(){
    return new ModelMapper();
}

ControllerMethod

@GetMapping(value = "/{id}")
public UserDTO getUserById(@PathVariable Long id, User user, ModelMapper modelMapper){
        user = repository.getOne(id);
        return modelMapper.map(user, UserDTO.class);
}

和UserDto

public class UserDTO {
    private Long id;
    private String firstName;
    private String lastName;
    private String userName;
    private String passWord;
    private String email;
    private List<UserStory> userStoryList;
    //getters and setters

现在我可以在屏幕上显示用户了。 我仍然想知道是否没有使用杰克逊和没有模型映射器和dto的解决方案?

6 个答案:

答案 0 :(得分:2)

使用您的实体(用户)通过REST公开用户数据可能不是一个好主意吗?你能为你的用户创建UserDTO来实现Serializable并通过REST发送这个DTO吗?在这种情况下,应该有必要将您从db检索到的User对象转换为UserDTO。

答案 1 :(得分:0)

注释@JsonIgnoreProperties不应该在UserRestController上,您需要序列化的是User,删除注释就可以了。杰克逊将帮助完成所有工作,如果您想忽略课程User上的某些字段,将@JsonIgnoreProperties移至用户类,并在页面上显示需要显示的@JsonProperty

答案 2 :(得分:0)

实现java.io.Serializable接口的orm类的可序列化

您正在使用@RestController,不需要@ResponseBody

此外,无需创建DTO类来转换响应json的ORM类。

OBS:对于双向关系,您将使用@JsonManagedReference,@ JsonBackReference

答案 3 :(得分:0)

请勿在控制器类中使用@JsonIgnoreProperties(ignoreUnknown = true)

在实体类中使用以下注释。那解决了那个问题。

@JsonIgnoreProperties({"hibernateLazyInitializer", "handler"})

答案 4 :(得分:0)

尝试将以下行添加到application.properties文件:

spring.jackson.serialization.fail-on-empty-beans=false

答案 5 :(得分:0)

spring repository.getOne() 和repository.findById() 是有区别的。使用 getOne() 您可以获得参考(代理),例如如果对象已经在同一个事务中被读取。使用 findById(),您总能按预期获得用户。