Programmation Java/Débogage

Outils de débogage

modifier

Pour déboguer sans TDD (test driven development), on peut ajouter le code suivant où l'on veut :

System.out.println("Valeur à afficher");System.exit(0);

Cependant ce n'est pas la meilleure façon de faire. Il est préférable d'utiliser un IDE supportant le débogage :

  • Poser des points d'arrêt pour suspendre l'exécution et inspecter les valeurs des attributs, des arguments, des variables locales.
  • Reprendre ensuite en mode pas à pas pour voir l'évolution des valeurs et voir quelle partie du code s'exécute.

Par exemple, Eclipse possède deux manières de lancer une application :

  • Le bouton de lancement normal ignore tous les points d'arrêts.
  • Le bouton de débogage représenté par une icône d'insecte pour le mot anglais "bug" (cafard), permet de prendre en compte les points d'arrêt pour suspendre l'exécution de l'application. Une perspective spécifique au débogage est alors affichée pour lister les expressions à pister et voir le code source.

VisualVM

modifier

VisualVM est un outil permettant d'explorer les informations sur les machines virtuelles Java qui tournent sur la machine (une JVM par application).

Cet outil était disponible avec la plateforme Java, édition standard (Java SE) dans les versions du JDK de la version 6 update 7 jusuq'à la version 8 incluse (voir https://docs.oracle.com/javase/8/docs/technotes/guides/visualvm/). Il est désormais disponible séparément sur le site https://visualvm.github.io/ .

jdb est un débogueur en ligne de commande fourni avec le JDK.

Pour plus de détails voir : Programmation Java/JDK#jdb.

Erreurs à la compilation

modifier

Fenêtre de warning ...java uses unchecked or unsafe operations

modifier

Une déclaration ne tient pas compte des surcharges possibles, une conversion de variable pose problème. Il faut donc en prévoir les exceptions. Par exemple pour manipuler des entiers :

try {
  i = i/2;
} catch (NumberFormatException e) {
  return 0;
}

<identifier> expected

modifier

L'ordre des déclarations n'est pas chronologique.

annotation type not applicable to this kind of declaration

modifier

Voir @Override.

cannot be dereferenced

modifier
  • Il suffit de retirer la conversion, le .toString() après la variable de type int sélectionnée.
  • Ou en ajouter une comme : (float), Float.valueOf(), ou .floatValue().

cannot find symbol - variable

modifier

Soit :

  • La variable mentionnée n'a pas été déclarée (ou elle a été mal écrite) ;
  • Sa déclaration trouve dans une condition (if ou try) fermée avant la ligne en erreur ;
  • Elle se trouve dans un package non importé ;
  • Si elle se trouve dans un autre fichier, les compiler depuis leur répertoire parent commun (les IDE le font automatiquement) :
javac Projet1/Classe1.java
javac Projet1/Classe2.java

cannot instantiate from arguments because actual and formal argument lists differ in length

modifier

Cela peut survenir suite à un Collections.sort(liste), alors même que le Collections.shuffle(liste) fonctionne. Il faut donc utiliser import java.util.Comparator; dans le trie :

import java.util.Collections;
import java.util.Comparator;
...
private List<T> list;
private Comparator<T> comparator = null;
if(this.comparator!= null) {
       Collections.sort(list, comparator);
    }

constructor variable in class classe cannot be applied to given types

modifier

exception IOException is never thrown in body of corresponding try statement

modifier

Un try inutile a été détecté. Cela peut être causé par la mise en commentaires de lignes devenues inutiles (instructions de débogage par exemple).

impossible de trouver la classe principale

modifier

Lancer la commande "java" vers un .class (et non un .java).

incompatible types: possible lossy conversion from float to int ou double to float

modifier

Il faut passer une conversion de format :

java.text.DecimalFormat df = new java.text.DecimalFormat("#");
monFloat = Float.valueOf(df.format(monDouble));

java.lang.NumberFormatException

modifier

Le contenu d'une chaîne de caractère ne peut être interprétée comme un nombre :

  • Soit à cause d'un symbole qui n'est pas un chiffre (ex : €)
  • ou bien une virgule à la place d'un point pour un nombre réel.

local variable referenced from a lambda expression must be final or effectively final

modifier

local variable ma_variable is accessed from within inner class; needs to be declared final

modifier

no suitable method found for - méthode

modifier

La méthode mentionnée n'a pas été déclarée, ou du moins pas avec ce type de paramètre (ex : remplacer .addAll() par .add()).

no suitable constructor found for Integer(double)

modifier

Convertir plus explicitement :

MonEntier = ((Number)MonDouble).intValue();

no suitable constructor for classe is abstract cannot be instanciated

modifier

S'il est possible de retirer le mot abstract de la déclaration de la classe, le faire.

non-static method/variable ... cannot be referenced from a static context

modifier

