IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Langage Java Discussion :

Comparable et généricité


Sujet :

Langage Java

  1. #1
    Membre régulier
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    124
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Octobre 2006
    Messages : 124
    Points : 87
    Points
    87
    Par défaut Comparable et généricité
    Bonjour à tous,

    J'ai implémenté une classe Tuple dont l'utilité est de pouvoir rassembler des éléments homogènes dans un conteneur non modifiable qui peut servir de clé à une map.

    Il y a une classe abstraite Tuple et deux sous-classes OrderedTuple et UnorderedTuple. Pour ces deux dernières, les éléments en nombre variables sont passés en arguments à l'unique constructeur (par la suite, on ne peut donc les modifier).

    J'ai introduit des méthodes hashCode et equals, et cette classe semble fonctionner. Le problème est que je souhaiterais en faire une classe comparable. Pour effectuer la comparaison, on itère sur les éléments du tuple. Il faudrait donc pouvoir comparer ceux-ci, mais c'est là que ça coince, vu que la méthode compareTo n'est pas connue pour le type de ces éléments.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    public abstract class Tuplelt;Kgt; implements Comparable {
     
      protected K[] elements;
     
      public K get (int n) throws RuntimeException {
        try {
          return ((K) elements[n]);
        } catch (ArrayIndexOutOfBoundsException e) {
          throw new RuntimeException(quot;elements[quot; + n + quot;] not in tuplequot;);
        }
      }
     
      public int hashCode () {
        Long L = new Long(elements[0].hashCode());
        for (int i=1; i lt; elements.length; i++) {
          long l = ((long) L.hashCode()) lt;lt; 23; // ça semble meilleur avec un premier
          l     += elements[i].hashCode();
          L      = new Long(l); 
        }
        return L.hashCode();
      }
     
      public boolean equals(Tuple t) {
        if (t == null) {
          return false;
        } else if (this.elements.length != t.elements.length) {
          return false;
        } else {
          for (int i=0; i lt; elements.length; i++) {
            if (! this.elements[i].equals(t.elements[i])) {
              return false;
            }
          }
        }
        return true;
      }
     
      public String toString () {
        StringBuffer sb = new StringBuffer(quot;[quot;);
        sb.append(elements[0]);
        for (int i=1; i lt; elements.length; i++) {
          sb.append(quot;,quot; + elements[i]);
        }
        sb.append(quot;]quot;);
        return sb.toString();
      }
     
      public int compareTo(Object o) throws ClassCastException {
        Tuple t;
        if (! (o instanceof Tuple)) {
          throw new ClassCastException(o + quot; is not a Tuplequot;);
        } else {
          t = (Tuple) o;
        }
        int i = 0;
        try {             // on itère sur les éléments   
          while (true) {  // tant qu'on n'a pas atteint la fin d'un tableau
                          // on compare les éléments
            if (! this.elements[i].equals(t.elements[i])) {
     
              // s'ils sont différents, on renvoie leur comparaison
              return this.get(i).compareTo(t.get(i));
     
            } else {
     
              // sinon, on compare les suivants
              i++;
     
            }
          }
        } catch (Exception e) {
          if (this.elements.length lt; t.elements.length)
            return -1;
          else
            return 1;
        }
      }
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    Graph/Tuple.java:66: cannot find symbol
    symbol  : method compareTo(java.lang.Object)
    location: class java.lang.Object
              return this.get(i).compareTo(t.get(i));
                                ^
    1 error
    Il me semblerait aussi logique que compareTo renvoie une exception si le type des éléments de l'instance est différent de celui du tuple passé en argument, mais, je ne vois pas du tout comment faire.

    Merci d'avance pour toute réponse,

    G.

  2. #2
    Expert éminent sénior
    Avatar de adiGuba
    Homme Profil pro
    Développeur Java/Web
    Inscrit en
    Avril 2002
    Messages
    13 938
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur Java/Web
    Secteur : Transports

    Informations forums :
    Inscription : Avril 2002
    Messages : 13 938
    Points : 23 190
    Points
    23 190
    Billets dans le blog
    1
    Par défaut
    Salut,


    C'est normal que tu ne puisses pas appeler la méthode compareTo(), car K n'est pas défini comme un objet Comparable.

    Tu as deux solutions.

    Soit définir K comme Comparable, mais dans ce cas tu ne pourras pas l'utiliser avec des objets non-comparable :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    public abstract class Tuplelt;K extends Comparablelt;Kgt;gt; ...
    Soit caster les objets K en "Comparable" avant d'appeler compareTo(), au risque de recevoir une exception si ce n'est pas le cas.


    • Le type Comparable est paramétré, et il est préférable de l'utiliser de la sorte lorsqu'on l'implémente :
      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      public abstract class Tuplelt;Kgt; implements Comparablelt;Tuplelt;Kgt;gt; { // OK
    • A quoi sert ce try/catch dans get() ? Pourquoi ne pas laisser remonter directement l'ArrayIndexOutOfBoundsException ?
    • Attention car equals(Tuple) ne redéfini pas equals(Object) ! Utilises l'annoation @Override pour être sûr
    • Tu sembes ignorer l'existance de la classe Arrays. Elle possède pourtant des implémentations bien mieux adaptés pour tes méthodes hashCode(), equals() et toString() :
      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      	@Override
      	public int hashCode() {
      		return Arrays.hashCode(this.elements);
      	}
       
      	@Override
      	public boolean equals(Object obj) {
      		if (obj==this) {
      			return true;
      		}
      		if (obj instanceof Tuplelt;?gt;) {
      			Tuplelt;?gt; other = (Tuplelt;?gt;) obj;
      			return Arrays.equals(this.elements, other.elements);
      		}
      		return false;
      	}
       
      	@Override
      	public String toString() {
      		return Arrays.toString(this.elements);
      	}



    a++

  3. #3
    Membre régulier
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    124
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Octobre 2006
    Messages : 124
    Points : 87
    Points
    87
    Par défaut
    Merci beaucoup pour ta réactivité, mais ni l'une ni l'autre solution ne fonctionne:

    Citation Envoyé par adiGuba Voir le message
    Tu as deux solutions.

    Soit définir K comme Comparable, mais dans ce cas tu ne pourras pas l'utiliser avec des objets non-comparable :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    public abstract class Tuplelt;K extends Comparablelt;Kgt;gt; ...
    Soit caster les objets K en "Comparable" avant d'appeler compareTo(), au risque de recevoir une exception si ce n'est pas le cas.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    public abstract class Tuplelt;Kgt; implements Comparablelt;Tuplelt;Kgt;gt; { // OK
    Proposition 1.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    $ javac -cp $CLASSPATH -d lib/ Graph/Tuple.java 
    Graph/Tuple.java:65: compareTo(K) in java.lang.Comparablelt;Kgt; cannot be applied to (java.lang.Comparable)
              return this.get(i).compareTo(t.get(i));
                                ^
    1 error
    Proposition 2:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    $ javac -cp $CLASSPATH -d lib/ Graph/Tuple.java 
    Graph/Tuple.java:65: cannot find symbol
    symbol  : method compareTo(java.lang.Object)
    location: class java.lang.Object
              return this.get(i).compareTo(t.get(i));
                                ^
    1 error
    En l'état, tout ce qui se passe va au-delà de mon entendement et de mes maigres compétences...

    a+,

  4. #4
    Modérateur
    Avatar de joel.drigo
    Homme Profil pro
    Ingénieur R&D - Développeur Java
    Inscrit en
    Septembre 2009
    Messages
    12 430
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 54
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Ingénieur R&D - Développeur Java
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2009
    Messages : 12 430
    Points : 29 131
    Points
    29 131
    Billets dans le blog
    2
    Par défaut
    Salut,

    Les deux extraits de code que tu as laissés dans l'extrait de citation tronquée de @AdiGuba ne sont pas les deux alternatives qu'il te proposait : le premier code est la première solution, et la seconde n'est pas accompagné d'un code. Le second bout de code était simplement un conseil d'implémentation de l'interface Comparable, qui est paramétrée. Il aurait peut être dû dire obligatoire que préférable. Utilises cette signature :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    public abstract class Tuplelt;K extends Comparablelt;Kgt;gt; implements Comparablelt;Tuplelt;Kgt;gt; {
    Ainsi, tu peux avoir une méthode de comparaison de Tuple<K>. Et elle peut appeler compareTo sur le retour de l'appel de get(int), qui renvoyant du K, impose à ce K d'être Comparable pour pouvoir avoir une méthode compareTo.
    L'expression "ça marche pas" ne veut rien dire. Indiquez l'erreur, et/ou les comportements attendus et obtenus, et donnez un Exemple Complet Minimal qui permet de reproduire le problème.
    La plupart des réponses à vos questions sont déjà dans les FAQs ou les Tutoriels, ou peut-être dans une autre discussion : utilisez la recherche interne.
    Des questions sur Java : consultez le Forum Java. Des questions sur l'EDI Eclipse ou la plateforme Eclipse RCP : consultez le Forum Eclipse.
    Une question correctement posée et rédigée et vous aurez plus de chances de réponses adaptées et rapides.
    N'oubliez pas de mettre vos extraits de code entre balises CODE (Voir Mode d'emploi de l'éditeur de messages).
    Nouveau sur le forum ? Consultez Les Règles du Club.

  5. #5
    Membre régulier
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    124
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Octobre 2006
    Messages : 124
    Points : 87
    Points
    87
    Par défaut
    Re-bonjour,

    Citation Envoyé par joel.drigo Voir le message
    Utilises cette signature :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    public abstract class Tuplelt;K extends Comparablelt;Kgt;gt; implements Comparablelt;Tuplelt;Kgt;gt; {
    Ainsi, tu peux avoir une méthode de comparaison de Tuple<K>. Et elle peut appeler compareTo sur le retour de l'appel de get(int), qui renvoyant du K, impose à ce K d'être Comparable pour pouvoir avoir une méthode compareTo.
    Désolé, ça ne fonctionne toujours pas:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    $ javac -cp $CLASSPATH -d lib/ Graph/Tuple.java
    Graph/Tuple.java:65: compareTo(K) in java.lang.Comparablelt;Kgt; cannot be applied to (java.lang.Comparable)
              return this.get(i).compareTo(t.get(i));
                                ^
    1 error

  6. #6
    Membre régulier
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    124
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Octobre 2006
    Messages : 124
    Points : 87
    Points
    87
    Par défaut (suite)
    Re-bonjour,

    Les choses évoluent un peu. Finalement, la classe-mère compile sous cette forme (ne me demandez pas pourquoi):

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    public abstract class Tuplelt;K extends Comparablegt; implements Comparable {
     
      protected K[] elements;
     
      // (autres méthodes omises)
     
      public int compareTo(Object o) throws ClassCastException {
        Tuple t;
        if (! (o instanceof Tuple)) {
          throw new ClassCastException(o + quot; is not a Tuplequot;);
        } else {
          t = (Tuple) o;
        }
        int i = 0;
        try {             // on itère sur les éléments   
          while (true) {  // tant qu'on n'a pas atteint la fin d'un des tableaux
                          // on compare les éléments
            if (! this.elements[i].equals(t.elements[i])) {
     
              // s'ils sont différents, on renvoie leur comparaison
              return this.get(i).compareTo(t.get(i));
     
            } else {
     
              // sinon, on compare les suivants
              i++;
     
            }
          }
        } catch (Exception e) {
          if (this.elements.length lt; t.elements.length)
            return -1;
          else
            return 1;
        }
      }
    }
    Par contre, la sous-classe ne compile pas:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    package Graph;
     
    public class OrderedTuplelt;K extends Comparablegt; extends Tuple {
     
      public OrderedTuple (K... elements) {
        this.elements = new K[elements.length];
        for (int i = 0; i lt; elements.length; i++)
          this.elements[i] = elements[i];
      }
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    $ javac -cp $CLASSPATH -d lib/ Graph/OrderedTuple.java
    Graph/OrderedTuple.java:6: generic array creation
        this.elements = new K[elements.length];
                        ^
    Note: Graph/OrderedTuple.java uses unchecked or unsafe operations.
    Note: Recompile with -Xlint:unchecked for details.
    1 error
    «generic array creation» ???

    a+,

  7. #7
    Modérateur
    Avatar de joel.drigo
    Homme Profil pro
    Ingénieur R&D - Développeur Java
    Inscrit en
    Septembre 2009
    Messages
    12 430
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 54
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Ingénieur R&D - Développeur Java
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2009
    Messages : 12 430
    Points : 29 131
    Points
    29 131
    Billets dans le blog
    2
    Par défaut
    Citation Envoyé par gvdmoort Voir le message
    Re-bonjour,



    Désolé, ça ne fonctionne toujours pas:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    $ javac -cp $CLASSPATH -d lib/ Graph/Tuple.java
    Graph/Tuple.java:65: compareTo(K) in java.lang.Comparablelt;Kgt; cannot be applied to (java.lang.Comparable)
              return this.get(i).compareTo(t.get(i));
                                ^
    1 error

    Oui, c'est parce que tu as écrit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    public int compareTo(Object o) throws ClassCastException {
        Tuple t;
        if (! (o instanceof Tuple)) {
          throw new ClassCastException(o + quot; is not a Tuplequot;);
        } else {
          t = (Tuple) o;
        }...
    Quand on utilise des types paramétrés, on les utilises partout sous forme paramétré :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    public int compareTo(Object o) throws ClassCastException {
        Tuplelt;Kgt; t;
        if (! (o instanceof Tuple)) {
          throw new ClassCastException(o + quot; is not a Tuplequot;);
        } else {
          t = (Tuplelt;Kgt;) o;
        }
        return compareTo(t);
    }
     
        // normalement c'est cette méthode que l'interface t'oblige à implémenter (ça compilera parce que ta classe est abstraite...)
        public int compareTo(Tuplelt;Kgt; t) throws ClassCastException {
            ...
        }
    L'expression "ça marche pas" ne veut rien dire. Indiquez l'erreur, et/ou les comportements attendus et obtenus, et donnez un Exemple Complet Minimal qui permet de reproduire le problème.
    La plupart des réponses à vos questions sont déjà dans les FAQs ou les Tutoriels, ou peut-être dans une autre discussion : utilisez la recherche interne.
    Des questions sur Java : consultez le Forum Java. Des questions sur l'EDI Eclipse ou la plateforme Eclipse RCP : consultez le Forum Eclipse.
    Une question correctement posée et rédigée et vous aurez plus de chances de réponses adaptées et rapides.
    N'oubliez pas de mettre vos extraits de code entre balises CODE (Voir Mode d'emploi de l'éditeur de messages).
    Nouveau sur le forum ? Consultez Les Règles du Club.

  8. #8
    Modérateur
    Avatar de joel.drigo
    Homme Profil pro
    Ingénieur R&D - Développeur Java
    Inscrit en
    Septembre 2009
    Messages
    12 430
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 54
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Ingénieur R&D - Développeur Java
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2009
    Messages : 12 430
    Points : 29 131
    Points
    29 131
    Billets dans le blog
    2
    Par défaut
    Citation Envoyé par gvdmoort Voir le message
    Par contre, la sous-classe ne compile pas:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    package Graph;
     
    public class OrderedTuplelt;K extends Comparablegt; extends Tuple {
     
      public OrderedTuple (K... elements) {
        this.elements = new K[elements.length];
     
    ...
    ...

    «generic array creation» ???
    On ne peut pas créer un tableau à partir d'un paramètre générique : on est obligé de mentionner un nom de classe.
    L'expression "ça marche pas" ne veut rien dire. Indiquez l'erreur, et/ou les comportements attendus et obtenus, et donnez un Exemple Complet Minimal qui permet de reproduire le problème.
    La plupart des réponses à vos questions sont déjà dans les FAQs ou les Tutoriels, ou peut-être dans une autre discussion : utilisez la recherche interne.
    Des questions sur Java : consultez le Forum Java. Des questions sur l'EDI Eclipse ou la plateforme Eclipse RCP : consultez le Forum Eclipse.
    Une question correctement posée et rédigée et vous aurez plus de chances de réponses adaptées et rapides.
    N'oubliez pas de mettre vos extraits de code entre balises CODE (Voir Mode d'emploi de l'éditeur de messages).
    Nouveau sur le forum ? Consultez Les Règles du Club.

  9. #9
    Rédacteur/Modérateur

    Avatar de bouye
    Homme Profil pro
    Information Technologies Specialist (Scientific Computing)
    Inscrit en
    Août 2005
    Messages
    6 840
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 47
    Localisation : Nouvelle-Calédonie

    Informations professionnelles :
    Activité : Information Technologies Specialist (Scientific Computing)
    Secteur : Agroalimentaire - Agriculture

    Informations forums :
    Inscription : Août 2005
    Messages : 6 840
    Points : 22 855
    Points
    22 855
    Billets dans le blog
    51
    Par défaut
    En fait, non...

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    public class OrderedTuplelt;K extends Comparablegt; extends Tuple {
     
        private final K[] elements;
     
        public OrderedTuple(K... elements) {
            this.elements = Arrays.copyOf(elements, elements.length);
        }
    }
    EDIT - ce qui revient en lisant les sources a faire un

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Array.newInstance(elements.getClass(), elements.length);
    suivi d'une recopie des éléments a mano.
    Mais bon voila, personne ne pense jamais a aller fouiller du cote des methodes des classe Arrays et Array...
    Merci de penser au tag quand une réponse a été apportée à votre question. Aucune réponse ne sera donnée à des messages privés portant sur des questions d'ordre technique. Les forums sont là pour que vous y postiez publiquement vos problèmes.

    suivez mon blog sur Développez.

    Programming today is a race between software engineers striving to build bigger and better idiot-proof programs, and the universe trying to produce bigger and better idiots. So far, the universe is winning. ~ Rich Cook

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. comparer des sons
    Par heidi79 dans le forum DirectX
    Réponses: 2
    Dernier message: 12/08/2003, 01h18
  2. [date] Comparer deux formats différents
    Par terziann dans le forum Requêtes
    Réponses: 17
    Dernier message: 21/07/2003, 15h21
  3. [LG][FAQ]comparer des fichiers
    Par lucke dans le forum Langage
    Réponses: 11
    Dernier message: 01/06/2003, 18h02
  4. [langage] Comparer Perl avec d'autres langages comme C ?
    Par Anonymous dans le forum Langage
    Réponses: 3
    Dernier message: 10/08/2002, 23h52
  5. Comparer des fichiers de données : Quel Langage ?
    Par Anonymous dans le forum Langages de programmation
    Réponses: 6
    Dernier message: 24/04/2002, 22h37

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo