ModelMapper, mapping list of Entites to List of DTO objects
Asked Answered
A

5

30

I am writing simple blog web application using Spring MVC framework. I am willing to add DTO layer to my app.

I decided to use ModelMapper framework for conversion from Entity objects to DTO objects used in my views.

I have just one problem. On my main page, I am showing a list of posts on my blog. In my view, it's just list of Post (Entity) objects. I want to change it to pass a list of PostDTO objects to my view. Is there any way to map List of Post objects to List of PostDTO object with single method call? I was thinking about writing converter that will convert this but I am not sure it's a good way to do it.

Also, I am using Lists of Entities in few more places like administrative panel or comment below every Post on my page.

Link to code of my app on GitHub repository: repository

Anguish answered 21/12, 2017 at 17:8 Comment(1)
If you have lazy relation to initialize, use @transactional or follow the link to another answer.Disparagement
U
66

You can create util class:

public class ObjectMapperUtils {

    private static final ModelMapper modelMapper;

    /**
     * Model mapper property setting are specified in the following block.
     * Default property matching strategy is set to Strict see {@link MatchingStrategies}
     * Custom mappings are added using {@link ModelMapper#addMappings(PropertyMap)}
     */
    static {
        modelMapper = new ModelMapper();
        modelMapper.getConfiguration().setMatchingStrategy(MatchingStrategies.STRICT);
    }

    /**
     * Hide from public usage.
     */
    private ObjectMapperUtils() {
    }

    /**
     * <p>Note: outClass object must have default constructor with no arguments</p>
     *
     * @param <D>      type of result object.
     * @param <T>      type of source object to map from.
     * @param entity   entity that needs to be mapped.
     * @param outClass class of result object.
     * @return new object of <code>outClass</code> type.
     */
    public static <D, T> D map(final T entity, Class<D> outClass) {
        return modelMapper.map(entity, outClass);
    }

    /**
     * <p>Note: outClass object must have default constructor with no arguments</p>
     *
     * @param entityList list of entities that needs to be mapped
     * @param outCLass   class of result list element
     * @param <D>        type of objects in result list
     * @param <T>        type of entity in <code>entityList</code>
     * @return list of mapped object with <code><D></code> type.
     */
    public static <D, T> List<D> mapAll(final Collection<T> entityList, Class<D> outCLass) {
        return entityList.stream()
                .map(entity -> map(entity, outCLass))
                .collect(Collectors.toList());
    }

    /**
     * Maps {@code source} to {@code destination}.
     *
     * @param source      object to map from
     * @param destination object to map to
     */
    public static <S, D> D map(final S source, D destination) {
        modelMapper.map(source, destination);
        return destination;
    }
}

And use it for your needs:

List<PostDTO> listOfPostDTO = ObjectMapperUtils.mapAll(listOfPosts, PostDTO.class);
Untried answered 22/12, 2017 at 9:42 Comment(9)
Based on your code I wrote something similar, but easier to understand for me. And most importand, "mine". I am generally new to Generics in Java. Thank you very much for your help.Anguish
I have a question, now how can we handle the custom mapping with this generics class. I need to map 2 very different objects which I can't do as it is with this class and I really liked the approach you are following! Thnx.Peen
@ajkush, take a look at modelmapper typemaps.Untried
Thank you @AndrewNepogoda ! Created a custom converter and added it to the mapper in ObjectMappingUtils.Peen
Thank's so much! Important solution for many situations!Lovieloving
Great answer, but How can I ignore the field in the DTO, which don't need to map from Entity?Fitts
@AndrewNepogoda What is the benfit of using stream over using the "Type" and "TypeToken" to directly map the said lists using the modelmapper?Dicotyledon
@DrishtiJain, I think there're no benefits. You can use both. "Type" and "TypeToken" looks even nicer. I voted this answer as well.Untried
@Hungnn, you can use type maps and skip properties like: typeMap.addMappings(mapper -> mapper.skip(Destination::setName));Untried
M
43

considering you have a list of Post Entity (postEntityList) and a PostDTO class, you can try following:

use the following imports to get the desired results

import org.modelmapper.ModelMapper;
import org.modelmapper.TypeToken;
import java.lang.reflect.Type;

use the below code

Type listType = new TypeToken<List<PostDTO>>(){}.getType();
List<PostDTO> postDtoList = modelmapper.map(postEntityList,listType);
Macario answered 30/1, 2019 at 18:52 Comment(1)
should be modelmapper.mapFrederickfredericka
R
20

Try the following simple way:

List<PostDTO> postDtoList = Arrays.asList(modelMapper.map(postEntityList, PostDTO[].class));
Redheaded answered 6/11, 2019 at 4:5 Comment(1)
Why is util class being used as solution when this is clearly easier and out of the box without the need of custom code that does the same?Republicanize
S
16

since you want to convert Entity to Dto, you can try the following one

List<PostDTO> entityToDto = modelMapper.map(postEntity, new TypeToken<List<PostDTO>>(){}.getType());
Specialist answered 18/6, 2019 at 4:8 Comment(1)
This seems like the most straightforward approachSpue
Z
5

Let's assume you are reading from the database, and want to convert from entities in the DB to DTOs

The service layer might be separated from the controller (always the best approach), then do this:

Service layer code:

@Override
    public List<PostDTO> convertEntityToDTOList( ){
            List<Post> post = postRepository.findAll();
            Type listType = new TypeToken<List<PostDTO>>(){}.getType();
            List<PostDTO> postDTOList = modelMapper.map(post, listType);
            return postDTOList;
        }

Then in your controller, add this:

public ResponseEntity<?> list(){
        List<PostDTO> postDTOList = iPost.convertEntityToDTOList();
        return new ResponseEntity<>(postDTOList, HttpStatus.OK);
    }

Note that the iPost is an interface that defines method. This is it:

public interface iPost {
   List<PostDTO> convertEntityToDTOList();
}

Credits to @André Carvalho

Zincography answered 10/10, 2019 at 14:3 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.