MVVM - निर्भरता इंजेक्शन
इस अध्याय में, हम संक्षेप में निर्भरता इंजेक्शन के बारे में चर्चा करेंगे। हमने पहले ही एक दूसरे से डेटा बाइंडिंग डिकॉउन्स व्यूज़ और व्यूमॉडल कवर कर लिए हैं, जो उन्हें स्पष्ट रूप से यह जानने के बिना संवाद करने की अनुमति देता है कि संचार के दूसरे छोर पर क्या चल रहा है।
अब हमें ग्राहक सेवाओं से अपने ViewModel को हटाने के लिए कुछ इसी तरह की आवश्यकता है।
ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग के शुरुआती दिनों में, डेवलपर्स ने अनुप्रयोगों में कक्षाओं के उदाहरण बनाने और पुनर्प्राप्त करने के मुद्दे का सामना किया है। इस समस्या के लिए विभिन्न समाधान प्रस्तावित किए गए हैं।
पिछले कुछ वर्षों से, निर्भरता इंजेक्शन और नियंत्रण के उलट (IoC) ने डेवलपर्स के बीच लोकप्रियता हासिल की है और कुछ पुराने समाधानों जैसे सिंग्लटन पैटर्न पर वरीयता प्राप्त की है।
निर्भरता इंजेक्शन / IoC कंटेनर
IoC और निर्भरता इंजेक्शन दो डिज़ाइन पैटर्न हैं जो निकटता से संबंधित हैं और कंटेनर मूल रूप से इन्फ्रास्ट्रक्चर कोड का एक हिस्सा है जो उन दोनों पैटर्न को आपके साथ करता है।
IoC पैटर्न निर्माण के लिए जिम्मेदारी सौंपने के बारे में है और निर्भरता इंजेक्शन पैटर्न उस वस्तु पर निर्भरता प्रदान करने के बारे में है जो पहले से ही निर्माण की गई है।
उन्हें निर्माण के लिए दो-चरण के दृष्टिकोण के रूप में माना जा सकता है। जब आप एक कंटेनर का उपयोग करते हैं, तो कंटेनर कई जिम्मेदारियां लेता है जो निम्नानुसार हैं -
- यह पूछे जाने पर एक वस्तु का निर्माण करता है।
- कंटेनर यह निर्धारित करेगा कि वह वस्तु किस पर निर्भर करती है।
- उन निर्भरताओं का निर्माण।
- निर्मित की जा रही वस्तु में उन्हें इंजेक्ट करना।
- पुनरावर्ती प्रक्रिया।
आइए नजर डालते हैं कि ViewModels और क्लाइंट सेवाओं के बीच डिकम्पलिंग को तोड़ने के लिए हम निर्भरता इंजेक्शन का उपयोग कैसे कर सकते हैं। हम उस से संबंधित निर्भरता इंजेक्शन का उपयोग करके AddEditCustomerViewModel फॉर्म सेविंग वायर को वायर करेंगे।
पहले हमें अपने प्रोजेक्ट में सर्विसेज फ़ोल्डर में एक नया इंटरफ़ेस बनाने की आवश्यकता है। यदि आपके पास अपनी परियोजना में एक सेवा फ़ोल्डर नहीं है, तो इसे पहले बनाएं और सेवा फ़ोल्डर में निम्न इंटरफ़ेस जोड़ें।
using MVVMHierarchiesDemo.Model;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MVVMHierarchiesDemo.Services {
public interface ICustomersRepository {
Task<List<Customer>> GetCustomersAsync();
Task<Customer> GetCustomerAsync(Guid id);
Task<Customer> AddCustomerAsync(Customer customer);
Task<Customer> UpdateCustomerAsync(Customer customer);
Task DeleteCustomerAsync(Guid customerId);
}
}
निम्नलिखित ICustomersRepository का कार्यान्वयन है।
using MVVMHierarchiesDemo.Model;
using System;
using System.Collections.Generic;
using System.Linq; using System.Text;
using System.Threading.Tasks;
namespace MVVMHierarchiesDemo.Services {
public class CustomersRepository : ICustomersRepository {
ZzaDbContext _context = new ZzaDbContext();
public Task<List<Customer>> GetCustomersAsync() {
return _context.Customers.ToListAsync();
}
public Task<Customer> GetCustomerAsync(Guid id) {
return _context.Customers.FirstOrDefaultAsync(c => c.Id == id);
}
public async Task<Customer> AddCustomerAsync(Customer customer){
_context.Customers.Add(customer);
await _context.SaveChangesAsync();
return customer;
}
public async Task<Customer> UpdateCustomerAsync(Customer customer) {
if (!_context.Customers.Local.Any(c => c.Id == customer.Id)) {
_context.Customers.Attach(customer);
}
_context.Entry(customer).State = EntityState.Modified;
await _context.SaveChangesAsync();
return customer;
}
public async Task DeleteCustomerAsync(Guid customerId) {
var customer = _context.Customers.FirstOrDefault(c => c.Id == customerId);
if (customer != null) {
_context.Customers.Remove(customer);
}
await _context.SaveChangesAsync();
}
}
}
Save हैंडलिंग करने का सरल तरीका AddEditCustomerViewModel में ICustomersRepository का एक नया उदाहरण जोड़ना और AddEditCustomerViewModel और CustomerListViewModelor को अधिभारित करना है।
private ICustomersRepository _repo;
public AddEditCustomerViewModel(ICustomersRepository repo) {
_repo = repo;
CancelCommand = new MyIcommand(OnCancel);
SaveCommand = new MyIcommand(OnSave, CanSave);
}
निम्न कोड में दिखाए अनुसार OnSave विधि को अपडेट करें।
private async void OnSave() {
UpdateCustomer(Customer, _editingCustomer);
if (EditMode)
await _repo.UpdateCustomerAsync(_editingCustomer);
else
await _repo.AddCustomerAsync(_editingCustomer);
Done();
}
private void UpdateCustomer(SimpleEditableCustomer source, Customer target) {
target.FirstName = source.FirstName;
target.LastName = source.LastName;
target.Phone = source.Phone;
target.Email = source.Email;
}
निम्नलिखित पूरा AddEditCustomerViewModel है।
using MVVMHierarchiesDemo.Model;
using MVVMHierarchiesDemo.Services;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MVVMHierarchiesDemo.ViewModel {
class AddEditCustomerViewModel : BindableBase {
private ICustomersRepository _repo;
public AddEditCustomerViewModel(ICustomersRepository repo) {
_repo = repo;
CancelCommand = new MyIcommand(OnCancel);
SaveCommand = new MyIcommand(OnSave, CanSave);
}
private bool _EditMode;
public bool EditMode {
get { return _EditMode; }
set { SetProperty(ref _EditMode, value); }
}
private SimpleEditableCustomer _Customer;
public SimpleEditableCustomer Customer {
get { return _Customer; }
set { SetProperty(ref _Customer, value); }
}
private Customer _editingCustomer = null;
public void SetCustomer(Customer cust) {
_editingCustomer = cust;
if (Customer != null) Customer.ErrorsChanged -= RaiseCanExecuteChanged;
Customer = new SimpleEditableCustomer();
Customer.ErrorsChanged += RaiseCanExecuteChanged;
CopyCustomer(cust, Customer);
}
private void RaiseCanExecuteChanged(object sender, EventArgs e) {
SaveCommand.RaiseCanExecuteChanged();
}
public MyIcommand CancelCommand { get; private set; }
public MyIcommand SaveCommand { get; private set; }
public event Action Done = delegate { };
private void OnCancel() {
Done();
}
private async void OnSave() {
UpdateCustomer(Customer, _editingCustomer);
if (EditMode)
await _repo.UpdateCustomerAsync(_editingCustomer);
else
await _repo.AddCustomerAsync(_editingCustomer);
Done();
}
private void UpdateCustomer(SimpleEditableCustomer source, Customer target) {
target.FirstName = source.FirstName;
target.LastName = source.LastName;
target.Phone = source.Phone;
target.Email = source.Email;
}
private bool CanSave() {
return !Customer.HasErrors;
}
private void CopyCustomer(Customer source, SimpleEditableCustomer target) {
target.Id = source.Id;
if (EditMode) {
target.FirstName = source.FirstName;
target.LastName = source.LastName;
target.Phone = source.Phone;
target.Email = source.Email;
}
}
}
}
जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो आपको समान आउटपुट दिखाई देगा लेकिन अब ViewModels अधिक शिथिल रूप से डिकॉउल्ड किए गए हैं।
जब आप ग्राहक जोड़ें बटन दबाते हैं, तो आपको निम्न दृश्य दिखाई देगा। जब उपयोगकर्ता किसी भी क्षेत्र को खाली छोड़ देता है, तो यह हाइलाइट हो जाएगा और सेव बटन अक्षम हो जाएगा।