0

0

MapStruct:在列表映射中传递并传播额外参数的教程

碧海醫心

碧海醫心

发布时间:2025-10-01 14:05:24

|

855人浏览过

|

来源于php中文网

原创

MapStruct:在列表映射中传递并传播额外参数的教程

本教程详细介绍了如何在MapStruct中,将一个额外的参数(如ID)从列表映射方法(如mapDTOs)向下传递并传播到其内部的单个对象映射方法(如map)。核心解决方案是利用@Context注解标记额外参数,并结合一个默认的代理方法来确保MapStruct能正确地将上下文参数传递给每个元素映射。

理解MapStruct中的参数传播需求

在进行对象映射时,我们经常会遇到需要将一个额外参数(例如,一个全局id、用户信息或某种配置)从一个顶层映射方法传递到其内部调用的子映射方法的情况。尤其是在处理集合(如list)的映射时,我们希望这个额外参数能够被集合中的每个元素的独立映射过程所感知和使用。

例如,假设我们有一个MyOM对象列表需要映射为MyEntity对象列表。同时,我们希望在每个MyEntity对象中都设置一个外部传入的id。如果只有一个单对象映射方法:

@Mapping(target = "id", expression = "java(id)")
MyEntity map(MyOM om, String id);

这个方法能够将传入的id设置到MyEntity的id字段。但当我们尝试为列表创建映射方法时,如何确保这个id参数也能被列表中的每个MyOM元素映射到其对应的MyEntity中呢?

List mapDTOs(List dtos, String id); // 如何让这个id传递下去?

直接在mapDTOs方法上使用@Mapping注解通常不足以解决这个问题,因为@Mapping主要用于定义源对象和目标对象之间的属性映射关系,而非参数的传递机制。

使用@Context注解进行参数传播

MapStruct从1.2版本开始引入了@Context注解,专门用于解决此类上下文参数的传递问题。当一个参数被@Context注解标记时,MapStruct会识别它为“上下文参数”,并尝试将其传递给所有可能被调用的、接受相同类型和名称的@Context参数的子映射方法。

1. 声明列表映射方法

首先,在列表映射方法中,将需要传播的额外参数标记为@Context。请注意,这个方法本身不需要@Mapping注解,因为其主要功能是委托给单个元素的映射。

import org.mapstruct.Context;
import java.util.List;

// 假设MyOM和MyEntity已经定义
// public class MyOM { /* ... */ }
// public class MyEntity { String id; /* ... */ }

@Mapper // 确保你的接口是MapStruct的Mapper
public interface MyMapper {

    // 单个对象映射方法,用于设置id
    @Mapping(target = "id", expression = "java(id)")
    MyEntity map(MyOM om, String id);

    // 列表映射方法,使用@Context传播id
    List mapDTOs(List dtos, @Context String id);
}

此时,如果直接运行,MapStruct可能会抱怨找不到一个接受@Context String id参数的单对象映射方法,或者生成一个不带id映射的新方法。这是因为MapStruct默认会寻找一个与列表元素类型匹配的单对象映射方法,而我们现有的map(MyOM om, String id)方法中的id并未被标记为@Context。

2. 添加代理(Default)方法

为了解决上述问题,我们需要在Mapper接口中添加一个default方法,作为MapStruct在处理列表时调用的“代理”。这个代理方法的作用是显式地将带有@Context注解的参数传递给原始的单对象映射方法。

Peech
Peech

Peech是一个为营销团队设计的生成式AI视频平台

下载
import org.mapstruct.Context;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import java.util.List;

@Mapper
public interface MyMapper {

    // 原始的单个对象映射方法
    @Mapping(target = "id", expression = "java(id)")
    MyEntity map(MyOM om, String id);

    // 列表映射方法,使用@Context传播id
    List mapDTOs(List dtos, @Context String id);

    // 代理方法:将带有@Context的id参数传递给原始的map方法
    default MyEntity mapContext(MyOM om, @Context String id) {
        return map(om, id);
    }
}

通过添加mapContext这个default方法,我们告诉MapStruct:当它需要映射单个MyOM并有一个@Context String id可用时,它应该调用mapContext。而mapContext内部又会调用我们预期的map(om, id)方法,从而实现了id参数的正确传播。

示例代码

假设我们有以下数据模型:

// 源对象
public class MyOM {
    private String name;
    // ... 其他字段
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
}

// 目标对象
public class MyEntity {
    private String id;
    private String entityName;
    // ... 其他字段
    public String getId() { return id; }
    public void setId(String id) { this.id = id; }
    public String getEntityName() { return entityName; }
    public void setEntityName(String entityName) { this.entityName = entityName; }
}

完整的Mapper接口:

import org.mapstruct.Context;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.factory.Mappers;

import java.util.List;

@Mapper
public interface MyMapper {

    MyMapper INSTANCE = Mappers.getMapper(MyMapper.class);

    // 单个对象映射方法,将om的name映射到entityName,并设置传入的id
    @Mapping(target = "entityName", source = "om.name")
    @Mapping(target = "id", expression = "java(id)")
    MyEntity map(MyOM om, String id);

    // 列表映射方法,使用@Context传播id
    List mapDTOs(List dtos, @Context String id);

    // 代理方法:将带有@Context的id参数传递给原始的map方法
    default MyEntity mapContext(MyOM om, @Context String id) {
        return map(om, id);
    }
}

使用示例:

import java.util.Arrays;
import java.util.List;

public class Main {
    public static void main(String[] args) {
        MyOM om1 = new MyOM();
        om1.setName("Object A");

        MyOM om2 = new MyOM();
        om2.setName("Object B");

        List omList = Arrays.asList(om1, om2);
        String globalId = "GLOBAL_UUID_123";

        List entityList = MyMapper.INSTANCE.mapDTOs(omList, globalId);

        for (MyEntity entity : entityList) {
            System.out.println("Entity ID: " + entity.getId() + ", Entity Name: " + entity.getEntityName());
        }
        // 预期输出:
        // Entity ID: GLOBAL_UUID_123, Entity Name: Object A
        // Entity ID: GLOBAL_UUID_123, Entity Name: Object B
    }
}

注意事项与总结

  1. MapStruct版本要求:@Context注解是在MapStruct 1.2版本中引入的。请确保你的项目使用的MapStruct版本不低于此。
  2. @Context参数的特性:@Context参数本身不被视为映射的源属性。它们的主要目的是在映射方法链中传递上下文信息。
  3. 代理方法的必要性:当你的单对象映射方法(如map(MyOM om, String id))中的额外参数并未标记为@Context,而列表映射方法(如mapDTOs(List dtos, @Context String id))中的额外参数标记为@Context时,代理方法是必需的。它充当了MapStruct在上下文参数和非上下文参数之间进行桥接的机制。
  4. 清晰的职责:通过这种方式,我们可以清晰地分离列表映射和单个元素映射的职责,同时确保上下文信息在整个映射过程中正确传递。
  5. 替代方案:对于更复杂的上下文管理,可以考虑将上下文参数封装在一个单独的上下文对象中,并在该对象上使用@Context,这样可以传递多个相关参数。

通过遵循上述指导,你可以在MapStruct中有效地管理和传播额外的上下文参数,从而实现更灵活和强大的映射逻辑。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

422

2023.08.02

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1078

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

169

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1356

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

16

2026.01.19

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

36

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

60

2025.11.17

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.9万人学习

C# 教程
C# 教程

共94课时 | 7.7万人学习

Java 教程
Java 教程

共578课时 | 51.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号