武装你的WEBAPI-OData与DTO

yizhihongxing

本文属于OData系列文章

Intro

前面写了很多有关OData使用的文章,很多读者会有疑问,直接将实体对象暴露给最终用户会不会有风险?$expand在默认配置的情况下,数据会不会有泄露风险?

答案是肯定的,由于OData的特性,提供给我们便捷同时也会带来一些风险。很多地方推荐使用DTO模式来隔离实体类与最终用户使用到类的关系,从而解决以上两个问题,OData同样也适用。

DTO

DTO代表Data Transfer Object,是一种设计模式,用于在不同层之间传输数据。它通常用于将数据从一个应用程序的逻辑层传输到另一个应用程序的界面层或持久化层,以及在分布式系统中传输数据。

DTO对象是纯数据对象,它包含要从一个应用程序传输到另一个应用程序的数据。它不包含业务逻辑或数据访问代码,因此它们不能直接与数据库交互或执行任何操作,而只是简单地保存数据。

DTO对象通常由开发人员创建,并且可以根据需要进行扩展。它们可以包含各种属性和方法,以提供使用方便和更好的可读性。使用DTO对象可以降低耦合度,使不同层之间的数据传输更加简单和安全。

AutoMapper

我们需要将实体对象与DTO进行转换,对于需要转换数量不是很多的情况,直接编写一个转换函数就方便了。

    public static class DeviceDataExtension
    {
        public static DeviceDataDto ToDeviceDataDto(this Datum deviceData)
        {
            if (deviceData == null) return null;
            DeviceDataDto deviceDataDto = new()
            {
                DataArray = deviceData.DataArray,
                DeviceId = deviceData.DeviceId,
                Timestamp = deviceData.Timestamp,
                Id = Guid.NewGuid().ToString()
            };
            return deviceDataDto;
        }
    }

但是如果需要映射的属性很多,或者有很多对象的情况,建议使用对象映射工具:AutoMapper。基础用法不详细说了,讲讲对OData的支持。

首先安装对OData支持的包,由于我使用默认的DI,还需要安装DI支持的包:

Install-Package AutoMapper.Extensions.Microsoft.DependencyInjection
Install-Package AutoMapper.AspNetCore.OData.EFCore

然后有三个要求:

  • 一定要对对象声明显示展开(explicit expansion)。
  • 调用IMapper的GetAsync()或者GetQueryAsync()方法。
  • 不能在Controller或者方法上使用[EnableQuery]特性:这个我熟,因为GetQueryAsync()函数需要利用ODataQueryOptions参数,如果同时使用[EnableQuery]会导致对结果再进行一次筛选,导致返回数据错误。

代码:

            services.AddAutoMapper(option =>
            {
                option.CreateMap<Datum, DeviceDataDto>()
                .ForMember(dest => dest.Id, opt => opt.MapFrom(src => Guid.NewGuid().ToString()))
                .ForPath(dest => dest.DataArray, opt => opt.MapFrom(src => src.DataArray))
                .ForAllMembers(w => w.ExplicitExpansion());
            });

        public DeviceDatasController(IMapper mapper)
        {
            _mapper = mapper;
        }
        
        [HttpGet]
        [ProducesResponseType(typeof(IEnumerable<DeviceDataDto>), Status200OK)]
        public async Task<IActionResult> GetAsync(string key, ODataQueryOptions<DeviceDataDto> options)
        {
            var insp = await _context.DeviceData.Where(w => w.DeviceId == key).GetQueryAsync(_mapper, options);
            return Ok(insp);
        }

这样,我们就可以正常使用 OData,同时也享受了的 DTO 的好处,即可以对 DeviceDataDto 使用的 OData 查询。使用的时候要注意,如果有导航属性,导航属性也需要配置映射。

参考资料

原文链接:https://www.cnblogs.com/podolski/archive/2023/05/08/17380728.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:武装你的WEBAPI-OData与DTO - Python技术站

(0)
上一篇 2023年5月8日
下一篇 2023年5月8日

