Rumah >pembangunan bahagian belakang >C++ >Bolehkah Kod Teras EF untuk Memilih DTO Tersuai daripada Hartanah Kanak-Kanak Digunakan Semula Dengan Cekap?
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?
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!