حقن التبعية

تقنية تستخدم في هندسة البرمجيات

في هندسة البرمجيات، يعد حقن التبعية (بالإنجليزية: dependency injection)‏ تقنية يتلقى فيها كائن كائنات أخرى يعتمد عليها. تسمى هذه الكائنات الأخرى التبعيات.

في العلاقة النموذجية «باستخدام»[1] يسمى الكائن المتلقي عميل (بالإنجليزية: Client)‏ ويسمى الكائن الذي تم تمريره (أي تم حقنه) خدمة (بالإنجليزية: Service)‏. يمكن أن يكون الكود الذي ينقل الخدمة إلى العميل أنواعًا كثيرة ويسمى الحاقن. بدلاً من تحديد العميل للخدمة التي سيستخدمها، يخبر الحاقن العميل بالخدمة التي سيستخدمها. تشير «الحقن» إلى تمرير التبعية (خدمة) إلى الكائن (العميل) الذي قد يستخدمها.

الخدمة تصبح جزءاً من حالة العميل.[2] يعد تمرير الخدمة إلى العميل، بدلاً من السماح للعميل ببناء الخدمة أو العثور عليها، شرطاً أساسياً للنمط.

النية من حقن التبعية هو تحقيق فصل الاهتمامات الخاصة بالبناء واستخدام الكائنات. يمكن أن يؤدي ذلك إلى زيادة إمكانية القراءة وإعادة استخدام الكود.

حقن التبعية هو شكل من أشكال التقنية الأوسع لعكس التحكم. لا يجب على العميل الذي يريد استدعاء بعض الخدمات معرفة كيفية إنشاء هذه الخدمات. بدلاً من ذلك، يفوض العميل مسؤولية توفير خدماته للكود الخارجي (الحاقن). لا يُسمح للعميل باستدعاء كود الحاقن[3] ؛ الحاقن هو الذي يبني الخدمات. ثم يقوم الحاقن بحقن (تمرير) الخدمات في العميل التي قد تكون موجودة بالفعل أو قد يتم بناؤها بواسطة الحاقن. ثم يستخدم العميل الخدمات. هذا يعني أن العميل لا يحتاج إلى معرفة الحاقن، وكيفية إنشاء الخدمات، أو حتى الخدمات الفعلية التي يستخدمها. يحتاج العميل فقط إلى معرفة الواجهات الجوهرية للخدمات لأن هذه تحدد كيفية استخدام العميل للخدمات. وهذا يفصل مسؤولية «الاستخدام» (بالإنجليزية: "use")‏ عن مسؤولية «البناء» (بالإنجليزية: "construction")‏.

نوايا

حقن التبعية يحل مشاكل مثل:[4]

  • كيف يمكن أن يكون التطبيق أو الصنف مستقل عن كيفية إنشاء كائناته؟
  • كيف يمكن تحديد طريقة إنشاء الكائنات في ملفات تكوين منفصلة؟
  • كيف يمكن لتطبيق دعم تكوينات مختلفة؟

إنشاء كائنات مباشرة داخل الصنف أمر غير مرن لأنه يلزم الصنف بكائنات معينة ويجعل من المستحيل تغيير التمثيل الفوري بشكل مستقل عن الصنف (دون الحاجة إلى تغيير). يوقف الصنف عن إمكانية إعادة استخدامه.

إذا كانت هناك حاجة إلى كائنات أخرى، ويجعل من الصعب اختبار الصنف لأنه لا يمكن استبدال الكائنات الحقيقية بكائنات وهمية.

لم يعد الصنف مسؤول عن إنشاء الكائنات التي يتطلبها، ولا يتعين عليه تفويض إنشاء مثيل لكائن مصنع كما هو الحال في نمط تصميم.[5]
انظر أيضًا صنف UML ومخطط التسلسل أدناه.

نظرة عامة

حقن التبعية يفصل إنشاء تبعيات العميل عن سلوك العميل، والذي يسمح لتصاميم البرنامج أن تكون مقترنة بشكل متساهل[6] ومتابعة انعكاس التبعية ومبادئ المسؤولية الواحدة.[7] يتناقض بشكل مباشر مع نمط محدد مواقع الخدمة، والذي يسمح للعملاء بمعرفة النظام الذي يستخدمونه للعثور على التبعيات. الحقن، الوحدة الأساسية لحقن التبعية، ليست آلية جديدة أو مخصصة. يعمل بنفس الطريقة التي يعمل بها «تمرير المعلمة».[8]

بالإشارة إلى «تمرير المعلمة» حيث أن الحقن يحمل ضمناً إضافياً أنه يتم القيام به لعزل العميل عن التفاصيل.

إن الحقن يتعلق أيضًا بما يتحكم في المرور (وليس العميل) مطلقًا وهو مستقل عن كيفية تحقيق التمرير، سواء عن طريق تمرير مرجع أو قيمة يشمل حقن التبعية أربعة أدوار:

  • كائن (ات) خدمة service لاستخدامها.
  • كائن العميل الذي يعتمد على الخدمة (الخدمات) التي يستخدمها.
  • الواجهات التي تحدد كيفية استخدام العميل للخدمات.
  • الحاقن، وهو المسؤول عن إنشاء الخدمات وحقنها في العميل.

كمثال:

  • خدمة - سيارة كهربائية أو غاز أو هجينة أو ديزل.
  • العميل - السائق الذي يستخدم السيارة بنفس الطريقة بغض النظر عن المحرك.
  • واجهة - تلقائية، تضمن ألا يضطر السائق إلى فهم تفاصيل المحرك مثل التروس.
  • حاقن - الوالد الذي اشترى السيارة للطفل وقرر أي نوع.

أي كائن يمكن استخدامه يمكن اعتباره خدمة. يمكن اعتبار أي كائن يستخدم كائنات أخرى عميلاً . لا علاقة للأسماء بماهية الكائنات وكل شيء يتعلق بالدور الذي تلعبه الكائنات في أي حقن.

الواجهات هي الأنواع التي يتوقع العميل تبعياتها. المشكلة هي ما يجعلها متاحة. قد تكون بالفعل أنواع من الواجهة يتم تنفيذها بواسطة الخدمات ولكن قد تكون أيضًا أصناف مجردة أو حتى الخدمات نفسها على الرغم من أن هذا الأخير قد ينتهك DIP[9] ويضحي بفك الارتباط الديناميكي الذي يتيح الاختبار. يتطلب فقط ألا يعرف العميل هويته وبالتالي لا يتعامل معه على أنه ملموس، على سبيل المثال من خلال إنشائها أو توسيعها.

يجب ألا يكون لدى العميل معرفة ملموسة بالتنفيذ المحدد لتبعياته. يجب أن يعرف فقط اسم الواجهة وواجهة برمجة التطبيقات (بالإنجليزية: API)‏. ونتيجة لذلك، لن يحتاج العميل إلى التغيير حتى إذا تغير ما وراء الواجهة. ومع ذلك، إذا تمت إعادة هيكلة الواجهة من صنف إلى نوع واجهة (أو العكس) فسيحتاج العميل إلى إعادة التجميع[10] هذا مهم إذا تم نشر العميل والخدمات بشكل منفصل. هذا الاقتران المؤسف هو واحد لا يمكن حله بالتبعية. يقدم الحاقن الخدمات إلى العميل. في كثير من الأحيان، يقوم أيضًا بإنشاء العميل. قد يربط الحاقن معًا رسمًا بيانيًا معقدًا جدًا للكائن عن طريق معاملة كائن مثل العميل وبعد ذلك كخدمة لعميل آخر. قد يكون الحاقن. في الواقع العديد من الكائنات تعمل معًا ولكن قد لا يكون العميل. قد تتم الإشارة إلى الحاقن بأسماء أخرى مثل: المجمع، الموفر، الحاوية، المصنع، البناء، الربيع، كود البناء، أو الرئيسي.

يمكن تطبيق حقن التبعية كنظام، واحد يطلب من جميع الكائنات صنف البناء والسلوك. يمكن أن يؤدي الاعتماد على إطار عمل حقن التبعية بالقيام بالإنشاء إلى حظر استخدام الكلمة المفتاحية "new"، أو بشكل أقل صرامة، السماح بالبناء المباشر لعناصر القيمة فقط.[11][12][13][14]

التصنيف

يعد عكس التحكم (IoC) أكثر عمومية من حقن التبعية. ببساطة، يعني انعكاس التحكم السماح لكود آخر بالاتصال بك بدلاً من الإصرار على إجراء الاستدعاء. مثال على عكس التحكم بدون حقن التبعية هو نمط طريقة القالب. هنا، يتم تحقيق تعدد الأشكال من خلال التصنيف الفرعي، أي الميراث.[15] يطبق حقن التبعية انعكاس التحكم من خلال التكوين، لذلك غالبًا ما يكون مطابقًا لنمط الإستراتيجية، ولكن في حين أن نمط الإستراتيجية مخصص للتبعيات لتكون قابلة للتبادل طوال عمر الكائن، في حقن التبعية، قد يتم استخدام مثيل واحد فقط من التبعية[16] هذا لا يزال يحقق تعدد الأشكال، ولكن من خلال التفويض والتكوين.

أطر حقن التبعية

أطر التطبيقات مثل CDI وتنفيذها مثل Weld وسبرينغ و Guice واطار عمل بلاي و Salta و Glassfish HK2 و Dagger وإطار التوسعة المدار (MEF) تدعم حقن التبعية ولكن غير الزامية للقيام بحقن التبعية.[17][17][18]

مزايا

  • يسمح حقن التبعية للعميل بمرونة كونه قابلاً للتكوين. فقط سلوك العميل يتم إصلاحه. قد يقوم العميل بأي شيء يدعم الواجهة الجوهرية التي يتوقعها العميل.[19]
  • يمكن استخدام حقن التبعية لتجسيد تكوين النظام في ملفات التكوين، مما يسمح بإعادة تكوينالنظام دون إعادة التجميع. يمكن كتابة تكوينات منفصلة التي تتطلب تنفيذات مختلفة للمكوّن. وهذا يشمل، على سبيل المثال لا الحصر، الاختبار.[20]
  • نظرًا لأن حقن التبعية لا يتطلب أي تغيير في سلوك الكود، يمكن تطبيقه على الكود القديم باعتباره إعادة هيكلة الكود.. والنتيجة هي عملاء أكثر استقلالية وأكثر سهولة في اختبار الوحدة في العزل باستخدام(بالإنجليزية: البذرة )‏ كائنات وهمية التي تحاكي كائنات أخرى ليست تحت الاختبار. غالبًا ما تكون سهولة الاختبار هذه أول فائدة ملحوظة عند استخدام حقن التبعية.[21]
  • يسمح حقن التبعية للعميل بإزالة جميع ما هو معروف بالنسبة للتطبيق المحدد الذي يحتاج إلى استخدامه. هذا يساعد على عزل العميل من تأثير تغييرات التصميم والعيوب. يعزز قابلية إعادة الاستخدام والاختبار وقابلية الصيانة.[22]
  • الحد من كود المتداول في كائنات التطبيق، حيث أن الكل يعمل لتهيئة أو إعداد التبعيات يتم معالجته بواسطة مكون المزود.
  • يسمح حقن التبعية بالتطويرالمتزامن أو المستقل. يمكن لمطورين اثنين بشكل مستقل تطوير أصناف تستخدم بعضها البعض، كل ما تحتاجه هو فقط معرفة الواجهة التي ستتواصل من خلالها الاصناف. غالبًا ما يتم تطوير البرنامج المساعدة من خلال متاجر الجهات الخارجية التي لا تتحدث أبدًا مع المطورين الذين أنشؤوا المنتج الذي يستخدم البرنامج المساعد.[23]
  • يقلل حقن التبعية الاقتران بين صنف والتبعية.[24][25]

سلبيات

  • يعمل حقن التبعية على إنشاء عملاء يطلبون توفير تفاصيل التكوين عن طريق كود البناء. يمكن أن يكون هذا مرهقًا عندما تتوفر افتراضيات واضحة.[23]
  • يمكن أن يجعل حقن التبعية من الصعب تتبع الكود (قراءتها) لأنها تفصل بين السلوك والبناء. هذا يعني أنه يجب على المطورين الرجوع إلى المزيد من الملفات لمتابعة كيفية أداء النظام.[23]
  • يتم تنفيذأطر حقن التبعية مع انعكاس أو برمجة ديناميكية. يمكن أن يعيق هذا استخدام أتمتة IDE، مثل "العثور على مراجع" و "إظهار التسلسل الهرمي للاستدعاءات "" وإعادة البناء الآمنة[26]
  • عادة ما يتطلب حقن التبعية المزيد من جهود التطوير المسبق حيث لا يمكن للمرء أن يستدعي أن يكون شيئًا صحيحًا في الوقت والمكان المطلوبين ولكن يجب أن يطلب حقنه ثم التأكد من أنه تم حقنه.[27]
  • يجبر حقن التبعية التعقيد على الانتقال من الأصناف إلى الروابط بين الأصناف التي قد لا تكون دائمًا مرغوبة أو سهلة الإدارة.[28]
  • يمكن لحقن التبعية أن يشجع على الاعتماد على إطار حقن التبعية.[29][30]

هيكل (بناء)

مخطط الصنف UML ومخطط التتابع


نموذج لصنف UML ومخطط تسلسلي لنمط تصميم حقن التبعية.[31][31]

فيUML الرسم التخطيطي لصنف أعلاه، لا يقوم صنف Client الذي يتطلب كائنات ServiceA و ServiceB بإنشاء صنفيServiceA1 و ServiceB1 مباشرة. بدلاً من ذلك، يقوم صنف Injector بإنشاء الكائنات وحقنها في Client، مما يجعل Client مستقلًا عن كيفية إنشاء الكائنات (أي الأصناف الملموسة التي يتم إنشاء مثيل لها). يوضح الرسم التخطيطي لتسلسل UML تفاعلات وقت التشغيل يقوم كائن Injector بإنشاء كائنات ServiceA1 و ServiceB1. بعد ذلك، يقوم Injector بإنشاء كائن Client وإدخال كائنات ServiceA1 و ServiceB1.

أمثلة

بدون حقن التبعية

في مثال جافا التالي، يحتوي صنف على خدمة متغير عضو تمت تهيئته بواسطة مُنشئ.

يتحكم العميل في تنفيذ الخدمة المستخدمة ويتحكم في بنائها. في هذه الحالة، يُقال أن العميل لديه تبعية ذات كود صلب على. (بالإنجليزية: ExampleService)‏

// An example without dependency injectionpublic class Client {  // Internal reference to the service used by this client  private ExampleService service;  // Constructor  Client() {    // Specify a specific implementation in the constructor instead of using dependency injection    service = new ExampleService();  }  // Method within this client that uses the services  public String greet() {    return "Hello " + service.getName();  }}

يعتبر حقنالتبعية تقنية بديلة لتهيئة متغير العضو بدلاً من إنشاء كائن خدمة بشكل صريح كما هو موضح أعلاه. يمكننا تعديل هذا المثال

باستخدام التقنيات المختلفة الموضحة والموضحة في الأقسام الفرعية أدناه..

أنواع حقن التبعية

هناك ثلاث طرق على الأقل يمكن لكائن العميل تلقي مرجع لوحدة خارجية:[32]

حقن المنشئ
يتم توفير التبعيات من خلال مُنشئ صنف العميل..
حقن المعيّن
يكشف العميل عن طريقة الضبط التي يستخدمها الحاقن لحقن التبعية..
حقن الواجهة
توفر واجهة التبعية طريقة حاقن تضخ التبعية في أي عميل يتم تمريره إليها. يجب على العملاء تنفيذ واجهة تكشف عن طريقة تعيين يقبل التبعية.

أنواع أخرى

من الممكن أن يكون لأطر حقن التبعية أنواع أخرى من الحقن بخلاف تلك المذكورة أعلاه.[33] قد تستخدم أطر الاختبار أيضًا أنواعًا أخرى. بعض أطر الاختبار الحديثة لا تتطلب حتى أن يقبل العملاء بنشاط حقن التبعية وبالتالي جعل الكود القديم قابل للاختبار. على وجه الخصوص، في لغة جافا، من الممكن استخدام الانعكاس لجعل السمات الخاصة عامة عند الاختبار وبالتالي قبول الحقن عن طريق التعيين.[34]

لا تقدم بعض محاولات عكس التحكم الإزالة الكاملة للتبعية، ولكن بدلاً من ذلك ببساطة استبدال أحد أشكال التبعية بأخرى. كقاعدة عامة، إذا لم يتمكن المبرمج من النظر إلى شيء سوى كود العميل والإخبار عن إطار العمل المستخدم، عندئذٍ يكون لدى العميل اعتماد مرتبط بالكود الصلب على الإطار.

حقن المنشئ

تتطلب هذه الطريقة من العميل توفير معلمة في مُنشئ للتبعية.

// ConstructorClient(Service service) {  // Save the reference to the passed-in service inside this client  this.service = service;}

حقن المعيّن

تتطلب هذه الطريقة من العميل توفير طريقة ضبط للتبعية.

// Setter methodpublic void setService(Service service) {  // Save the reference to the passed-in service inside this client.  this.service = service;}

حقن الواجهة

ببساطة العميل هو الذي ينشر واجهة الدور لطرق تعيين تبعيات العميل. يمكن استخدامه لتحديد الطريقة التي يجب أن يتحدث بها الحاقن مع العميل عند حقن التبعيات.

// Service setter interface.public interface ServiceSetter {  public void setService(Service service);}// Client classpublic class Client implements ServiceSetter {  // Internal reference to the service used by this client.  private Service service;  // Set the service that this client is to use.  @Override  public void setService(Service service) {    this.service = service;  }}

مقارنة حقن المنشئ

يُفضل عند إنشاء جميع التبعيات أولاً لأنه يمكن استخدامها لضمان أن يكون كائن العميل دائمًا في حالة صالحة، على العكس من ذلك هو أن تكون بعض مراجع التبعية الخاصة به خالية (لم يتم تعيينها). ومع ذلك، من تلقاء نفسها، تفتقر إلى المرونة لتغيير تبعياتها في وقت لاحق. يمكن أن تكون هذه خطوة أولى نحو جعل العميل غير قابل للتغيير وبالتالي خيط آمن.

// ConstructorClient(Service service, Service otherService) {  if (service == null) {    throw new InvalidParameterException("service must not be null");  }  if (otherService == null) {    throw new InvalidParameterException("otherService must not be null");  }  // Save the service references inside this client  this.service = service;  this.otherService = otherService;}

مقارنة حقن المعيّن (الضبط)

يتطلب من العميل توفير طريقة ضبط لكل تبعية. وهذا يعطي حرية التلاعببحالة المراجع التبعية في أي وقت. يوفر هذا المرونة، ولكن إذا كان هناك أكثر من تبعية واحدة يجب حقنها، فمن الصعب على العميل التأكد من حقن جميع التبعيات قبل أن يتم توفير العميل للاستخدام.

// Set the service to be used by this clientpublic void setService(Service service) {  if (service == null) {    throw new InvalidParameterException("service must not be null");  }  this.service = service;}// Set the other service to be used by this clientpublic void setOtherService(Service otherService) {  if (otherService == null) {    throw new InvalidParameterException("otherService must not be null");  }  this.otherService = otherService;}

نظرًا لأن هذه الحقن يحدث بشكل مستقل، فلا توجد طريقة لمعرفة متى ينتهي الحاقن من توصيل العميل. يمكن ترك التبعية فارغة ببساطة عن طريق الفشل في استدعاء أداة ضبطها. هذا يفرض التحقق من اكتمال الحقن من وقت تجميع العميل حتى وقت استخدامه.

// Set the service to be used by this clientpublic void setService(Service service) {  this.service = service;}// Set the other service to be used by this clientpublic void setOtherService(Service otherService) {  this.otherService = otherService;}// Check the service references of this clientprivate void validateState() {  if (service == null) {    throw new IllegalStateException("service must not be null");  }  if (otherService == null) {    throw new IllegalStateException("otherService must not be null");  }}// Method that uses the service referencespublic void doSomething() {  validateState();  service.doYourThing();  otherService.doYourThing();}

مقارنة حقن الواجهة

تتمثل ميزة حقن الواجهة في أن التبعيات يمكن أن تكون جاهلة تمامًا بعملائها، ومع ذلك لا يزال بإمكانها تلقي مرجع إلى عميل جديد واستخدامه، وإرسال مرجع إلى الذات إلى العميل. بهذه الطريقة، تصبح التبعيات عن طريق الحقن. الفكرة هي أن طريقة الحقن (التي يمكن أن تكون مجرد طريقة ضبط كلاسيكية) يتم توفيرها من خلال واجهة.

لا يزال هناك حاجة إلى مجمعلتعريفالعميل وتبعياته. سيأخذ المُجمِّع مرجع إلى العميل، ويحولها إلى واجهة المحدد التي تعيّن تلك التبعية وتمريرها إلى كائن التبعية هذا الذي يتحول ويمرر مرجع إلى الذات إلى العميل..