相关文章

  • asp.net子窗体与父窗体交互实战分享

    ASP.NET 子窗体与父窗体交互实战分享 本文主要介绍ASP.NET中子窗体与父窗体交互的实现方法。涉及到了IFrame嵌套、控件之间的通信等具体技术细节。 IFrame父子窗体嵌套实现方式 父子窗体嵌套方式主要有两种,一种是利用IFrame实现,另一种则是采用模态窗口的方式。这里介绍第一种方式。 在ASP.NET中,可以在主页面中嵌入一个IFrame页面…

    C# 2023年6月3日
    00
  • C#结合AForge实现摄像头录像

    C#结合AForge实现摄像头录像攻略 本攻略将详细讲解如何使用C#和AForge库实现摄像头录像功能。 准备工作 在开始编写代码之前,您需要准备以下环境和工具: C#编程环境 AForge.NET库 AForge.NET库简介 AForge.NET是一个开源的计算机视觉和人工智能框架,支持图像处理、视频处理、人脸识别、机器学习等功能。在本文中,我们将使用其…

    C# 2023年6月3日
    00
  • C# Socket编程实现简单的局域网聊天器的示例代码

    下面我将为您详细讲解如何使用C# Socket编程实现局域网聊天器的示例代码。 1. 简介 Socket编程是指通过套接字(socket)实现网络通信的编程,可以实现多种类型的网络通信,包括TCP、UDP等。在局域网中,可以使用Socket编程实现简单的聊天器,实现用户之间的即时通信。 2. Socket编程基础知识 在开始开发局域网聊天器之前,需要了解So…

    C# 2023年5月31日
    00
  • ASP.net(c#)用类的思想实现插入数据到ACCESS例子

    让我们来详细讲解一下如何用类的思想实现插入数据到 ACCESS 数据库。这里我们使用 ASP.NET (C#)编写代码。 1. 创建与数据库连接的类 在 ASP.NET 中,我们使用 System.Data.OleDb 命名空间来操作 ACCESS 数据库。首先,我们需要创建一个类来封装与数据连接相关的操作。 using System.Data.OleDb;…

    C# 2023年6月3日
    00
  • C#框架winform实现简单点餐系统

    下面是详细讲解“C#框架winform实现简单点餐系统”的完整攻略。 1. 准备工作 在正式开始开发点餐系统之前,需要准备好相关的工具和资源,以下是准备工作的具体步骤: 1.1 安装Visual Studio Visual Studio是Windows平台下的一款集成开发环境,它支持多种编程语言,其中包括C#。因此,我们需要安装Visual Studio来进…

    C# 2023年6月3日
    00
  • 在Blazor中使用Chart.js生成图表

    1. 在Blazor中使用Chart.js 首先,从Chart.js官方网站下载Chart.js库文件。推荐下载这个构建好的版本https://cdnjs.com/libraries/Chart.js,最新版是v4.2.1 在Blazor项目中把刚刚下载好的Chart.js放到wwwroot目录下。 在Blazor项目中的Pages文件夹下_Host.csh…

    C# 2023年4月18日
    00
  • ASP.NET Core中如何利用多种方式给Action传参

    在ASP.NET Core中,您可以使用多种方式将参数传递给Action。以下是一些常见的方法: 1. 通过路由参数传递参数 在ASP.NET Core中,您可以通过路由参数将参数传递给Action。以下是一个示例: [Route("products/{id}")] public IActionResult GetProduct(int …

    C# 2023年5月17日
    00
  • C#打印日志的方法总结

    针对“C#打印日志的方法总结”,以下是详细的攻略: 什么是日志 在开发过程中,我们需要记录一些关键信息来方便排查问题或者进行后续分析。而日志就是我们记录这些信息的工具。日志可以记录程序运行时的各种信息,包括但不限于:debug信息、错误堆栈信息、请求和响应信息等。而我们可以使用一些工具来进行日志的打印和管理。 C#中的日志打印方法 1.使用Trace和Deb…

    C# 2023年5月15日
    00
合作推广
合作推广
分享本页
返回顶部