Rumah > pembangunan bahagian belakang > C++ > Bolehkah Kod Teras EF untuk Memilih DTO Tersuai daripada Hartanah Kanak-Kanak Digunakan Semula Dengan Cekap?

Bolehkah Kod Teras EF untuk Memilih DTO Tersuai daripada Hartanah Kanak-Kanak Digunakan Semula Dengan Cekap?

Linda Hamilton
Lepaskan: 2025-01-23 19:48:14
asal
859 orang telah melayarinya

Can EF Core Code for Selecting Custom DTOs from Child Properties Be Reused Efficiently?

Bolehkah Kod untuk Memilih Objek DTO Tersuai untuk Harta Kanak-Kanak Digunakan Semula dengan Teras EF?

Teras Rangka Kerja Entiti membolehkan pertanyaan menggunakan ungkapan untuk menukar kepada objek DTO. Fungsi ini berfungsi dengan baik untuk objek dan mana-mana koleksi kanak-kanak, seperti yang ditunjukkan oleh Model yang disediakan:

public class Model
{
    public int ModelId { get; set; }
    public string ModelName { get; set; }

    public virtual ICollection<ChildModel> ChildModels { get; set; }

    // Other properties, collections, etc.

    public static Expression<Func<Model, ModelDto>> AsDto =>
        model => new ModelDto
        { 
            ModelId = model.ModelId,
            ModelName = model.ModelName,
            ChildModels = model.ChildModels.AsQueryable().Select(ChildModel.AsDto).ToList()
        };
}
Salin selepas log masuk

Dan Pertanyaan:

dbContext.Models.Where(m => SomeCriteria).Select(Model.AsDto).ToList();
Salin selepas log masuk

Walau bagaimanapun, persoalan timbul: bagaimana boleh serupa tingkah laku dapat dicapai untuk entiti kanak-kanak yang bukan koleksi? Sebagai contoh, jika kelas Model termasuk sifat:

public AnotherChildModel AnotherChildModel { get; set; }
Salin selepas log masuk

Penukaran boleh ditambahkan pada ungkapan:

public static Expression<Func<Model, ModelDto>> AsDto =>
    model => new ModelDto
    { 
        ModelId = model.ModelId,
        ModelName = model.ModelName,
        ChildModels = model.ChildModels.AsQueryable().Select(ChildModel.AsDto).ToList(),
        AnotherChildModel = new AnotherChildModelDto
        {
            AnotherChildModelId = model.AnotherChildModelId
        }
    };
Salin selepas log masuk

Tetapi mengulangi kod ini setiap kali model anak kedua perlu ditukar kepada objek DTO adalah tidak diingini. Adakah terdapat alternatif untuk menggunakan .Select() untuk satu entiti?

Penyelesaian

Beberapa perpustakaan menawarkan penyelesaian intuitif untuk masalah ini:

1. LINQKit:
Dengan LINQKit, operasi .Select() boleh digunakan pada entiti individu menggunakan ExpandableAttribute:

[Expandable(nameof(AsDtoImpl))]
public static ModelDto AsDto(Model model)
{
   _asDtoImpl ??= AsDtoImpl() .Compile();
   return _asDtoImpl(model);
}

private static Func<Model, ModelDto> _asDtoImpl;

private static Expression<Func<Model, ModelDto>> AsDtoImpl =>
    model => new ModelDto
    { 
        ModelId = model.ModelId,
        ModelName = model.ModelName,
        ChildModels = model.ChildModels.AsQueryable().Select(ChildModel.AsDto).ToList(),
        AnotherChildModel = new AnotherChildModelDto
        {
            AnotherChildModelId = model.AnotherChildModelId
        }
    };
Salin selepas log masuk

Pertanyaan kemudiannya boleh ditulis sebagai:

dbContext.Models
   .Where(m => SomeCriteria).Select(m => Model.AsDto(m))
   .AsExpandable()
   .ToList();
Salin selepas log masuk

2. NeinLinq:
Sama seperti LINQKit, NeinLinq menggunakan atribut InjectLambda:

[InjectLambda]
public static ModelDto AsDto(Model model)
{
   _asDto ??= AsDto() .Compile();
   return _asDto(model);
}

private static Func<Model, ModelDto> _asDto;

private static Expression<Func<Model, ModelDto>> AsDto =>
    model => new ModelDto
    { 
        ModelId = model.ModelId,
        ModelName = model.ModelName,
        ChildModels = model.ChildModels.AsQueryable().Select(ChildModel.AsDto).ToList(),
        AnotherChildModel = new AnotherChildModelDto
        {
            AnotherChildModelId = model.AnotherChildModelId
        }
    };
Salin selepas log masuk

Pertanyaan boleh diubah suai seperti berikut:

dbContext.Models
   .Where(m => SomeCriteria).Select(m => Model.AsDto(m))
   .ToInjectable()
   .ToList();
Salin selepas log masuk

3. DelegateDecompiler:
DelegateDecompiler menawarkan pendekatan ringkas menggunakan atribut Computed:

[Computed]
public static ModelDto AsDto(Model model)
  => new ModelDto
    { 
        ModelId = model.ModelId,
        ModelName = model.ModelName,
        ChildModels = model.ChildModels.AsQueryable().Select(ChildModel.AsDto).ToList(),
        AnotherChildModel = new AnotherChildModelDto
        {
            AnotherChildModelId = model.AnotherChildModelId
        }
    }
Salin selepas log masuk

Pertanyaan boleh dipermudahkan kepada:

dbContext.Models
   .Where(m => SomeCriteria).Select(m => Model.AsDto(m))
   .Decompile()
   .ToList();
Salin selepas log masuk

Setiap perpustakaan ini mencapai matlamat dengan mengubah suai pokok ungkapan sebelum pemprosesan Teras EF, dengan itu mengelakkan keperluan untuk kod berulang. Selain itu, ketiga-tiganya memerlukan panggilan untuk menyuntik IQueryProvider masing-masing.

Atas ialah kandungan terperinci Bolehkah Kod Teras EF untuk Memilih DTO Tersuai daripada Hartanah Kanak-Kanak Digunakan Semula Dengan Cekap?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan