INTRODUCTION À L INJECTION DE DÉPENDANCES EN JAVA



Documents pareils
Environnements de développement (intégrés)

Pour signifier qu'une classe fille hérite d'une classe mère, on utilise le mot clé extends class fille extends mère

Types d applications pour la persistance. Outils de développement. Base de données préexistante? 3 modèles. Variantes avec passerelles

Programmer en JAVA. par Tama

Remote Method Invocation (RMI)

TP1 : Initiation à Java et Eclipse

TP1 : Initiation à Java et Eclipse

Quelques patterns pour la persistance des objets avec DAO DAO. Principe de base. Utilité des DTOs. Le modèle de conception DTO (Data Transfer Object)

Auto-évaluation Programmation en Java

Projet Active Object

TD/TP PAC - Programmation n 3

Formation Webase 5. Formation Webase 5. Ses secrets, de l architecture MVC à l application Web. Adrien Grand <jpountz@via.ecp.fr> Centrale Réseaux

Générer du code à partir d une description de haut niveau

Encapsulation. L'encapsulation consiste à rendre les membres d'un objet plus ou moins visibles pour les autres objets.

Ce document décrit la démarche à suivre pour installer les outils de développement et compiler le projet TANAGRA.

Java Licence Professionnelle Cours 7 : Classes et méthodes abstraites

Langage et Concepts de Programmation Objet. 1 Attributs et Méthodes d instance ou de classe. Travaux Dirigés no2

as Architecture des Systèmes d Information

Initiation à JAVA et à la programmation objet.

Création d objet imbriqué sous PowerShell.

Formation, Audit, Conseil, Développement, UX WinRT Silverlight WPF Android Windows Phone

Tp 1 correction. Structures de données (IF2)

Cours 1: Java et les objets

TD/TP PAC - Programmation n 3

LMI 2. Programmation Orientée Objet POO - Cours 9. Said Jabbour. jabbour@cril.univ-artois.fr

Info0604 Programmation multi-threadée. Cours 5. Programmation multi-threadée en Java

Structure d un programme et Compilation Notions de classe et d objet Syntaxe

IFT287 Exploitation de base de données relationnelles et orientées objet. Laboratoire Mon premier programme Java en Eclipse

Premiers Pas en Programmation Objet : les Classes et les Objets

TD/TP 1 Introduction au SDK d Android

25 mars. Tutoriel sur Laravel. Préparé par : Lydiane Beaulne-Bélisle. Ceci est un tutorial qui montre comment débuter avec le Framework PHP Laravel.

Héritage presque multiple en Java (1/2)

Présentation. Au programme. Fonctionnement. A l issue de ce module vous devriez...

Java Licence professionnelle CISII,

Java 7 Les fondamentaux du langage Java

TP n 2 Concepts de la programmation Objets Master 1 mention IL, semestre 2 Le type Abstrait Pile

TP1. Outils Java Eléments de correction

Institut Supérieur de Gestion. Cours pour 3 ème LFIG. Java Enterprise Edition Introduction Bayoudhi Chaouki

Systeme d'exploitation

Université de Bangui. Modélisons en UML

Table des matières L INTEGRATION DE SAS AVEC JMP. Les échanges de données entre SAS et JMP, en mode déconnecté. Dans JMP

Dossier. Développer en Java sur téléphone mobile. Benjamin Damécourt UFR SITEC Master 2 EESC 11 janvier 2012

Remote Method Invocation Les classes implémentant Serializable

Bases Java - Eclipse / Netbeans

La persistance des données dans les applications : DAO, JPA, Hibernate... COMPIL 2010 francois.jannin@inp-toulouse.fr 1

Devenez un véritable développeur web en 3 mois!

INITIATION AU LANGAGE JAVA

Tutoriel BlueJ. Michael Kölling Mærsk Institute University of Southern Denmark. Version 1.4 fr-2 par le groupe Sigma 1 pour BlueJ Version 1.2.

Utiliser Java sans BlueJ

Programmation Orientée Objet en C#

< Atelier 1 /> Démarrer une application web

RMI le langage Java XII-1 JMF

Messagerie & Groupeware. augmentez l expertise de votre capital humain

Introduction à Java. Matthieu Herrb CNRS-LAAS. Mars

Sommaire Introduction... 3 Le but du projet... 3 Les moyens utilisés... 3 Informations sur le client FTP... 4 Pourquoi une version Linux et

Langage Java. Classe de première SI

DA MOTA Anthony - Comparaison de technologies : PhoneGap VS Cordova

Objets et Programmation. origine des langages orientés-objet

NFP 121. Java et les Threads. Présentation : Thierry Escalarasse Mai 2007

Eclipse atelier Java

P r ob lé m a t iq u e d e la g é n é r icit é. Pr in cip e d e la g é n é r icit é e n Ja v a ( 1 /3 )

Méthode de Test. Pour WIKIROUTE. Rapport concernant les méthodes de tests à mettre en place pour assurer la fiabilité de notre projet annuel.

Généralités sur le Langage Java et éléments syntaxiques.

Programmation Objet Java Correction

RAPPELS SUR LES METHODES HERITEES DE LA CLASSE RACINE Object ET LEUR SPECIALISATION (i.e. REDEFINITION)

Corrigé des exercices sur les références

Programmation par les Objets en Java

CRM pour le Service clients et l Assistance technique

Compte Rendu d intégration d application

Annexe : La Programmation Informatique

Java au cœur de la base de données Oracle

MANUEL UTILISATEUR. Application 4trip

Sélection d un moteur de recherche pour intranet : Les sept points à prendre en compte

Synchro et Threads Java TM

Page 1 sur 5 TP3. Thèmes du TP : l la classe Object. l Vector<T> l tutorial Interfaces. l Stack<T>

Anne Tasso. Java. Le livre de. premier langage. 10 e édition. Avec 109 exercices corrigés. Groupe Eyrolles, , ISBN :

Procédure d installation Smart Map 3

Freeway 7. Nouvelles fonctionnalités

Apprendre la Programmation Orientée Objet avec le langage Java (avec exercices pratiques et corrigés)

Java c est quoi? Java. Java. Java : Principe de fonctionnement 31/01/ Vue générale 2 - Mon premier programme 3 - Types de Programme Java

Processus d Informatisation

Threads. Threads. USTL routier 1

Gestion distribuée (par sockets) de banque en Java

WINDOWS SHAREPOINT SERVICES 2007

Une introduction à Java

basée sur le cours de Bertrand Legal, maître de conférences à l ENSEIRB Olivier Augereau Formation UML

Importer une bibliographie au format «texte» dans Zotero

Prénom : Matricule : Sigle et titre du cours Groupe Trimestre INF1101 Algorithmes et structures de données Tous H2004. Loc Jeudi 29/4/2004

Cours en ligne Développement Java pour le web

Projet de Veille Technologique

V11. Release 1. Nouveaux appareils. Nouvelles fonctionnalités. Plus de flexibilité.

Introduction à Eclipse

Outil de démonstration : Application PassNFC

TD Objets distribués n 3 : Windows XP et Visual Studio.NET. Introduction à.net Remoting

Guide d utilisation. Version 1.1

Automatisation de l administration système

Chapitre 2. Classes et objets

TP, première séquence d exercices.

Cette application développée en C# va récupérer un certain nombre d informations en ligne fournies par la ville de Paris :

Java DataBaseConnectivity

Transcription:

INTRODUCTION À L INJECTION DE DÉPENDANCES EN JAVA Andr0 29 octobre 2015

Table des matières 1 Introduction 5 2 Qu est ce que l injection de dépendances? 7 2.1 Mais une dépendance, c est quoi?......................... 7 2.2 Donnons une définition à l injection de dépendances................ 8 3 Présentation de bibliothèques d injection de dépendances 9 4 Utiliser l injection de dépendances 11 4.1 Déclarer une dépendance.............................. 11 4.2 Satisfaire les dépendances............................. 12 4.2.1 Google Guice................................. 12 4.2.2 Dagger.................................... 13 5 Des fonctionnalités supplémentaires 15 5.1 Nommer les injections............................... 15 5.2 Ses propres annotations pour nommer une injection................ 16 5.3 Fournir des instances différentes.......................... 17 6 Conclusion 19 6.1 Remerciements................................... 19 3

1 Introduction L injection de dépendances demande des compétences confirmées dans la programmation orientée objet. Dans la mesure du possible, les bases seront expliquées mais certaines notions seront considérées comme acquises, notamment l utilisation des interfaces. Les prérequis pour ce tutoriel sont : la programmation orientée objet ; des connaissances basiques en java ; un gestionnaire de dépendances comme Maven est un plus si vous désirez exécuter les exemples. [[information]] Tous les codes sources de ce tutoriel sont disponibles sur ce projet GitHub afin que vous puissiez consulter des exemples fonctionnels directement exécutables dans un terminal. 5

2 Qu est ce que l injection de dépendances? 2.1 Mais une dépendance, c est quoi? Un développeur qui se soucie de la qualité logicielle et de la maintenance de son application va tenter d élaborer une architecture logicielle adaptée. Il existe plusieurs solutions, parmi lesquelles on trouve l injection de dépendances. Reste à savoir ce qu est une dépendance Pour illustrer ce concept, prenons un cas simple : une classe voudrait invoquer une ou plusieurs méthodes d une autre classe. Une implémentation pourrait correspondre au code ci-dessous. public class PizzaManagerImpl implements PizzaManager { private final PizzaDao pizzadao; public PizzaManagerImpl(PizzaDao pizzadao) { this.pizzadao = pizzadao; @Override public List<Pizza> menu() { return pizzadao.getall(); La responsabilité de PizzaManagerImpl est de renvoyer toutes les pizzas du menu d une pizzeria. Pour ce faire, cette classe a besoin d une autre classe pour faire l intermédiaire entre la base de donnée et elle-même (ceci dans un souci d architecturer convenablement le code). La classe possède donc une dépendance vers la classe PizzaDao. Cette dépendance entraine un couplage fort puisque le manager ne peut pas se passer du dao des pizzas. [[information]] Le dao est un concept bien connu dans les applications, notamment celles qui communiquent avec une base de données. Sa responsabilité est d abstraire les opérations usuelles sur une base de données (ou tout autre système de persistance des données) pour une ressource donnée. Ainsi, partout ailleurs dans l application, il suffit d utiliser le dao adéquat. Dans cet exemple, le dao est consacré à la récupération d une liste de pizzas. Pour éviter de rajouter la responsabilité d instanciation des classes et pour réduire les dépendances entre les classes, l injection de dépendances se place comme une solution adaptée. Raison pour laquelle ce patron de conception fait l objet de ce tutoriel. 7

2 Qu est ce que l injection de dépendances? 2.2 Donnons une définition à l injection de dépendances Pour être efficaces, prenons une définition mûrement réfléchie par la communauté de Wikipedia comme point de départ : L injection de dépendances (Dependency Injection) est un mécanisme qui permet d implémenter le principe de l inversion de contrôle. Il consiste à créer dynamiquement (injecter) les dépendances entre les différentes classes en s appuyant sur une description (fichier de configuration ou métadonnées) ou de manière programmatique. Ainsi les dépendances entre composants logiciels ne sont plus exprimées dans le code de manière statique mais déterminées dynamiquement à l exécution. Source :Version de l article Injection de dépendances de Wikipedia datée du 4 juin 2014 Cette définition utilise des termes génériques pour toucher le plus de domaines possible. Dans le cas ci-présent, le tutoriel aborde l injection de dépendances dans le langage java et par ses bibliothèques. Il est possible d adapter un poil cette définition comme suit : L injection de dépendances (Dependency injection) est un mécanisme qui permet d implémenter le principe de l inversion de contrôle (souvent abrégé par l acronyme IoC). Elle consiste à injecter dynamiquement les dépendances pour différentes classes en s appuyant sur un ou plusieurs modules. Ainsi, les dépendances entre des classes d implémentation et une interface ne sont plus exprimées dans le code de manière statique mais déterminées dynamiquement. Il se dégage de cette définition un concept important : pour pouvoir utiliser l injection de dépendances, il est obligatoire de penser l architecture de son logiciel avec des interfaces et de coupler leurs implémentations grâce à des modules. En plus de permettre l injection, c est une bonne pratique de conception pour éviter des couplages forts entre différents modules d un logiciel. Bien entendu, l injection de dépendances ne se limite pas à l injection par interface. Il existe 4 types d injections de dépendances : injection par constructeur ; injection par interface ; injection par mutateur ; injection par champs. Toutes ces injections ne seront pas abordées puisque les bibliothèques Java ne supportent pas tous les types d injection. Si vous êtes intéressés par une injection spécifique, des nombreuses ressources sont disponibles sur internet. 8

3 Présentation de bibliothèques d injection de dépendances Des bibliothèques sur l injection de dépendances, il en existe des tas dans quasiment tous les langages. Par conséquent, il serait difficile de toutes les couvrir. Sachant toutefois que les différences entre ces bibliothèques sont minimes et se jouent plutôt sur la syntaxe qu au niveau des performances vis-à-vis du compilateur par exemple. [[information]] Notez que certaines bibliothèques sont plus performantes que d autres dans des utilisations bien précises. Ce niveau de performance ne sera pas abordé dans le cadre de ce tutoriel puisqu il s agit là d un cadre plus avancé de l injection de dépendances. Devant les innombrables bibliothèques traitant de notre sujet, un premier parti portera sur le langage. Les exemples seront illustrés par le java, étant donné son aspect pédagogique et qu il s agit d un des langages les plus connus par les développeurs à travers le monde à ce jour (où du moins, un des plus faciles à prendre en main). Google Guice : À l époque, cette bibliothèque avait été développée par Google pour tirer parti de toute la puissance de Java, notamment sur les annotations et les génériques. Ceci pour faire face à une alternative vieillissante, Spring. Encore aujourd hui, c est l une des solutions les plus utilisées dans les applications, par son développement actif et sa communauté qui a mené à d autres projets comme Robo Guice. Dagger : Développé par une boite très active dans la conception de bibliothèques opensources, Square, Dagger ne se distingue pas par son utilisation, hormis sur quelques points, mais par son support natif d Android et par ses meilleures performances. Cependant, cette bibliothèque souffre de limitations techniques lorsque des choses plus complexes sont nécessaires. Les différences sont minimes et se jouent à pas grand chose. Raison pour laquelle aucune distinction ne sera faite dans les exemples de la section suivante. Si un choix doit être fait pour l une de ces librairies, tout est une question de besoin. Si un conseil devait vous être donné, ce serait le suivant : si votre projet est conséquent, privilégiez Google Guice. Sinon, privilégiez la légèreté et la facilité de prise en main de Dagger. 9

4 Utiliser l injection de dépendances Pour illustrer l injection de dépendances, Google Guice sera utilisé pour sa maturité et sa fiabilité. Pour consulter le code source compilable et testable, rendez-vous sur le dépôt git du tutoriel. 4.1 Déclarer une dépendance Google Guice construit les instances des classes d une application et satisfait leurs dépendances. Il utilise les annotations du package com.google.inject ou, plus standard, javax.inject. Leurs annotations Inject identifient les constructeurs ou les attributs à injecter. Utiliser l annotation @Inject sur le constructeur ou l attribut d une classe permet de créer une nouvelle instance des paramètres du constructeur ou de l attribut. Lorsqu une nouvelle instance est requise par l application, Google Guice satisfera les dépendances nécessaires en invoquant les constructeurs voulus. Donc, tout se passe à l exécution! Pour illustrer l injection sur le constructeur, PizzaManagerImpl désire une instance pour pizzadao du type PizzaDao. Attention, il faut quand même sauvegarder (si c est le comportement désiré) les paramètres du constructeur dans les attributs, pour pouvoir les utiliser dans la classe. Comme le nom de l injection l indique, Google Guice va faire l injection au niveau du constructeur et pas sur les attributs : public class PizzaManagerImpl implements PizzaManager { private final PizzaDao pizzadao; @Inject public PizzaManagerImpl(PizzaDao pizzadao) { this.pizzadao = pizzadao; @Override public List<Pizza> menu() { return pizzadao.getall(); //... [[information]] - Google Guice est compatible avec les annotations com.google.inject et javax.inject alors que Dagger n est compatible qu avec ces dernières. Dans le doute, utilisez javax.inject, puisque cette solution a le mérite d être la plus standard et la plus utilisée. - Contrairement à Google Guice, lorsqu une annotation est placée au niveau du constructeur plutôt que sur un attribut avec Dagger, les dépendances sont résolues au moment de la compilation et non plus à l exécution. Les performances à l usage seront bien meilleures. Une solution équivalente à l exemple précédent pour l injection sur les attributs, PizzaManagerImpl peut placer l annotation d injection @Inject sur l attribut pour se le faire injecter automatiquement (sans passer par le constructeur). Pour rappel, même si cette solution semble plus 11

4 Utiliser l injection de dépendances simple, élégante et facile, il ne faut pas oublier qu aucune optimisation ne sera faite ni pour Dagger ni pour Google Guice avec cette solution. Elle est donc plus coûteuse. public class PizzaManagerImpl implements PizzaManager { @Inject PizzaDao pizzadao; //... 4.2 Satisfaire les dépendances Pour satisfaire les dépendances d une application, des modules sont développés afin de renseigner les classes d implémentation. Les bibliothèques peuvent alors utiliser ce module pour invoquer le constructeur voulu, pour le constructeur ou l attribut annoté par une annotation d injection. La seule réelle différence dans l usage des bibliothèques Google Guice et Dagger réside dans cette satisfaction des dépendances. Alors que Google Guice utilisera une classe abstraite de sa bibliothèque, Dagger modernisera le concept en n utilisant que des annotations. Puisque les solutions sont vraiment différentes, les deux bibliothèques seront expliquées dans cette section. 4.2.1 Google Guice Une nouvelle classe doit étendre la classe AbstractModule de la bibliothèque Google Guice. Cette dernière classe nous oblige à mettre en œuvre la méthode configure(). C est dans cette méthode qu une interface ou une classe abstraite renseigne sa classe concrète pour pouvoir être instanciée. Par exemple, la classe abstraite AbstractPayment est, par définition, non instanciable. C est pourquoi, le module spécifie une de ses classes filles comme classe d implémentation pour que l application puisse connaître la classe qu il devra instancier à l exécution. Ce procédé fonctionne de la même manière avec les interfaces. Ceci se fait très simplement avec des méthodes de la classe AbstractModule : bind(...) pour renseigner l interface ou la classe abstraite, suivi de to(...) pour renseigner la classe concrète. public class PizzaManagerModuleGuice extends AbstractModule { @Override protected void configure() { bind(abstractpayment.class).to(creditcardpayment.class); Pour utiliser ce module, dans la classe principale de l application avec la méthode main, il faut créer l injecteur avec la méthode statique Guice.createInjector(...) pour pouvoir injecter toutes les instances nécessaires dans une classe et l utiliser. Dans cet exemple, Guice fournit une instance de la classe AppPizzaManagerGuice pour pouvoir injecter les classes voulues. public class AppPizzaManagerGuice implements Runnable { @Inject PizzaManager pizzamanager; @Inject PaymentManager paymentmanager; 12

4.2 Satisfaire les dépendances public static void main(string[] args) { final Injector injector = Guice.createInjector(new PizzaManagerModuleGuice()); final AppPizzaManagerGuice app = injector.getinstance(apppizzamanagerguice.class); app.run(); @Override public void run() { System.out.println(paymentManager.payWithPayPal(pizzaManager.menu())); 4.2.2 Dagger Un module avec Dagger ne doit étendre aucune classe parent mais doit faire bon usage des annotations. La classe du module va être annotée par @Module et renseigner en paramètre la classe principale de l application. Puis, toutes les dépendances seront résolues par des méthodes. Leurs types de retour correspondent à l interface ou la classe abstraite, la valeur retournée dans le corps de la méthode à la classe d implémentation pour le type de retour et la méthode est annotée par l annotation @Provides. @Module( injects = AppPizzaManagerDagger.class ) public class PizzaManagerModuleDagger { @Provides AbstractPayment providecreditcard() { return new CreditCardPayment(); Pour utiliser ce module, il faut obtenir le graphe des injections avec l invocation de la méthode statique ObjectGraph.create(...) qui accepte un ou plusieurs modules. final ObjectGraph objectgraph = ObjectGraph.create(new PizzaModule()); Afin d utiliser ce graphe, il faut amorcer les injections. Habituellement, cela requiert l injection de la classe principale. Pour l exemple, la classe AppPizzaManagerDagger est utilisée pour démarrer l injection de dépendances. Nous demandons au graphe de fournir une instance injectée de la classe : public class AppPizzaManagerDagger implements Runnable { @Inject PizzaManager pizzamanager; @Inject PaymentManager paymentmanager; public static void main(string[] args) { final ObjectGraph objectgraph = ObjectGraph.create(new PizzaManagerModuleDagger()); final AppPizzaManagerDagger app = objectgraph.get(apppizzamanagerdagger.class); app.run(); 13

4 Utiliser l injection de dépendances @Override public void run() { System.out.println(paymentManager.payWithPayPal(pizzaManager.menu())); 14

5 Des fonctionnalités supplémentaires Jusqu à maintenant, il a été présenté la fonctionnalité principale de l injection de dépendances mais il en existe bien d autres, dont certaines qui peuvent varier d une bibliothèque à l autre ou d un langage à l autre. L idée n est pas de toutes les énumérer ni de toutes les expliquer. Par contre, il en existe quelques unes qui sont assez génériques et présentes dans la plupart des solutions. 5.1 Nommer les injections Imaginez deux attributs du même type (par exemple, d une classe abstraite) mais pour lesquels vous désirez des implémentations différentes. La fonctionnalité de base consiste à lier une classe abstraite (ou une interface) à une classe concrète (ou d implémentation). À partir de ce constat, il ne serait pas possible de fournir deux implémentations différentes. Si vous tentez de faire : bind(abstractpayment.class).to(paypalpayment.class); bind(abstractpayment.class).to(creditcardpayment.class); La bibliothèque prendra en compte le dernier «binding». Toutes les injections pour AbstractPayment seront instanciées par CreditCardPayment. Pour pallier ce problème, une des solutions les plus communes est la possibilité de nommer ses injections grâce à l annotation @Named("your-name"). Le mode de payement par PayPal pourrait être nommé par paypal et la carte de crédit par creditcard. Ainsi, PaymentManager peut disposer de deux attributs typés de AbstractPayment avec deux implémentations différentes. public class PaymentManagerImpl implements PaymentManager { @Inject @Named("paypal") AbstractPayment paypal; @Inject @Named("creditcard") AbstractPayment creditcard; //... Petite modification aussi du côté du module puisqu il faut spécifier à Google Guice que toutes les injections du type AbstractPayment et annotées par le nom «paypal» ou «creditcard» doivent posséder l implémentation PayPalPayment ou CreditCardPayment (réciproquement). public class PizzaManagerModuleGuice extends AbstractModule { @Override protected void configure() { bind(abstractpayment.class).annotatedwith(names.named("paypal")).to(paypalpayment.c bind(abstractpayment.class).annotatedwith(names.named("creditcard")).to(creditcardp //... 15

5 Des fonctionnalités supplémentaires Pour Dagger, il suffit de créer deux nouvelles méthodes annotées par @Provides et @Named("name") en renvoyant la classe d implémentation voulue. public class PizzaManagerModuleDagger { @Provides @Named("paypal") AbstractPayment providepaypal() { return new PayPalPayment(); @Provides @Named("creditcard") AbstractPayment providecreditcard() { return new CreditCardPayment(); //... 5.2 Ses propres annotations pour nommer une injection Une solution qui peut paraitre un poil plus élégante, en fonction du développeur, est la possibilité de créer ses propres annotations pour nommer une injection. Ainsi, cela consiste à créer une nouvelle annotation en spécifiant que c est une annotation de type «binding», qu elle peut s appliquer sur des attributs, des paramètres et des méthodes et qu elle est interprétée à l exécution du programme. @BindingAnnotation @Target({FIELD, PARAMETER, METHOD) @Retention(RUNTIME) public @interface PayPal { L exemple du mode de payement va légèrement évoluer pour PayPal puisqu il faut dorénavant annoter le champ par @PayPal plutôt que par @Named( paypal ). java hl_lines="2" public class PaymentManagerImpl implements PaymentManager { @Inject @PayPal AbstractPayment paypal; @Inject @Named("creditcard") AbstractPayment creditcard; //... Par contre, réelle nouveauté dans le module de Google Guice puisqu il n est plus possible de renseigner l injection dans la méthode configure(). La mise en œuvre se rapproche fortement de Dagger puisqu il est nécessaire de créer une nouvelle méthode annotée par @Provides et @PayPal en renvoyant une instance de la classe d implémentation voulue pour le type retourné de la méthode. public class PizzaManagerModuleGuice extends AbstractModule { @Override protected void configure() { bind(abstractpayment.class).annotatedwith(names.named("creditcard")).to(creditcardp @Provides @PayPal public AbstractPayment providepaypal() { return new PayPalPayment(); //... 16

5.3 Fournir des instances différentes 5.3 Fournir des instances différentes Parfois, il est nécessaire de vouloir retourner plusieurs instances différentes pour une injection. Il existe aussi plusieurs possibilités, une de ces options étant l injection d un attribut grâce à Provider<T>. Provider<T> crée une nouvelle instance de T à chaque invocation de la méthode get() sur l attribut. java hl_lines= 4 15 public class PaymentManagerImpl implements PaymentManager { @Inject @PayPal AbstractPayment paypal ; @Inject @Named( creditcard ) AbstractPayment credit- Card ; @Inject Provider paymentprovider ; @Override public Payment paywithpaypal(list pizzas) { return getpayment(paypal.pay(pizzas)) ; @Override public Payment paywithcreditcard(list pizzas) { return getpayment(creditcard.pay(pizzas)) ; private Payment getpayment(double total) { final Payment payment = paymentprovider.get() ; payment.settotal(total) ; return payment ; 17

6 Conclusion Ce tutoriel aborde les bases de l injection de dépendances, mais des bases tout de même assez solides pour une utilisation courante dans des petits et moyens projets. Pour en savoir plus sur l injection de dépendances, de nombreuses ressources existent sur internet, notamment la documentation officielle des bibliothèques mises en œuvre dans ce tutoriel. Sachez aussi que toutes les sources des exemples de ce tutoriel sont disponibles sur ce dépôt git. Libre à vous de vous inspirer de ce projet pour vos projets personnels, ou d y apporter des améliorations si vous tombez sur des petites trouvailles à force de pratique ; toute contribution directe de votre part est la bienvenue. Dans des tutoriels ouverts comme celui-ci, chacun peut apporter sa pierre à l édifice. 6.1 Remerciements L icône de ce tutoriel est distribuée sous licence CC BY-SA 2.0, par tamasrepus et distribuée via la plateforme Flickr. Merci à la communauté de Zeste de Savoir pour leurs retours sur ce tutoriel lorsqu il était en bêta. Merci à Coyote pour la relecture de ce tutoriel. 19