从入门到精通:C# 中 AutoMapper 的深度解析与实战应用

 在 C# 开发领域,尤其是企业级应用开发过程中,不同层次和模块之间的数据传递与对象转换是常见需求。例如,从数据库读取的实体类,在传递到前端时,往往需要转换为更简洁、安全的数据传输对象(DTO) 。手动编写对象属性映射代码不仅繁琐,还容易出错,而AutoMapper正是解决这一痛点的强大工具。本文将带你从入门到精通,全面掌握 AutoMapper 的使用技巧,助力你在 C# 项目开发中更高效地处理对象映射。

一、AutoMapper 基础概念:为什么选择它?

AutoMapper 是一个开源的对象 - 对象映射库,它的核心使命是简化不同对象类型之间的属性映射过程。想象一下,在一个用户管理系统中,数据库实体类User包含了诸如PasswordHash等敏感字段,而对外提供的 API 接口返回的UserDto对象,显然不能包含这些敏感信息,并且还可能需要对属性进行重新组合和计算,比如将FirstName和LastName合并为FullName。如果手动编写映射代码,不仅代码量庞大,而且后续对象结构发生变化时,维护成本极高。

AutoMapper 通过定义映射规则,能够自动完成属性的匹配与转换,极大地提高了开发效率,同时增强了代码的可维护性和可读性。无论是简单的属性名对应映射,还是复杂的计算属性、嵌套对象映射,AutoMapper 都能轻松应对。

二、快速入门:AutoMapper 的基础使用

2.1 安装与配置

首先,我们需要通过 NuGet 包管理器安装 AutoMapper。在 Visual Studio 的 “管理 NuGet 程序包” 界面,搜索并安装AutoMapper和AutoMapper.Extensions.Microsoft.DependencyInjection(用于在ASP.NET Core 项目中集成依赖注入)。

安装完成后,我们可以通过以下代码进行基础配置:

using AutoMapper;// 源类(数据库实体)public class User{public int Id { get; set; }public string FirstName { get; set; }public string LastName { get; set; }public DateTime BirthDate { get; set; }}// 目标类(DTO)public class UserDto{public int Id { get; set; }public string FullName { get; set; }public int Age { get; set; }}// 配置映射var config = new MapperConfiguration(cfg =>{cfg.CreateMap<User, UserDto>().ForMember(dest => dest.FullName, opt => opt.MapFrom(src => $"{src.FirstName} {src.LastName}")).ForMember(dest => dest.Age, opt => opt.MapFrom(src => DateTime.Now.Year - src.BirthDate.Year));});// 创建映射器实例var mapper = config.CreateMapper();

2.2 执行映射

有了映射器实例后,执行对象映射就变得非常简单:

// 源对象var user = new User { Id = 1, FirstName = "John", LastName = "Doe", BirthDate = new DateTime(1990, 1, 1) };// 执行映射var userDto = mapper.Map<UserDto>(user);// 输出结果:// userDto.FullName: "John Doe"// userDto.Age: 33 (假设当前年份为2023)

通过上述代码,我们清晰地看到,AutoMapper 根据我们定义的规则,自动将User对象的属性转换并填充到了UserDto对象中。

三、进阶用法:AutoMapper 的更多可能性

3.1 集合映射

在实际项目中,处理集合类型的映射十分常见,比如查询多个用户并返回用户 DTO 列表。AutoMapper 支持直接对集合进行映射:

var users = new List<User>{new User { Id = 1, FirstName = "John", LastName = "Doe" },new User { Id = 2, FirstName = "Jane", LastName = "Smith" }};// 直接映射整个列表var userDtos = mapper.Map<List<UserDto>>(users);

AutoMapper 会自动遍历集合中的每个元素,并按照定义的映射规则进行转换。

3.2 反向映射

有时候,我们不仅需要从源对象映射到目标对象,还需要从目标对象反向映射回源对象,这时可以使用ReverseMap()方法:

var config = new MapperConfiguration(cfg =>{cfg.CreateMap<User, UserDto>().ForMember(dest => dest.FullName, opt => opt.MapFrom(src => $"{src.FirstName} {src.LastName}")).ReverseMap().ForMember(src => src.FirstName, opt => opt.MapFrom(dest => dest.FullName.Split(' ')[0])).ForMember(src => src.LastName, opt => opt.MapFrom(dest => dest.FullName.Split(' ')[1]));});

通过ReverseMap(),我们可以快速创建双向映射规则,进一步提高开发效率。

3.3 条件映射

在某些情况下,属性的映射需要满足特定条件,比如只有当用户年龄大于 18 岁时,才将年龄映射到目标对象:

var config = new MapperConfiguration(cfg =>{cfg.CreateMap<User, UserDto>().ForMember(dest => dest.Age, opt => opt.MapFrom(src => DateTime.Now.Year - src.BirthDate.Year)).ForMember(dest => dest.ShouldShowAge, opt => opt.Condition(src => DateTime.Now.Year - src.BirthDate.Year > 18));});

这里使用Condition方法,根据年龄条件决定是否映射ShouldShowAge属性。

3.4 扁平化映射

当处理嵌套对象时,我们可能希望将嵌套属性 “扁平化” 到目标对象中,例如:

// 源类(包含嵌套对象)public class Order{public int OrderId { get; set; }public Customer Customer { get; set; }}public class Customer{public string Name { get; set; }public Address Address { get; set; }}public class Address{public string City { get; set; }public string Street { get; set; }}// 目标类(扁平化属性)public class OrderDto{public int OrderId { get; set; }public string CustomerName { get; set; }public string CustomerCity { get; set; }}// 配置映射var config = new MapperConfiguration(cfg =>{cfg.CreateMap<Order, OrderDto>().ForMember(dest => dest.CustomerName, opt => opt.MapFrom(src => src.Customer.Name)).ForMember(dest => dest.CustomerCity, opt => opt.MapFrom(src => src.Customer.Address.City));});

通过ForMember指定嵌套属性的映射路径,实现了扁平化映射。

四、高级技巧:应对复杂场景

4.1 自定义值解析器

对于复杂的映射逻辑,简单的属性映射或条件映射可能无法满足需求,这时可以使用自定义值解析器(Value Resolver)。例如,精确计算用户年龄(考虑生日是否已过):

using AutoMapper;// 自定义解析器public class AgeResolver : IValueResolver<User, UserDto, int>{public int Resolve(User source, UserDto destination, int destMember, ResolutionContext context){var today = DateTime.Today;var age = today.Year - source.BirthDate.Year;if (source.BirthDate > today.AddYears(-age))age--;return age;}}// 配置映射var config = new MapperConfiguration(cfg =>{cfg.CreateMap<User, UserDto>().ForMember(dest => dest.Age, opt => opt.MapFrom<AgeResolver>());});

通过实现IValueResolver接口,我们可以将复杂的计算逻辑封装在解析器中,使映射配置更加清晰。

4.2 继承映射

当涉及到继承关系的对象映射时,AutoMapper 也提供了很好的支持。例如:

// 基类public class Animal{public string Name { get; set; }public int Age { get; set; }}// 子类public class Dog : Animal{public string Breed { get; set; }}// 目标类public class AnimalDto{public string Name { get; set; }public int Age { get; set; }public string Type { get; set; }}// 配置映射var config = new MapperConfiguration(cfg =>{cfg.CreateMap<Animal, AnimalDto>().Include<Dog, AnimalDto>().ForMember(dest => dest.Type, opt => opt.MapFrom(src => src.GetType().Name));cfg.CreateMap<Dog, AnimalDto>().ForMember(dest => dest.Type, opt => opt.MapFrom(src => "Dog"));});

通过Include方法,我们可以指定子类的映射规则,同时在基类映射中进行通用配置。

4.3 配置文件分离

随着项目规模的扩大,映射配置可能会变得非常复杂,将映射配置拆分到独立的配置文件中是一个良好的实践。我们可以通过继承Profile类来实现:

using AutoMapper;// 自定义映射配置文件public class UserProfile : Profile{public UserProfile(){CreateMap<User, UserDto>().ForMember(dest => dest.FullName, opt => opt.MapFrom(src => $"{src.FirstName} {src.LastName}")).ForMember(dest => dest.Age, opt => opt.MapFrom(src => DateTime.Now.Year - src.BirthDate.Year));}}// 注册所有Profilevar config = new MapperConfiguration(cfg =>{cfg.AddProfile<UserProfile>();// 可添加多个Profile});

这种方式使得映射配置更加模块化,便于维护和管理。

五、在ASP.NET Core 中的集成

在ASP.NET Core 项目中,我们可以通过依赖注入方便地使用 AutoMapper。在Program.cs文件中进行如下配置:

using AutoMapper;using Microsoft.Extensions.DependencyInjection;var builder = WebApplication.CreateBuilder(args);// 注册AutoMapperbuilder.Services.AddAutoMapper(typeof(Program)); // 扫描当前程序集中的所有Profile// 或者手动注册Profile// builder.Services.AddAutoMapper(typeof(UserProfile));var app = builder.Build();// 在Controller中使用app.MapGet("/users/{id}", (int id, IMapper mapper) =>{var user = GetUserFromDatabase(id); // 假设的获取用户方法var userDto = mapper.Map<UserDto>(user);return Results.Ok(userDto);});app.Run();

通过依赖注入,我们可以在控制器、服务等组件中轻松获取IMapper实例,完成对象映射操作。

六、性能优化与常见问题

6.1 性能优化

为了提高 AutoMapper 的映射性能,我们需要注意以下几点:

  • 避免重复创建映射器实例:映射器实例的创建开销较大,应尽量在应用程序启动时创建一次,并通过依赖注入进行共享,而不是在每次映射时创建。
  • 预编译映射:在开发或测试环境中,使用AssertConfigurationIsValid()方法验证映射配置的正确性,这可以在首次映射时提高性能。

6.2 常见问题与解决方案

  • 映射失败:常见原因包括属性名不匹配、类型无法转换、嵌套对象未配置映射等。解决方法是仔细检查映射规则,使用ForMember手动指定映射关系,或者添加自定义类型转换器。
  • 循环引用:当两个对象相互引用时,可能会导致循环引用问题。可以使用PreserveReferences()方法保留引用关系,避免无限递归映射。
  • 映射逻辑复杂:如果映射规则中包含过多的条件和计算逻辑,会使代码变得难以维护。此时应将复杂逻辑封装到自定义值解析器或单独的服务类中。

七、总结与展望

从基础的属性映射到复杂场景下的高级应用,AutoMapper 在 C# 开发中展现出了强大的灵活性和实用性。通过合理运用 AutoMapper 的各种特性,我们能够显著提升开发效率,减少手动映射带来的错误风险。随着项目需求的不断变化和技术的发展,AutoMapper 也在持续更新和完善,未来它将为开发者带来更多便捷的功能和优化。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/bicheng/84920.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【热更新知识】学习一 Lua语法学习

1、注释 1.1 单行注释 --注释内容 --单行注释 print打印函数 1.2 多行注释&#xff0c;三种方式 --[[注释内容]] --[[注释内容]]-- --[[注释内容--]] --[[ 多行 注释 ]]--[[ 第二种多行注释 1 2 ]]----[[ 第三种 多行 注释 --]] 2、简单变量 2.1 声明变量&#xff0c…

React 第三方状态管理库的比较与选择

在现代前端开发中,状态管理是一个重要的环节。选择合适的状态管理库可以极大地提高项目的可维护性和开发效率。本文将对几种流行的状态管理库进行比较,包括Valtio、XState、MobX、Recoil和Zustand,帮助开发者在实际项目中做出明智的选择。 1. Valtio 1.1. 设计理念 Valti…

《Kafka 在实时消息系统中的高可用架构设计》

Kafka 在实时消息系统中的高可用架构设计 引言 在当今互联网社交应用中&#xff0c;实时消息系统已成为核心基础设施。以中性互联网公司为例&#xff0c;其每天需要处理数十亿条消息&#xff0c;涵盖一对一聊天、群组互动、直播弹幕等多种场景。特别是在大型直播活动中&#…

SKUA-GOCAD入门教程-第八节 线的创建与编辑3

8.1.4根据面对象创建曲线 (1)从曲面生成曲线 从曲面边界生成曲线您可以从选定的曲面边界创建一条单段曲线。 1、选择 Curve commands > New > Borders > One 打开从曲面的一条边界创建曲线对话框。 图1 在“Name名称”框中,输入要创建的曲线的名称。

Unity编辑器-获取Projectwindow中拖拽内容的路径

参考 Unity Editor 实现给属性面板上拖拽赋值资源路径 API Event DragAndDrop 示例 Mono脚本 using UnityEngine; public class TestScene : MonoBehaviour {[SerializeField] string testName; }Editor脚本 重写InspectorGUI&#xff0c;在该函数中通过Event的Type参数获…

重要的城市(图论 最短路)

分析 a ≠ b的从a到B的最短路&#xff0c;才有重要城市。 求出最短路&#xff0c;才能确定重要城市。 是多源最短路&#xff0c;n ≤ 200&#xff0c;可用Floyd。 若a到b&#xff0c;只有一条最短路&#xff0c;那么 a到b的路径上的点&#xff08;除了a、b&#xff09;都是…

50种3D效果演示(OpenGL)

效果&#xff1a; 一、只需打开命令行&#xff08;Windows 可用 cmd&#xff09;&#xff0c;输入&#xff1a; pip install PyQt5 PyOpenGL numpy二、用命令行进入保存 .py 文件的目录&#xff0c;运行&#xff1a; python openGL_3d_demo.py三、建立python文件命名openGL_3…

Java大模型开发入门 (6/15):对话的灵魂 - 深入理解LangChain4j中的模型、提示和解析器

前言 在上一篇文章中&#xff0c;我们见证了AiService注解的惊人威力。仅仅通过定义一个Java接口&#xff0c;我们就实现了一个功能完备的AI聊天服务。这感觉就像魔法一样&#xff01; 但作为专业的工程师&#xff0c;我们知道“任何足够先进的技术&#xff0c;都与魔法无异”…

用Rust如何构建高性能爬虫

习惯了使用Python来写爬虫&#xff0c;如果使用Rust需要有哪些考量&#xff1f; 根据我了解的Rust 在性能、资源效率和并发处理方面完胜 Python&#xff0c;但是 Python 在开发速度和生态成熟度上占优。所以说&#xff0c;具体用那种模式&#xff0c;结合你项目特点做个详细的…

CentOS7报错:Cannot find a valid baseurl for repo: base/7/x86_64

这个错误通常出现在 CentOS/RHEL 7 系统中&#xff0c;当你尝试运行 yum update 或 yum install 时&#xff0c;系统无法连接到默认的软件仓库&#xff08;repository&#xff09;。 可能的原因 网络连接问题&#xff1a;系统无法访问互联网或仓库服务器。错误的仓库配置&…

云平台|Linux部分指令

目录 云平台 操作系统&#xff08;镜像&#xff09; 管理应用实例 远程连接 远程连接工具 linux相关命令&#xff08;重点&#xff09; 云平台 1、阿里云&#xff08;学生免费&#xff0c;不包流量 流量0.8---1G&#xff09; 2、腾讯云&#xff08;抢&#xff09; 3、华…

