繁体   English   中英

Ef Core - 使用 Automapper 从无键视图获取数据

[英]Ef Core - Use Automapper to get data from keyless View

我需要知道是否可以使用 Automapper 从数据库视图映射 DTO 属性。 或者也许在 DbContext 模型配置中是可能的。 让我们想象一下,我有一个包含其他相关数据的视图的商业目的,但为了简洁起见,我使类变得简单

相关的掘金

  • EF 核心 v3.1.7
  • AutoMapper v10.0.0

我有一个实体

public class Foo {
    public int Id { get; set; }
    public string Name { get; set; }
}

我有一个无钥匙视图

public class BarVW {
    public int FooId { get; set; }
    public string FooName { get; set; }
}

该视图构建在 DB Initializer 类中

context.Database.ExecuteSqlRaw(
    @"CREATE OR REPLACE VIEW v_Bar AS 
        select
            f.Id as FooId,
            f.[Name] as FooName
        from
            Foo f
        -- where some logic that makes a Foo appear in view"
);

然后将视图分配给 DbContext 类中的一个 DbSet

modelBuilder.Entity<BarVW>(eb =>
    {
        eb.HasNoKey();
        eb.ToView("v_Bar");
    });

public DbSet<BarVW> BarVW { get; set; }

在我的 FooDto 中,我需要一个附加属性到我的实体类,以表明这个 Foo 存在于我的数据库视图 v_Bar 中

public class FooDto {
    public int Id { get; set; }
    public string Name { get; set; }
    public bool IsBar { get; set; }
}

对于 Automapper 配置,我有以下内容,但我知道没有一种方法可以从 BarVW DbSet 映射我的 dto IsBar属性

CreateMap<Foo, FooDto>()
   .ForMember(dto => dto.IsBar, opt => ??? ); // don't know what to put here

实现您想要的一个简单方法是引入导航属性(例如Bars )并在映射配置中使用它(例如opt.MapFrom(src => src.Bars.Any()) )。

这是一个完全工作的示例控制台程序,它演示了这种方法:

using System.Diagnostics;
using System.Linq;
using AutoMapper;
using AutoMapper.QueryableExtensions;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;

namespace IssueConsoleTemplate
{
    public class Foo
    {
        public int Id { get; set; }
        public string Name { get; set; }
        
        public BarVW Bar { get; set; }
    }

    public class BarVW
    {
        public int FooId { get; set; }
        public string FooName { get; set; }
        
        public Foo Foo { get; set; }
    }
    
    public class Context : DbContext
    {
        public virtual DbSet<Foo> Foo { get; set; }
        public virtual DbSet<BarVW> BarVW { get; set; }

        protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        {
            optionsBuilder
                .UseSqlServer(
                    @"Data Source=.\MSSQL14;Integrated Security=SSPI;Initial Catalog=So63850736")
                .UseLoggerFactory(LoggerFactory.Create(b => b
                    .AddConsole()
                    .AddFilter(level => level >= LogLevel.Information)))
                .EnableSensitiveDataLogging()
                .EnableDetailedErrors();
        }

        protected override void OnModelCreating(ModelBuilder modelBuilder)
        {
            modelBuilder.Entity<Foo>()
                .HasData(
                    new Foo {Id = 1, Name = "Fo"},
                    new Foo {Id = 2, Name = "Foo"},
                    new Foo {Id = 3, Name = "Fooo"});

            modelBuilder.Entity<BarVW>(
                eb =>
                {
                    eb.HasKey(e => e.FooId);
                    eb.ToView("v_Bar");
                    eb.HasOne(e => e.Foo)
                        .WithOne(e => e.Bar)
                        .HasForeignKey<BarVW>(e => e.FooId);
                });
        }
    }
    
    public class FooDto
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public bool IsBar { get; set; }
    }

    internal static class Program
    {
        private static void Main()
        {
            using var context = new Context();

            context.Database.EnsureDeleted();
            context.Database.EnsureCreated();
            
            context.Database.ExecuteSqlRaw(
                @"CREATE VIEW [v_Bar] AS 
    select
        f.[Id] as [FooId],
        f.[Name] as [FooName]
    from
        [Foo] f
    where
        f.[Id] >= 1 and f.[Id] <= 2"
            );
            
            var config = new MapperConfiguration(
                cfg => cfg
                    .CreateMap<Foo, FooDto>()
                        .ForMember(dto => dto.IsBar, opt => opt.MapFrom(src => src.Bar != null)));
            
            var result = context.Foo
                .ProjectTo<FooDto>(config)
                .ToList();
            
            Debug.Assert(result.Count == 3);
            Debug.Assert(result.Count(dto => dto.IsBar) == 2);
        }
    }
}

为查询生成的 SQL 如下所示:

SELECT [f].[Id], CASE
    WHEN [v].[FooId] IS NOT NULL THEN CAST(1 AS bit)
    ELSE CAST(0 AS bit)
END AS [IsBar], [f].[Name]
FROM [Foo] AS [f]
LEFT JOIN [v_Bar] AS [v] ON [f].[Id] = [v].[FooId]

您可以使用.NET Fiddle运行示例代码。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM