Skip to content
Advertisement

java.lang.StackOverflowError: null [Spring Boot, Hibernate]

I have two classes User.java and Address.java and there is a one-to-one bi-directional mapping between them.

But when I try to get the address using the User class I get an “java.lang.StackOverflowError: null” exception.

The same thing happens when I try to get the User from the Address class.

User.java

JavaScript

Address.java

JavaScript

MainController.java

JavaScript

I am using the test() function to put data in the database and it is working fine. database image

But when I call the fetch() function I am getting the following error

JavaScript

Updated MainController.java

JavaScript

Advertisement

Answer

TLDR: you aren’t actually saving anything anywhere, but it’s easy to fix. Here’s my code and my explanation:

MainController.java:

JavaScript

User.java:

JavaScript

Address.java:

JavaScript

AddressRepository.java:

JavaScript

UserRepository.java:

JavaScript

UserDAO.java:

JavaScript

A DAO has no connection to the database, it’s intent is what the acronym stands for, simply to transfer data, and that’s it. When you make a repository, you can stick your objects there by saving them in the repository. Notice that by extending the CrudRepository with correct generics, you don’t even need to implement the methods yourself. The save method actually saves the POJO, and returns the saved version, which is why I did user = userRepository.save(user), which may seem counterintuitive at first, but it simply helps ensure that everything is as you expect. If you then want to send the UserDAO object as a response, you can create it using the user object that is returned from the database, maybe something like:

JavaScript

Please take notice of what is happening inside the test method in MainController. First, we create the POJO User object and set its fields. Then we have to save it to the repository, it is only persisted after you call save method of the repository. Please note that the user object is saved again once it is updated with the address.

This is a very crude way to do things, it is best to create a service layer and do this there with the @Transactional annotation, which would mean that everything is rolled back in case something goes wrong inside a method annotated as @Transactional.

Also, using CascadeType.ALL may be not what you want, please refer to this answer.

Inside fetch method, I ensure that the user indeed exists, which is not guaranteed. To avoid 500 errors, it’s important to have a fallback mechanism for when something doesn’t work.

As a final side note, you shouldn’t be storing raw passwords like that, you should at least use hashing with salt and pepper, or use one of the many available libraries for implementing such functionality (although it can be quite fun getting down and dirty with the code itself). You should also consider how much information you are revealing when something does go wrong, as you don’t want to give away too much information which could be used to deanonimise a specific user, or even learn more about your code and the system architecture.

User contributions licensed under: CC BY-SA
6 People found this is helpful
Advertisement