لكي يكون لحقن الواجهة قيمة، التبعية يجب أن تقوم بشيء، بالإضافة إلى إرجاع مرجع لنفسها.. يمكن أن يكون هذا بمثابة مصنع أو مجمع فرعي لحل تبعيات أخرى، وبالتالي تجريد بعض التفاصيل من المجمع الرئيسي.. يمكن أن يكون عدًا مرجعيًا حتى تعرف التبعية عدد العملاء الذين يستخدمونه. إذا احتفظت التبعية بمجموعة من العملاء، فيمكنها لاحقًا حقنهم جميعًا بمثيل مختلف عن نفسها

// Service setter interface.public interface ServiceSetter {  public void setService(Service service);}// Client classpublic class Client implements ServiceSetter {  // Internal reference to the service used by this client.  private Service service;  // Set the service that this client is to use.  @Override  public void setService(Service service) {    this.service = service;  }}// Injector classpublic class ServiceInjector { Set<ServiceSetter> clients; public void inject(ServiceSetter client) {  clients.add(client);  client.setService(new ServiceFoo()); } public void switchToBar() {  for (Client client : clients) {   client.setService(new ServiceBar());  } }}// Service classespublic class ServiceFoo implements Service {}public class ServiceBar implements Service {}

تجميع الأمثلة

يعد التجميع يدويًا داخل الطريقة (الدالة) (بالإنجليزية: main)‏ الرئيسية إحدى طرق تنفيذ حقن التبعية .

