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
Linda Hamiltonasal
2025-01-23 19:48:14860semak imbas

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()
        };
}

Dan Pertanyaan:

dbContext.Models.Where(m => SomeCriteria).Select(Model.AsDto).ToList();

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; }

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
        }
    };

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
        }
    };

Pertanyaan kemudiannya boleh ditulis sebagai:

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

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
        }
    };

Pertanyaan boleh diubah suai seperti berikut:

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

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
        }
    }

Pertanyaan boleh dipermudahkan kepada:

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

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!

Kenyataan:
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