Cela peut se produire en invoquant une méthode ou une variable dans sa classe, ou une autre classe.

  • Si elle est bien statique :
    • Au sein de la même classe, retirer le this. en préfixe.
    • Depuis une autre classe, ajouter le nom de la classe en préfixe.
    • Sinon, il faut rendre la variable appelée dans la méthode statique, statique aussi.
  • Si elle n'est pas statique :
    • On peut rendre la méthode statique non statique.
    • Sinon, ajouter ou remplacer le "this." par le nom de la variable désignant la classe (ex : "c.") :
MaClasse c = new MaClasse();
c.MaMethode();

not a statement

modifier

Il manque sûrement les parenthèses après le nom d'une méthode.

reference to assertequals is ambiguous

modifier

Il convient d'assurer un typage plus fort des paramètres de assertequals(), par exemple avec .intValue().

unclosed character literal

modifier

Il manque le délimiteur apostrophe final, ou bien l'apostrophe comme caractère doit être préfixée d'un anti-slash.

Ou bien utiliser une chaîne de caractères pour zéro ou plus d'un caractère.

unclosed string literal

modifier

Il manque le délimiteur guillemet final, ou bien le guillemet comme caractère dans la chaîne doit être préfixé d'un anti-slash.

unreachable statement

modifier

Une partie de code ne sera jamais exécutée, parce qu'elle est juste après un break ou un return.

unreported exception ...; must be caught or declared to be thrown

modifier

Une exception est lancée explicitement (throw new NomDeTException("Message");) ou par une méthode déclarant pouvoir lancer cette exception sans que la méthode appelante ait déclarée avec throws NomDeLException.

  • Soit il faut ajouter une capture de l'exception dans la méthode si elle peut la traiter complétement à son niveau (journalisation, message à l'utilisateur, arrêter une procédure en cours...),
  • Soit plus probablement il faut que la méthode déclare lancer ce type d'exception avec le mot-clé throws à la fin de la déclaration de la méthode.

Il se peut que la méthode puisse traiter partiellement l'exception (ex : journaliser l'erreur) et qu'elle doive la relancer pour qu'elle soit traitée complétement ailleurs (remonter jusqu'au code de l'interface graphique pour l'affichage d'un message d'erreur). Dans ce cas, un mélange des deux solutions est à faire :

  • Capturer l'exception pour journaliser l'erreur et la relancer avec throw
  • Déclarer ce type d'exception avec le mot-clé throws à la fin de la déclaration de la méthode.

variable is already defined

modifier

Si la variable est redéfinie dans une condition différente, il faut les séparer avec des { }, ex :

switch (args[0].toString()) {
  case "1": {
    a = 1;
    break;
  }
  case "2": {
    a = 2;
    break;
  }
}

variable might not have been initialized

modifier

Il faut assigner une valeur par défaut à l'objet. Ex :

String[] a = {"", ""};
BufferedWriter bw = null;

Erreurs à l'exécution

modifier

Si deux variables de type String identiques ne peuvent pas être comparées

modifier

L'opérateur == vérifie les références au lieu du contenu.

Il faut donc utiliser la méthode .equals().

String s = "test";
StringBuffer sb = new StringBuffer("test");
System.out.println(s == new String(sb));         // false !
System.out.println(s.equals(new String(sb)));    // true :)

Si une division fait toujours 0.0

modifier

Ajouter (float) avant.

NullPointerException

modifier

Soulevée si l'on applique une méthode sur un objet null. On peut donc changer l'appel ou lever l'exception :

    try {
        ...
    } catch (NullPointerException npe) {
        npe.printStackTrace(); 
    }

aucun attribut manifest principal dans ...jar

modifier

Il n'y a pas de méthode main() dans le .jar, ce fichier ne peut donc pas être exécuté directement, mais peut être utilisé par une application ayant une méthode main().

Class names, classe, are only accepted if annotation processing is explicitly requested

modifier

Les classes définies à la compilation (par exemple avec javac -classpath) sont introuvables.

Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 1

modifier

Un des tableau de la ligne indiquée est appelé avec un indice négatif ou supérieur à sa taille.

java.lang.ArithmeticException: / by zero

modifier

Une division par zéro s'évite généralement par un if sur le dénominateur, ou un throw d'une exception.

Stream closed

modifier

Retirer le .close() de la boucle du .read() ou du .write(), même s'il est situé après. Si cela ne suffit pas, le retirer de la méthode.

Unable to load native library: Can't load IA 32-bit .dll on a AMD 64-bit platform

modifier

Une application utilisant du code natif a tenté de chargé une bibliothèque de code dynamique (.dll sous windows, .so sous Linux/Mac) 32 bit sur une machine virtuelle Java 64 bits. L'inverse provoque également une variante de ce message d'erreur.

Solutions possibles :

  • Installer la version 32 ou 64 bits du JRE requise.
  • Ou chercher une version de l'application ayant une bibliothèque de code dynamique adaptée à la JVM 32 ou 64 bits.

Une application Java utilisant du code natif bien conçue ne devrait jamais rencontrer ce problème : il est possible de détecter la version de Java (32 ou 64 bits) avant de charger la bibliothèque dynamique appropriée, ce qui signifie avoir une version 32 bits et une version 64 bits de cette bibliothèque dynamique (ex : mylib-32.dll et mylib-64.dll).