您的位置:

Orika 是 Java 中一个易于使用和可扩展的对象映射框架

Orika 是一个 Java 对象映射框架,允许将一个类的属性值映射到另一个类实例的属性值,同时具有良好的性能和易用性。对于开发人员而言,大多数映射需求都可以通过 Orika 来实现。 Orika 还提供了许多定制选项和非常详细的文档供使用者参考。除此之外,它还是一个开源框架,所以可以轻松找到维护者和用户社区。

一、Orika 概述

Orika 框架是由手头忙于 Spring Boot 后端开发的开发者由于复杂对象转换等问题创造的。Orika 的目标是提供一个使用简单,性能优良,扩展性好的框架。Orika 是在 Apache License 下发布的。

Orika 可以在任何 Java 应用程序中使用,并在性能上优于其他大多数映射框架。 Orika 不仅仅只是映射简单的属性。 Orika 还支持从一个映射中排除某些字段,包含复杂集合的处理方式等等。

二、Orika 的使用

为了在应用程序中使用 Orika,您需要使用 Maven 或 Gradle 获得 Orika 依赖。以下是 Maven 的配置示例:

    <dependency>
        <groupId>ma.glasnost.orika</groupId>
        <artifactId>orika-core</artifactId>
        <version>1.5.2</version>
    </dependency>

使用 Orika 非常简单。你需要创建一个 MapperFactory,这个 MapperFactory 用来构造 Mapper 实例。

    MapperFactory mapperFactory = new DefaultMapperFactory.Builder().build();

构建 MapperFactory 对象后,你可以从 MapperFactory 中获取一个实例化的 Mapper,同时使用 MapStruct 注解 来自动生成 Mapper 接口:

    @Mapper
    public interface PersonMapper {
        @Mapping(source = "firstName", target = "name")
        PersonDto personToPersonDto(Person person);
        Person personDtoToPerson(PersonDto personDto);
    }

    PersonMapper personMapper = mapperFactory.getMapperFacade(PersonMapper.class);

在这个例子中,我们将 Person 对象转变为 PersonDto 对象。注册 Mapper 是一个非常显式的过程,这使得 Orika 的行为和获得基本类型的映射器非常相似。

三、Orika 的性能

由于 Orika 的核心是映射对象,因此它的性能是非常重要的。 Orika 实现的关键在于,它在映射两个对象时,不会反射获取目标属 性的名称和类型。取而代之的是,它将映射分解为类似于一个转换表的方法,其中每个属性都被处理为一个映射。 Orika 在内部使用了一个 HashTable 来存储已经进行过映射的类,因此它能够缓存这个信息,以便下次使用这个类时,能快速地找到属性映射和构造函数。

性能测试的例子是,将含有一千万个对象的列表进行映射:

    List<Person> persons = createPersons(10000000);
    List<PersonDto> dtos;

    MapperFactory mapperFactory = new DefaultMapperFactory.Builder().build();
    PersonMapper mapper = mapperFactory.getMapperFacade(PersonMapper.class);

    long t1 = System.currentTimeMillis();
    dtos = mapperFacade.mapAsList(persons, PersonDto.class);
    long t2 = System.currentTimeMillis();
    System.out.println("Orika takes " + (t2 - t1) + "ms to map " + persons.size() + " objects");

在我的笔记本电脑上,基于 Orika 的这种映射方式大约需要 4 秒钟。结果显示 Orika 的速度非常快,可以很好地处理10万以上的对象列表。

四、Orika 的高级功能

Orika 提供了一系列高级映射选项。这些是常用选项的一个子集:

  • 映射的上下文:此功能允许您从 MapperFacade 中的全局映射中获取更多的信息,例如当前的目标对象。
  • 自定义映射:您可以通过自定义转换器来扩展到其他类型。这对于处理复杂类型很有用,例如将文本转换为日期。
  • 嵌套映射:Orika 可以从一个带嵌套对象的源映射到带嵌套对象的目标。
  • 忽略特定属性。

在以下示例中,我将演示如何在 Orika 中使用自定义转换器和嵌套映射。

考虑以下两个类:

    public class Person {
        private String name;
        private String lastName;
        private LocalDate birthDate;
    }

    public class PersonDto {
        private String name;
        private String lastName;
        private String formattedBirthDate;
    }

我们将使用自定义转换器将 LocalDate 转换为字符串。

    mapperFactory.getConverterFactory().registerConverter(new CustomDateConverter());

自定义转换器:

    public class CustomDateConverter extends BidirectionalConverter<LocalDate, String> {
        @Override
        public String convertTo(LocalDate localDate, Type<? extends String> type, MappingContext mappingContext) {
            return localDate.format(DateTimeFormatter.ofPattern("yyyy-MM-dd"));
        }

        @Override
        public LocalDate convertFrom(String s, Type<? extends LocalDate> type, MappingContext mappingContext) {
            return LocalDate.parse(s, DateTimeFormatter.ofPattern("yyyy-MM-dd"));
        }
    }

我们需要映射 Person -> PersonDto,现在我们可以使用自定义转换器了:

    @Mapper
    public interface PersonMapper {
        @Mappings({
                @Mapping(source = "firstName", target = "name"),
                @Mapping(source = "lastName", target = "lastName"),
                @Mapping(source = "birthDate", target = "formattedBirthDate", converter = CustomDateConverter.class)
        })
        PersonDto personToPersonDto(Person person);
    }

现在我们将演示如何处理映射嵌套对象的情况。我们再添加一个 Student 类,其内部包含一个 Person 类型的属性以及其他属性:

    public class Student {
        private Person person;
        //Other properties
    }

我们将使用另一个 Mapper(用于将 Person 映射到 PersonDto ),然后在 StudentMapper 中引用它并进行嵌套映射。我们还将完成对 Person 和 Student 的映射。

    @Mapper
    public interface StudentMapper {
        @Mapping(source = "person", target = "personDto")
        StudentDto studentToStudentDto(Student student);

        @Mapping(source = "name", target = "name")
        PersonDto personToPersonDto(Person person);
    }

注意,我们使用了 @Mapping(source = "person", target = "personDto") 将 Person 映射到 PersonDto 。在 StudentDto 中,我们使用的是新的转换字段 personDto 。

总结

Orika 是一个简单易用、高性能、支持高级映射选项的 Java 映射框架。您可以通过 Maven 或 Gradle 获得 Orika,同时 Orika 在性能上优于其他大多数映射框架,可以帮助开发者更快速地编写应用程序,同时极大提升了开发工作的便捷性。