Développons en Java v 2.40 Copyright (C) 1999-2023 Jean-Michel DOUDOUX. |
|||||||
Niveau : | Confirmé |
Depuis la version 1.1 de Java, il est possible de créer et de gérer dynamiquement des objets.
L'introspection est un mécanisme qui permet de connaître le contenu d'une classe dynamiquement. Il permet notamment de savoir ce que contient une classe sans en avoir les sources. Ces mécanismes sont largement utilisés dans des outils de type IDE (Integrated Development Environnement : environnement de développement intégré).
Pour illustrer ces différents mécanismes, ce chapitre va construire une classe utilitaire qui proposera un ensemble de méthodes fournissant des informations sur une classe donnée.
Les différentes classes utiles pour l'introspection sont rassemblées dans le package java.lang.reflect.
Voici le début de cette classe qui attend dans son constructeur une chaîne de caractères précisant la classe sur laquelle elle va travailler.
Exemple ( code Java 1.1 ) : |
Ce chapitre contient plusieurs sections :
Les instances de la classe Class sont des objets représentant les classes du langage. Il y aura une instance représentant chaque classe utilisée : par exemple la classe String, la classe Frame, la classe Class, etc ... Ces instances sont crées automatiquement par la machine virtuelle lors du chargement de la classe. Il est ainsi possible de connaître les caractéristiques d'une classe de façon dynamique en utilisant les méthodes de la classe Class. Les applications telles que les débogueurs, les inspecteurs d'objets et les environnements de développement doivent faire une analyse des objets qu'ils manipulent en utilisant ces mécanismes.
La classe Class est définie dans le package java.lang.
La classe Class permet :
La classe Class ne possède pas de constructeur public mais il existe plusieurs façons d'obtenir un objet de la classe Class.
La méthode getClass() définit dans la classe Object renvoie une instance de la classe Class. Par héritage, tout objet Java dispose de cette méthode.
Exemple ( code Java 1.1 ) : |
Résultat : |
La classe Class possède une méthode statique forName() qui permet à partir d'une chaîne de caractères désignant une classe d'instancier un objet de cette classe et de renvoyer un objet de la classe Class pour cette classe.
Cette méthode peut lever l'exception ClassNotFoundException.
Exemple ( code Java 1.1 ) : |
Résultat : |
Il est possible d'avoir un objet de la classe Class en écrivant type.class où type est le nom d'une classe.
Exemple ( code Java 1.1 ) : |
Résultat : |
La classe Class fournit de nombreuses méthodes pour obtenir des informations sur la classe qu'elle représente. Voici les principales méthodes :
Méthodes | Rôle |
static Class forName(String) | Instancier un objet de la classe dont le nom est fourni en paramètre et renvoie un objet Class la représentant |
Class[] getClasses() | Renvoyer les classes et interfaces publiques qui sont membres de la classe |
Constructor[] getConstructors() | Renvoyer les constructeurs publics de la classe |
Class[] getDeclaredClasses() | Renvoyer un tableau des classes définies comme membres dans la classe |
Constructor[] getDeclaredConstructors() | Renvoyer tous les constructeurs de la classe |
Field[] getDeclaredFields() | Renvoyer un tableau de tous les attributs définis dans la classe |
Method[] getDeclaredMethods() | Renvoyer un tableau de toutes les méthodes |
Field[] getFields() | Renvoyer un tableau des attributs publics |
Class[] getInterfaces() | Renvoyer un tableau des interfaces implémentées par la classe |
Method[] getMethod() | Renvoyer un tableau des méthodes publiques de la classe incluant celles héritées |
int getModifiers() | Renvoyer un entier qu'il faut décoder pour connaître les modificateurs de la classe |
Package getPackage() | Renvoyer le package de la classe |
Class<?>[] getPermittedSubclasses() | Renvoyer un tableau des type autorisés à hériter du type scellé (Java 17) |
Classe getSuperClass() | Renvoyer la classe mère de la classe |
boolean isArray() | Indiquer si la classe est un tableau |
boolean IsInterface() | Indiquer si la classe est une interface |
boolean isSealed() | Indiquer si le type est scellé (Java 17) |
Object newInstance() | Créer une nouvelle instance de la classe |
En utilisant les méthodes de la classe Class, il est possible d'obtenir quasiment toutes les informations sur une classe.
La classe Class possède une méthode getSuperClass() qui retourne un objet de la classe Class représentant la classe mère si elle existe sinon elle retourne null.
Pour obtenir toute la hiérarchie d'une classe il suffit d'appeler successivement cette méthode sur l'objet qu'elle a retourné.
Exemple ( code Java 1.1 ) : méthode qui retourne un vecteur contenant les classes mères |
La classe Class possède une méthode getModifiers() qui retourne un entier représentant les modificateurs de la classe. Pour décoder cette valeur, la classe Modifier possède plusieurs méthodes qui attendent cet entier en paramètre et qui retournent un booléen selon leur fonction : isPublic(), isAbstract(), isFinal(), ...
La classe Modifier ne contient que des constantes et des méthodes statiques qui permettent de déterminer les modificateurs d'accès :
Méthode | Rôle |
boolean isAbstract(int) | Renvoyer true si le paramètre contient le modificateur abstract |
boolean isFinal(int) | Renvoyer true si le paramètre contient le modificateur final |
boolean isInterface(int) | Renvoyer true si le paramètre contient le modificateur interface |
boolean isNative(int) | Renvoyer true si le paramètre contient le modificateur native |
boolean isPrivate(int) | Renvoyer true si le paramètre contient le modificateur private |
boolean isProtected(int) | Renvoyer true si le paramètre contient le modificateur protected |
boolean isPublic(int) | Renvoyer true si le paramètre contient le modificateur public |
boolean isStatic(int) | Renvoyer true si le paramètre contient le modificateur static |
boolean isSynchronized(int) | Renvoyer true si le paramètre contient le modificateur synchronized |
boolean isTransient(int) | Renvoyer true si le paramètre contient le modificateur transient |
boolean isVolatile(int) | Renvoyer true si le paramètre contient le modificateur volatile |
Ces méthodes étant static il est inutile d'instancier un objet de type Modifier pour les utiliser.
Exemple ( code Java 1.1 ) : |
La classe Class possède une méthode getInterfaces() qui retourne un tableau d'objets de type Class contenant les interfaces implémentées par la classe.
Exemple ( code Java 1.1 ) : |
La classe Class possède une méthode getFields() qui retourne les attributs public de la classe. Cette méthode retourne un tableau d'objets de type Field.
La classe Class possède aussi une méthode getField() qui attend en paramètre un nom d'attribut et retourne un objet de type Field si celui-ci est défini dans la classe ou dans une de ses classes mères. Si la classe ne contient pas d'attribut dont le nom correspond au paramètre fourni, la méthode getField() lève une exception de la classe NoSuchFieldException.
La classe Field représente un attribut d'une classe ou d'une interface et permet d'obtenir des informations sur cet attribut. Elle possède plusieurs méthodes :
Méthode | Rôle |
String getName() | Retourner le nom de l'attribut |
Class getType() | Retourner un objet de type Class qui représente le type de l'attribut |
Class getDeclaringClass() | Retourner un objet de type Class qui représente la classe qui définit l'attribut |
int getModifiers() | Retourner un entier qui décrit les modificateurs d'accès. Pour les connaître précisément il faut utiliser les méthodes static de la classe Modifier. |
Object get(Object) | Retourner la valeur de l'attribut pour l'instance de l'objet fournie en paramètre. Il existe aussi plusieurs méthodes getXXX() où XXX représente un type primitif et qui renvoient la valeur dans ce type. |
Exemple ( code Java 1.1 ) : |
L'exemple ci-dessous présente une méthode qui permet de formater sous forme de chaîne de caractères les paramètres d'une méthode fournis sous la forme d'un tableau d'objets de type Class.
Exemple ( code Java 1.1 ) : |
La méthode getName() de la classe Class renvoie une chaîne de caractères formatée qui précise le type de la classe. Ce type est représenté par une chaîne de caractères qu'il faut décoder pour l'extraire.
Si le type de la classe est un tableau alors la chaîne commence par un nombre de caractères '[' correspondant à la dimension du tableau.
Ensuite la chaîne contient un caractère qui précise un type primitif ou un objet. Dans le cas d'un objet, le nom de la classe de l'objet avec son package complet est contenu dans la chaîne suivie d'un caractère ';'.
Caractère |
Type |
B | byte |
C | char |
D | double |
F | float |
I | int |
J | long |
Lclassname; | classe ou interface |
S | short |
Z | boolean |
Exemple :
La méthode getName() de la classe Class représentant un objet de type float[10][5] renvoie « [[F »
Pour simplifier les traitements, la méthode formatParametre() ci-dessous retourne une chaîne de caractères qui décode le contenu de la chaîne retournée par la méthode getName() de la classe Class.
Exemple : |
La classe Class possède une méthode getConstructors() qui retourne un tableau d'objets de type Constructor contenant les constructeurs de la classe.
La classe Constructor représente un constructeur d'une classe et possède plusieurs méthodes :
Méthode | Rôle |
String getName() | Retourner le nom du constructeur |
Class[] getExceptionTypes() | Retourner un tableau de type Class qui représente les exceptions qui peuvent être propagées par le constructeur |
Class[] getParametersType() | Retourner un tableau de type Class qui représente les paramètres du constructeur |
int getModifiers() | Retourner un entier qui décrit les modificateurs d'accès. Pour les connaître précisément il faut utiliser les méthodes static de la classe Modifier. |
Object newInstance(Object[]) | Instancier un objet en utilisant le constructeur avec les paramètres fournis à la méthode |
Exemple ( code Java 1.1 ) : |
L'exemple ci-dessus utilise la méthode rechercherParamètres() définie précédemment pour simplifier les traitements.
Pour consulter les méthodes d'un objet, il faut obtenir sa classe et lui envoyer le message getMethod(), qui renvoie les méthodes publiques qui sont déclarées dans la classe ou qui sont héritées des classes mères.
Elle renvoie un tableau d'instances de la classe Method du package java.lang.reflect.
Une méthode est caractérisée par un nom, une valeur de retour, une liste de paramètres, une liste d'exceptions et une classe d'appartenance.
La classe Method contient plusieurs méthodes :
Méthode | Rôle |
Class[] getParameterTypes | Renvoyer un tableau de classes représentant les paramètres. |
Class getReturnType | Renvoyer le type de la valeur de retour de la méthode. |
String getName() | Renvoyer le nom de la méthode |
int getModifiers() | Renvoyer un entier qui représente les modificateurs d'accès |
Class[] getExceptionTypes | Renvoyer un tableau de classes contenant les exceptions propagées par la méthode |
Class getDeclaringClass[] | Renvoyer la classe qui définit la méthode |
Exemple ( code Java 1.1 ) : |
L'exemple ci-dessus utilise les méthodes formatParametre() et rechercherParametres() définies précédemment pour simplifier les traitements.
Pour consulter toutes les méthodes d'un objet, il faut obtenir sa classe et lui envoyer le message getDeclaredMethods(), qui renvoie toutes les méthodes qui sont déclarées dans la classe ou qui sont héritées des classes mères quelque soit leur accessibilité.
Elle renvoie un tableau d'instances de la classe Method du package java.lang.reflect.
Exemple : |
L'exemple ci-dessus utilise les méthodes formatParametre() et rechercherParametres() définies précédemment pour simplifier les traitements.
Par convention, la valeur d'une propriété est gérée grâce à deux méthodes public :
Même si elle ne le propose pas directement, il est possible d'utiliser l'API Reflection pour obtenir les getters et les setters d'une classe.
Exemple : |
En Java 17, la classe Class est enrichie de deux nouvelles méthodes pour obtenir des informations relatives aux classes scellées :
La méthode getPermittedSubclasses() renvoie un tableau de type java.lang.Class qui contient toutes les sous-classes directes autorisées d'une classe scellée. L'ordre des classes n'est pas déterminé. Elle renvoie un tableau vide si la classe n'est pas scellée. Elle renvoie null si la classe du type est un tableau ou un type primitif.
La méthode isSealed() renvoie un booléen qui indique si le type (la classe ou l'interface) est scellé.
Exemple ( code Java 17 ) : |
Résultat : |
L'API Reflection permet de créer dynamiquement des instances d'un type.
La méthode statique forName() de la classe Class permet de charger dynamiquement une classe dont le nom pleinement qualifié est fourni en paramètre. Elle renvoie une instance de la classe Class qui encapsule la classe chargée.
La méthode newInstance() de la classe Class permet de créer une instance de la classe et d'invoquer son constructeur par défaut.
Exemple ( code Java 1.4 ) : |
A partir de Java 5, la classe Class est générique.
Exemple ( code Java 5.0 ) : |
La méthode newInstance() de la classe Class présente plusieurs contraintes :
A partir de la version 1.1, le package java.lang.reflect propose la classe Constructor pour créer des instances en invoquant un constructeur quelconque d'une classe.
La méthode getDeclaredConstructor() de la classe Class permet d'obtenir une instance de la classe Constructor qui encapsule le constructeur dont les types des paramètres ont été fournis à la méthode getDeclaredConstructor().
La méthode getDeclaredMethod() attend en paramètre un tableau d'objets de type Class qui doit contenir les types de chaque paramètre dans l'ordre de leur définition dans la signature du constructeur souhaité.
La classe Constructor propose la méthode newInstance() qui attend en paramètre un tableau de type Object devant contenir les valeurs qui seront fournies lors de l'invocation du constructeur.
Exemple : |
A partir de Java 5, les classes Class et Constructor sont génériques.
Exemple ( code Java 5.0 ) : |
Si une exception est levée lors de l'invocation du constructeur, celle-ci est chaînée dans une exception checked de type TargetInvocationException.
L'API Reflection permet d'invoquer dynamiquement une méthode d'un objet.
Pour invoquer dynamiquement une méthode d'une instance, il faut utiliser la méthode invoke(Object obj, Object[] args) de la classe java.lang.Method qui possède plusieurs paramètres :
Exemple : |
Exemple : |
Résultat : |
Une exception de type IllegalArgumentException est levée si aucune méthode dont la signature correspond aux types passés en paramètre n'est trouvée.
Exemple : |
Résultat : |
Comme le second paramètre de la méthode invoke() est un varargs, il est possible de passer un tableau de type Object de taille 0 pour indiquer qu'il n'y a pas de paramètre à la méthode.
Exemple : |
Résultat : |
Si la valeur null est passée comme paramètre de la méthode invoke() pour invoquer une méthode sans paramètre alors le compilateur émet un warning.
Exemple : |
Résultat : |
Le compilateur signale par son warning qu'il n'est pas en mesure de déterminer si la valeur null concerne la valeur du premier élément du varargs ou un tableau d'objets null.
Le résultat à l'exécution est tout de même celui attendu.
Résultat : |
Lors de l'invocation dynamique d'une méthode en utilisant la méthode invoke(), si une exception est levée par la méthode invoquée alors celle-ci est chainée dans une exception de type java.lang.reflect.InvocationTargetException.
Exemple : |
Exemple ( code Java 5.0 ) : |
Résultat : |
Pour obtenir l'exception levée par la méthode exécutée, il faut utiliser la méthode getCause() de l'exception InvocationTargetException.
Si la méthode à invoquer est static alors il faut passer null comme valeur du premier paramètre qui correspond à l'instance à invoquer.
Exemple : |
Résultat : |
Les méthodes getMethod(String name, Class[] parameterTypes) et getMethods() ne permettent de renvoyer que des méthodes publiques. Pour obtenir des méthodes privées, il faut utiliser les méthodes getDeclaredMethod() et getDeclaredMethods().
Par défaut, l'invocation dynamique d'une méthode inaccessible, par exemple déclarée private, lève une exception de type IllegalAccessException.
Exemple : |
Résultat : |
La méthode getDeclaredMethod() ne peut qu'accéder aux méthodes qui sont déclarées dans la classe elle-même : elle ne permet pas d'accéder aux méthodes des super-classes.
Par défaut, les restrictions d'accès à une méthode s'appliquent aussi lors de l'utilisation de l'API Reflection.
La classe Method hérite de la classe AccessibleObject qui possèdent la méthode setAccessible(). Elle attend en paramètre un booléen : elle permet avec la valeur true de retirer les vérifications d'accessibilité qui seront faites pour permettre un accès par introspection à la méthode encapsulée. Cela permet de contourner les vérifications d'accès et ainsi d'accéder à une méthode déclarée privée, protected ou package-private uniquement en utilisant l'API Reflection.
Exemple : |
Il est nécessaire de tenir compte de plusieurs points lors de l'utilisation de l'introspection pour invoquer une méthode dont le type d'un paramètre est un type générique.
Exemple ( code Java 5.0 ) : |
Exemple ( code Java 5.0 ) : |
Résultat : |
Bien que le type générique de la classe soit Integer, la méthode n'est pas trouvée par introspection en précisant le type Integer comme paramètre.
A cause de l'implémentation des generics qui utilise le type erasure, le type generic original est perdu à la compilation pour laisser le type Object. Lorsque le type de la méthode est un type généric, il faut le remplacer par le type Object lorsque l'API Reflection est utilisée pour invoquer la méthode.
Exemple ( code Java 5.0 ) : |
Résultat : |
L'API Reflection permet la mise en oeuvre de puissantes fonctionnalités : c'est une des raisons qui fait qu'elle est fréquemment utilisée par de nombreux frameworks parmi lesquels Spring ou Hibernate.
Cependant certaines fonctionnalités peuvent aussi être utilisées à des fins malveillantes qui peuvent nuire à la sécurité d'une application (invocation de méthodes, modifications de la valeur de champs, ... même si les modificateurs de ces membres ne permettent normalement pas leur accès, ...).
Les accès à un objet en utilisant l'API Reflection se font en utilisant une implémentation de l'interface AccessibleObject. Pour contourner les vérifications de l'accessibilité aux éléments d'un objet, il faut mettre à true la propriété access en utilisant la méthode setAccessible(). Par contre cela ne désactive pas les vérifications faites par le SecurityManager, s'il y en a un d'activé.
Par défaut, aucun SecurityManager n'est activé dans une JVM. Pour en activer un, il faut soit :
Lorsqu'un SecurityManager est activé sur une JVM, il est nécessaire d'autoriser la permission de type ReflectPermission dont le nom est "suppressAccessChecks" pour pouvoir utiliser des fonctionnalités de l'API Reflection. Si cette permission n'est pas donnée, alors une exception est levée par la méthode checkPermission() lors de l'utilisation de ces fonctionnalités.
Exemple : |
Résultat : |
Il est nécessaire de définir ou de modifier la politique de sécurité pour accorder la permission "suppressAccessChecks" à la classe java.lang.reflect.ReflectPermission.
Il est possible de définir son propre fichier qui contient la définition de la politique de sécurité à appliquer.
Le fichier TestExecuterMethode.policy |
Il faut préciser le fichier comme valeur de la propriété java.security.policy de la JVM.
Exemple : |
Il est important de modifier la valeur de la propriété avant l'activation du SecurityManager sinon il faut ajouter une permission autorisant la modification des propriétés système.
Attention : il faut autoriser la permission "suppressAccessChecks" avec précaution en limitant son effet uniquement sur les classes connues pour en avoir besoin. Typiquement, dans l'exemple ci-dessus, cette permission est donnée à toutes les classes dans la JVM ce qui peut être à l'origine de problèmes de sécurité.
Une exception de type SecurityException est levée si la méthode setAccessible() est invoquée sur une instance de type Constructor pour la classe Class.
Exemple : |
Résultat : |
Les annotations permettent d'ajouter des métadonnées dans le code source Java. Ces métadonnées peuvent être exploitées dans le code source, à la compilation ou à l'exécution en utilisant l'API Reflection.
Elle permet d'accéder aux annotations définies sur un type, une méthode, un champ ou un paramètre de manière dynamique à l'exécution.
Pour pouvoir utiliser l'API Reflection sur une annotation à l'exécution, il est nécessaire que la définition de l'annotation soit faite avec l'annotation @Retention à laquelle la valeur RetentionPolicy.RUNTIME est utilisée en paramètre.
Exemple ( code Java 5.0 ) : |
Telle que définie, l'annotation peut s'utiliser sur un type (une classe ou une interface).
Exemple ( code Java 5.0 ) : |
Il est possible d'utiliser l'API Reflection pour accéder dynamiquement aux annotations utilisées sur une classe.
La méthode getAnnotations() de la classe Class permet d'obtenir un tableau de type Annotation qui contient toutes les annotations définies sur la classe.
Exemple : |
Résultat : |
La méthode getAnnotation() de la classe Class permet d'obtenir une instance de type Annotation encapsulant l'annotation utilisée sur la classe dont le type correspond à celui passé en paramètre.
Exemple ( code Java 5.0 ) : |
Telle que définie, l'annotation peut s'utiliser sur une méthode.
Exemple ( code Java 5.0 ) : |
Il est possible d'utiliser l'API Reflection pour accéder dynamiquement aux annotations utilisées sur une méthode.
La méthode getDeclaredAnnotations() de la classe Method permet d'obtenir un tableau de type Annotation qui contient toutes les annotations définies sur la méthode.
Exemple ( code Java 5.0 ) : |
Résultat : |
La méthode getAnnotation() de la classe Method permet d'obtenir une instance de type Annotation encapsulant l'annotation utilisée sur la méthode dont le type est passé en paramètre.
Exemple ( code Java 5.0 ) : |
Résultat : |
Telle que définie, l'annotation peut s'utiliser sur un paramètre d'une méthode.
Exemple ( code Java 5.0 ) : |
Il est possible d'utiliser l'API Reflection pour accéder dynamiquement aux annotations utilisées sur les paramètres d'une méthode.
Exemple ( code Java 5.0 ) : |
Résultat : |
La méthode getParameterAnnotations() renvoie un tableau à deux dimensions de type Annotation qui contient pour chaque paramètre, les annotations qui lui sont associées.
Telle que définie, l'annotation peut s'utiliser sur un champ d'une classe.
Exemple ( code Java 5.0 ) : |
Il est possible d'utiliser l'API Reflection pour accéder dynamiquement aux annotations utilisées sur un champ.
La méthode getDeclaredAnnotations() de la classe Field permet d'obtenir un tableau de type Annotation qui contient toutes les annotations définies sur le champ.
Exemple ( code Java 5.0 ) : |
Résultat : |
La méthode getAnnotation() de la classe Field permet d'obtenir une instance de type Annotation encapsulant l'annotation utilisée sur le champ dont le type est passé en paramètre.
Exemple ( code Java 5.0 ) : |
Résultat : |
Développons en Java v 2.40 Copyright (C) 1999-2023 Jean-Michel DOUDOUX. |