AI首次自主发现人工生命

转&#xff1a; 近日&#xff0c;人工智能领域迎来了一项革命性的突破。Transformer 论文作者之一的 Llion Jones 与前谷歌研究人员 David Ha 共同创立的人工智能公司 Sakana AI&#xff0c;联合MIT、OpenAI、瑞士AI实验室IDSIA等机构的研究人员&#xff0c;共同提出了一种名为…

Day.31

变量类型&#xff1a; name: str "Alice" age: int 30 height: float 1.75 is_student: bool False 注解&#xff1a; def add(a: int, b: int) -> int: return a b def greet(name: str) -> None: print(f"Hello, {name}") 定义矩形类&a…

光谱数据分析的方法有哪些?

光谱数据分析是通过特征光谱识别物质结构与成分的核心技术&#xff0c;其标准化流程如下&#xff1a; ‌一、数据预处理‌&#xff08;消除干扰噪声&#xff09; ‌去噪平滑‌ Savitzky-Golay滤波&#xff1a;保留光谱特征峰形&#xff0c;消除高频噪声。 移动平均法&#…

RabbitMQ的使用--Spring AMQP(更新中)

1.首先是创建项目 在一个父工程 mq_demo 的基础上建立两个子模块&#xff0c;生产者模块publisher&#xff0c;消费者模块 consumer 创建项目&#xff1a; 建立成功&#xff1a; 删除多余文件 创建子模块1&#xff1a;publisher&#xff08;生产者模块&#xff09; 右键---…

DAY 31 文件的规范拆分和写法

浙大疏锦行 今日的示例代码包含2个部分 notebook文件夹内的ipynb文件&#xff0c;介绍下今天的思路项目文件夹中其他部分&#xff1a;拆分后的信贷项目&#xff0c;学习下如何拆分的&#xff0c;未来你看到的很多大项目都是类似的拆分方法 知识点回顾 规范的文件命名规范的文件…

EtherCAT至TCP/IP异构网络互联:施耐德M580 PLC对接倍福CX5140解决方案

一、项目背景与需求 某智能工厂致力于打造高度自动化的生产流水线&#xff0c;其中部分核心设备采用EtherCAT协议进行通信&#xff0c;以实现高速、高精度的控制&#xff0c;例如基于EtherCAT总线的倍福&#xff08;Beckhoff&#xff09;CX5140PLC&#xff0c;它能够快速响应设…

[学习] FIR多项滤波器的数学原理详解:从多相分解到高效实现(完整仿真代码)

FIR多项滤波器的数学原理详解&#xff1a;从多相分解到高效实现 文章目录 FIR多项滤波器的数学原理详解&#xff1a;从多相分解到高效实现引言一、FIR滤波器基础与多相分解原理1.1 FIR滤波器数学模型1.2 多相分解的数学推导1.3 多相分解的物理意义 二、插值应用中的数学原理2.1…

Java并发编程实战 Day 22:高性能无锁编程技术

【Java并发编程实战 Day 22】高性能无锁编程技术 文章简述 在高并发场景下&#xff0c;传统的锁机制&#xff08;如synchronized、ReentrantLock&#xff09;虽然能够保证线程安全&#xff0c;但在高竞争环境下容易引发性能瓶颈。本文深入探讨无锁编程技术&#xff0c;重点介绍…

打破语言壁垒!DHTMLX Gantt 与 Scheduler 文档正式上线中文等多语言版本!

你还在为英文技术文档望而却步吗&#xff1f;现在好消息来了&#xff01;DHTMLX 团队宣布&#xff0c;其两款明星组件——DHTMLX Gantt&#xff08;甘特图&#xff09;与 DHTMLX Scheduler&#xff08;日程排程器&#xff09;的官方文档&#xff0c;现已全面支持中文、德语、韩…