public class Injector {  public static void main(String[] args) {    // Build the dependencies first    Service service = new ExampleService();    // Inject the service, constructor style    Client client = new Client(service);    // Use the objects    System.out.println(client.greet());  }}

ينشئ المثال أعلاه الرسم البياني للكائن يدويًا ثم يستدعيه عند نقطة واحدة لبدء العمل. من المهم ملاحظة أن هذا الحاقن ليس نقيًا. يستخدم أحد الكائنات التي يقوم ببنائها. لها علاقة بناء فقط مع ExampleService ولكنها تمزج البناء واستخدام العميل. لا ينبغي أن يكون هذا شائعًا. ومع ذلك، لا مفر منه. تمامًا مثل البرمجيات الموجهة للكائنات تحتاج إلى طريقة ثابتة غير موجهة للكائنات مثل ()main للبدء، يحتاج الرسم البياني للكائن المحقون بالتبعية إلى نقطة إدخال واحدة على الأقل (يفضل واحدة فقط) لبدء كل شيء.

قد لا يكون هذا البناء اليدوي في الطريقة الرئيسية مباشرأ وقد يتضمن استدعاء بناة أو مصانع أو أنماط بناء أخرى أيضًا. يمكن أن يكون هذا متقدمًا ومجرّدًا إلى حد ما. يتم تجاوز الخط من حقن التبعية اليدوي إلى حقن تبعية الإطار بمجرد أن كود البناء لم يعد مخصصًا للتطبيق بل أصبح عالميًا).[35]

يمكن لإطارات مثل سبرينغ إنشاء هذه الكائنات نفسها وتوصيلها معًا قبل إرجاع مرجع إلى العميل. يمكن نقل جميع الإشارة إلى ExampleService الملموسة من الكود إلى بيانات التكوين.

import org.springframework.beans.factory.BeanFactory;import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;public class Injector { public static void main(String[] args) {  // -- Assembling objects -- //  BeanFactory beanfactory = new ClassPathXmlApplicationContext("Beans.xml");  Client client = (Client) beanfactory.getBean("client");  // -- Using objects -- //  System.out.println(client.greet()); }}

تسمح إطارات مثل سبرينغ بتفريغ تفاصيل التجميع في ملفات التكوين يقوم هذا الكود (أعلاه) ببناء الكائنات توصيلها معًا وفقًا لـ Beans.xml (أدناه). لا تزال خدمة قيد الإنشاء على الرغم من أنها مذكورة أدناه فقط. يمكن تعريف الرسم البياني للكائن الطويل والمعقد بهذه الطريقة، والصنف الوحيد المذكور في الكود ستكون هي صنف طريقة الإدخال، وهي في هذه الحالة (دالة تحية) ()greet.

 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">  <bean id="service" class="ExampleService">  </bean>  <bean id="client" class="Client">    <constructor-arg value="service" />     </bean></beans>

في المثال أعلاه، لم يكن يتعين على العميل والخدمة الخضوع لأية تغييرات يوزودها إطار سبرينغ. يسمح لهم بالبقاء كبوجو(بالإنجليزية: POJOs )‏ بسيطة.[36][37][38] يوضح هذا كيف يمكن لspring أن يربط الخدمات والعملاء الذين يجهلون تمامًا وجودها. لا يمكن قول ذلك إذا تمت إضافة التعليقات التوضيحية الخاصة بSpring إلى الأصناف. من خلال منع التعليقات التوضيحية الاستدعاءات الخاصة بسبرينغ من الانتشار بين العديد من الأصناف، يظل النظام معتمدًا بشكل فضفاض فقط على سبرينغ.[29] قد يكون هذا مهمًا إذا كان النظام ينوي البقاء على قيد الحياة في سبرينغ. اختيار الحفاظ على POJOs صافية لا يأتي بدون تكلفة. بدلاً من بذل الجهد لتطوير ملفات التكوين المعقدة وصيانتها، من الممكن ببساطة استخدام التعليقات التوضيحية لوضع علامة على الأصناف) وترك سبرينغ يقوم بتتمة العمل. يمكن أن يكون حل التبعيات بسيطًا إذا اتبعت المصطلح عليه عادةً مثل المطابقة حسب النوع أو بالاسم. هذا هو اختيار الاصطلاح على التكوين.[39] ويمكن القول أيضًا أنه عند إعادة البناءإلى إطارآخر، فإن إزالة التعليقات التوضيحية المحددة للإطار ستكون جزءًا بسيطًا من المهمة

[40] ويتم الآن توحيد العديد من التعليقات التوضيحية بالحقن.[41][42]

import org.springframework.beans.factory.BeanFactory;import org.springframework.context.ApplicationContext;import org.springframework.context.annotation.AnnotationConfigApplicationContext;public class Injector { public static void main(String[] args) {  // Assemble the objects  BeanFactory beanfactory = new AnnotationConfigApplicationContext(MyConfiguration.class);  Client client = beanfactory.getBean(Client.class);  // Use the objects  System.out.println(client.greet()); }}
import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.ComponentScan;import org.springframework.context.annotation.Configuration;@ComponentScanpublic class MyConfiguration {  @Bean  public Client client(ExampleService service) {    return new Client(service);  }}
@Componentpublic class ExampleService {  public String getName() {    return "  }}

مقارنة التجميع

لا تختلف تنفيذات الحاقن المختلفة (المصانع، مواقع الخدمة، وحاويات حقن التبعية فيما يتعلق بحقن التبعية. ما يحدث فرقاً هو المكان الذي يسمح لهم باستخدامه. قم بنقل الاستدعاءات إلى المصنع أو محدد الخدمة خارج العميل وإلى الرئيسية (بالإنجليزية: main)‏ والمفاجئ الرئيسي يجعل حاوية حقن التبعية جيدة إلى حد ما. من خلال نقل كل المعرفة عن الحاقن إلى الخارج، يتم ترك عميل نظيف، خالٍ من المعرفة بالعالم الخارجي. ومع ذلك، يمكن اعتبار أي كائن يستخدم كائنات أخرى عميلاً. الكائن الذي يحتوي على main ليس استثناء. هذا الكائن الرئيسي لا يستخدم حقن التبعية. في الواقع استخدام نمط محدد مواقع الخدمة. لا يمكن تجنب ذلك لأنه يجب أن يتم اختيار تنفذيات الخدمة في مكان ما.

لا يؤدي إخارج التبعيات إلى ملفات التكوين إلى تغيير هذه الحقيقة. ما يجعل هذه حقاً جزءًا من التصميم الجيد هو أن محدد الخدمة لا ينتشر في جميع أنحاء قاعدة الكود. يقتصر على مكان واحد لكل تطبيق. هذا يترك بقية الكود الاساسي متاح لاستخدام حقن التبعية لعملاء (نظيفين).

نمط حقن التبعية

كانت الأمثلة حتى الآن أمثلة بسيطة للغاية حول إنشاء سلسلة (سلسلة نصية). ومع ذلك، يمكن أنه يكون نمط حقن التبعية أكثر فائدة عند إنشاء رسم بياني للكائن حيث تتواصل الكائنات عبر الرسائل.

ستستمر الكائنات التي تم إنشاؤها في main طوال عمر البرنامج. النمط النموذجي هو إنشاء الرسم البياني ثم استدعاء طريقة واحدة على كائن واحد لإرسال تدفق التحكم في الرسم البياني للكائن. تمامًا كما في main أن نقطة الدخول إلى الكود الثابت، فإن هذه الطريقة هي نقطة الدخول إلى الكود غير الثابت للتطبيقات.

.

public static void main(String[] args) throws IOException {  // Construction code.  Greeter greeter = new Greeter(System.out); // This may be many lines that connect many objects  // Behavior code.  greeter.greet(); // This is one call to one method on one object in the object graph}class Greeter {  public void greet() {    this.out.println("Hello world!");  }  public Greeter(PrintStream out) {    this.out = out;  }  private PrintStream out;}

مثال أنغولار جي إس

في إطار عمل أنغولار، هناك ثلاث طرق فقط يمكن للمكون (الكائن أو الوظيفة) الوصول مباشرةً إلى تبعياته:

  1. يمكن للمكون إنشاء التبعية، عادةً باستخدام عامل new.
  2. يمكن للمكون أن يبحث عن التبعية بالإشارة إلى متغير عام.
  3. يمكن أن ينتقل المكون إلى التبعية إليه عند الحاجة..

أول خيارين لإنشاء التبعيات أو البحث عنها ليسا الأمثل لأنهما يكودا التبعية بشكل ثابت للمكون.

. وهذا يجعل من الصعب، إن لم يكن من المستحيل، تعديل التبعيات. هذا يمثل مشكلة خاصة في الاختبارات، حيث من المستحسن في الغالب توفير تبعيات وهمية لعزل الاختبار.

الخيار الثالث هو الأكثر قابلية للتطبيق، لأنه يزيل مسؤولية تحديد موقع التبعية من المكون. يتم تسليم التبعية ببساطة إلى المكون.

function SomeClass(greeter) { this.greeter = greeter;}SomeClass.prototype.doSomething = function(name) { this.greeter.greet(name);}

في المثال أعلاه، لا تهتم SomeClass بإنشاء أو تحديد مكان تبعية المحيي، بل يتم ببساطة تسليم المحيي عند إنشاء مثيل لها. هذا أمر مرغوب فيه، ولكنه يضع مسؤولية الحصول على التبعية على الكود الذي يبني بعض الأصناف SomeClass.

لإدارة مسؤولية إنشاء التبعية، يحتوي كل تطبيق أنغولار جي إس، على حاقن. الحاقن هو محدد مواقع الخدمة المسؤول عن البناء والبحث عن التبعيات.

فيما يلي مثال على استخدام خدمة الحاقن:

// Provide the wiring information in a modulevar myModule = angular.module('myModule', []);// Teach the injector how to build a greeter service.// greeter is dependent on the $window service.// The greeter service is an object that// contains a greet method.myModule.factory('greeter', function($window) { return {  greet: function(text) {   $window.alert(text);  } };});

قم بإنشاء حاقن جديد يمكنه توفير المكونات المعرفة في نموذج myModule وطلب خدمة المحيي من الحاقن. (يتم ذلك تلقائيًا تلقائيًا عن طريق أنغولار جي اس بووتستراب (بالإنجليزية: AngularJS bootstrap)‏.

var injector = angular.injector(['myModule', 'ng']);var greeter = injector.get('greeter');

إن طلب التبعيات يحل مشكلة التكويد الصلب، ولكنه يعني أيضًا أنه يجب تمرير الحاقن في كل التطبيق. تمرير الحاقن يخالف قانون ديميتر. لعلاج هذا، نستخدم تدوينًا توضيحيًا في قوالب HTML الخاصة بنا، لتسليم مسؤولية إنشاء المكونات إلى الحاقن، كما في هذا المثال:

<div ng-controller="MyController"> <button ng-click="sayHello()">Hello</button></div>
function MyController($scope, greeter) { $scope.sayHello = function() {  greeter.greet('Hello World'); };}

عندما يقوم أنغلار جي إس، بتجميع HTML، فإنه يعالج توجيه ng-controller، والذي بدوره يطلب من الحاقن إنشاء مثيل من وحدة التحكم وتبعياتها

.

injector.instantiate(MyController);

تمثيل صنف، فيمكنه تلبية جميع تبعيات MyController دون أن تعرف وحدة التحكم عن الحاقن. يعلن كود التطبيق ببساطة التبعيات التي يحتاجها، دون الحاجة إلى التعامل مع الحاقن. هذا الإعداد لا يخالف قانون ديميتر.

سي شارب

مثال على حقن المنشئ، حقن المعيّن وحقن الواجهة على (#C)

using System;namespace DependencyInjection{    // An interface for the library    interface IGamepadFunctionality    {        String GetGamepadName();        void SetVibrationPower(float InPower);    }    // Concrete implementation of the xbox controller functionality    class XBoxGamepad : IGamepadFunctionality    {        readonly String GamepadName = "XBox Controller";        float VibrationPower = 1.0f;        public String GetGamepadName() => GamepadName;        public void SetVibrationPower(float InPower) => VibrationPower = Math.Clamp(InPower, 0.0f, 1.0f);    }    // Concrete implementation of the playstation controller functionality    class PlaystationJoystick : IGamepadFunctionality    {        readonly String ControllerName = "Playsation joystick";        float VibratingPower = 100.0f;        public String GetGamepadName() => ControllerName;        public void SetVibrationPower(float InPower) => VibratingPower = Math.Clamp(InPower * 100.0f, 0.0f, 100.0f);    }    // Concrete implementation of the steam controller functionality    class SteamController : IGamepadFunctionality    {        readonly String JoystickName = "Steam controller";        double Vibrating = 1.0;        public String GetGamepadName() => JoystickName;        public void SetVibrationPower(float InPower) => Vibrating = Convert.ToDouble(Math.Clamp(InPower, 0.0f, 1.0f));    }    // An interface for gamepad functionality injections    interface IGamepadFunctionalityInjector    {        void InjectFunctionality(IGamepadFunctionality InGamepadFunctionality);    }    class CGamepad : IGamepadFunctionalityInjector    {        IGamepadFunctionality _GamepadFunctionality;        public CGamepad()        {        }        // Constructor injection        public CGamepad(IGamepadFunctionality InGamepadFunctionality) => _GamepadFunctionality = InGamepadFunctionality;        // Setter injection        public void SetGamepadFunctionality(IGamepadFunctionality InGamepadFunctionality) => _GamepadFunctionality = InGamepadFunctionality;        // Interface injection        public void InjectFunctionality(IGamepadFunctionality InGamepadFunctionality) => _GamepadFunctionality = InGamepadFunctionality;        public void Showcase()        {            String Message = String.Format("We're using the {0} right now, do you want to change the vibrating power?\r\n", _GamepadFunctionality.GetGamepadName());            Console.WriteLine(Message);        }    }    enum EPlatforms: byte    {        Xbox,        Playstation,        Steam    }    class CGameEngine    {        EPlatforms _Platform;        CGamepad _Gamepad;        public void SetPlatform(EPlatforms InPlatform)        {            _Platform = InPlatform;            switch(_Platform)            {                case EPlatforms.Xbox:                    // injects dependency on XBoxGamepad class through Constructor Injection                    _Gamepad = new CGamepad(new XBoxGamepad());                    break;                case EPlatforms.Playstation:                    _Gamepad = new CGamepad();                    // injects dependency on PlaystationJoystick class through Setter Injection                    _Gamepad.SetGamepadFunctionality(new PlaystationJoystick());                    break;                case EPlatforms.Steam:                    _Gamepad = new CGamepad();                    // injects dependency on SteamController class through Interface Injection                    _Gamepad.InjectFunctionality(new SteamController());                    break;            }            _Gamepad.Showcase();        }    }    class Program    {        static void Main(string[] args)        {            Console.WriteLine("Hello World!");                       CGameEngine Engine = new CGameEngine();            Engine.SetPlatform(EPlatforms.Steam);            Engine.SetPlatform(EPlatforms.Xbox);            Engine.SetPlatform(EPlatforms.Playstation);        }    }}

ملحق: مسرد المصطلحات الإنجليزية

مَسرد المفردات وفق أقسام المقالة
المقدمة
هندسة البرمجياتsoftware engineering
يتلقى فيها كائن كائنات أخرىObject receives other objects
التبعياتdependencies
في العلاقة النموذجية "باستخدام"In the typical "using" relationship
عميل(بالإنجليزية: client)‏
المحقونinjected
خدمةservice
يمكن أن يكون الكود الذي ينقل الخدمة إلى العميل أنواعًا كثيرة ويسمى الحاقنcan be many kinds of things and is called the injector
الحاقنinjector
تمرير التبعيةpassing of a dependency
الكائن (العميل)client object
حالة العميلclient's state
بناءbuild
العثور عليهاfind the service
شرطاً أساسياً للنمطfundamental requirement of the pattern
النيةintent
حقن التبعيةdependency injection
فصل الاهتمامات[[:en:Separation of concerns|separation of concerns]]
زيادة إمكانية القراءةincrease readability
إعادة استخدام الكودcode reuse
شكل من أشكال التقنية الأوسع لعكس التحكمbroader technique of inversion of control
استدعاء بعض الخدماتcall some services
معرفة كيفية إنشاء هذه الخدماتto know how to construct those service
يفوضdelegates
للكود الخارجي (الحاقن)external code (the injector)
لا يُسمح للعميل باستدعاء كود الحاقنThe client is not allowed to call the injector code
الحاقن هو الذي يبني الخدماتit is the injector that constructs the services
تمريرpasses
قد يتم بناؤها بواسطة الحاقنbe constructed by the injector
كيفية إنشاء الخدماتhow to construct the services
حتى الخدمات الفعلية التي يستخدمهاeven which actual services it is using
معرفة الواجهات الجوهرية للخدماتintrinsic interfaces of the services
يفصل مسؤولية "الاستخدام""responsibility of "use
مسؤولية "البناء""responsibility of "construction
نوايا
التطبيقapplication
الصنفclass
مستقل عن كيفية إنشاء كائناتهbe independent of how its objects are created
طريقة إنشاء الكائناتway objects are created
في ملفات تكوين منفصلة
reated be specified in separate configuration files
تكويناتconfigurations
دعمsupport
إنشاء كائنات مباشرة داخل الصنفCreating objects directly within the class
التمثيلinstantiation
يوقف الصنف عن إمكانية إعادة استخدامهIt stops the class from being reusable
ويجعل من الصعب اختبار الصنف(بالإنجليزية: it makes the class hard to test class)‏
لا يمكن استبدال الكائنات الحقيقية بكائنات وهميةbecause real objects can not be replaced with mock objects
تفويضdelegate
مثيلinstantiation
لكائن مصنعto a factory object
نمط تصميمAbstract Factory
نظرة عامة
إنشاء تبعيات العميلcreation of a client's dependencies
سلوك العميلclient's behavior
لتصاميم البرنامج أن تكون مقترنة بشكل متساهلprogram designs to be loosely coupled
متابعة انعكاس التبعيةto follow the dependency inversion
مبادئ المسؤولية واحدةsingle responsibility principles
يتناقض بشكل مباشر مع نمط محدد مواقع الخدمةIt directly contrasts with the service locator pattern
للعملاءclients
بمعرفة النظام الذي يستخدمونه للعثور على التبعياتto know about the system they use to find dependencies
الحقنinjection
الوحدة الأساسية لحقن التبعيةthe basic unit of dependency injection
تمرير المعلمة"parameter passing"
يحملcarries
ضمناً إضافياًthe added implication
يتم القيام به لعزل العميل عن التفاصيلto isolate the client from details
يخالف قانون ديميترLaw of Demeter.

انظر أيضًا

المراجع

روابط خارجية

🔥 Top keywords: الصفحة الرئيسةخاص:بحثالشيهانة العزازتصنيف:أفلام إثارة جنسية أمريكيةغازي القصيبيتصنيف:أفلام إثارة جنسيةملف:Arabic Wikipedia Logo Gaza (3).svgباية حسينصالح بن عبد الله العزازمشعل الأحمد الجابر الصباحمجزرة مستشفى المعمدانييوتيوبتصنيف:ممثلات إباحيات أمريكياتكاليدونيا الجديدةالصفحة الرئيسيةمتلازمة XXXXالقمة العربية 2024عملية طوفان الأقصىميا خليفةكليوباترامحمد نور (لاعب كرة قدم سعودي)البيت بيتي (مسلسل)عبد العزيز بن محمد بن عياف آل مقرنحمد بن عيسى بن سلمان آل خليفةعبد اللطيف عبد الحميدعادل إمامعبد القادر الجيلانيصلاة الاستخارةكريستيانو رونالدومحمدالدوري الإنجليزي الممتازمحمد بن سلمان آل سعودتصنيف:أسماء إناث عربيةسكسي سكسي لافرواتسابسلوفاكياأسماء جلالدوري أبطال أوروباأحمد عبد الله الأحمد الصباح