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 Perl Discussion :

Convertir liste plate en hash spécifique


Sujet :

Langage Perl

  1. #1
    Responsable Perl et Outils

    Avatar de djibril
    Homme Profil pro
    Inscrit en
    Avril 2004
    Messages
    19 820
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Avril 2004
    Messages : 19 820
    Points : 498 771
    Points
    498 771
    Par défaut Convertir liste plate en hash spécifique
    Bonjour,

    Question bête :
    à partir d'une liste my @liste = qw / a b c d/;comment créer le hash suivant $hash->{a}{b}{c}{d} = undef;?

  2. #2
    Responsable Perl et Outils

    Avatar de djibril
    Homme Profil pro
    Inscrit en
    Avril 2004
    Messages
    19 820
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Avril 2004
    Messages : 19 820
    Points : 498 771
    Points
    498 771
    Par défaut
    La liste peut être de taille quelconque.

  3. #3
    Expert confirmé

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2009
    Messages
    3 577
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 58
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Avril 2009
    Messages : 3 577
    Points : 5 753
    Points
    5 753
    Par défaut
    Ce n'est pas "strictement" ce que tu veux (le undef est remplacé par un hash vide) :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
     
    $ perl -E 'my @liste = qw /a b c d/;my %hash;my $hash = \%hash;foreach my $level (@liste) { $hash->{$level} = {}; $hash = $hash->{$level} };use Data::Dumper; say Data::Dumper->Dump([\%hash],[qw(*hash)])'                                                             %hash = (
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => {}
                                       }
                              }
                     }
            );
    (normal que j'arrive plus à utiliser les boutons d'édition de message, par exemple pour la balise CODE ?)
    Plus j'apprends, et plus je mesure mon ignorance (philou67430)
    Toute technologie suffisamment avancée est indiscernable d'un script Perl (Llama book)
    Partagez vos problèmes pour que l'on partage ensemble nos solutions : je ne réponds pas aux questions techniques par message privé
    Si c'est utile, say

  4. #4
    Rédacteur/Modérateur

    Avatar de Lolo78
    Homme Profil pro
    Conseil - Consultant en systèmes d'information
    Inscrit en
    Mai 2012
    Messages
    3 612
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Yvelines (Île de France)

    Informations professionnelles :
    Activité : Conseil - Consultant en systèmes d'information
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mai 2012
    Messages : 3 612
    Points : 12 256
    Points
    12 256
    Billets dans le blog
    1
    Par défaut
    Hum, réponse pas très propre (et pas testée)...

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    my @liste = qw / a b c d/;
    @liste = map { "{$_}"} @liste;
    my $string = join '', '$hash->', @liste, " = undef";
    eval $string;
    Bon, finalement, un petit test sous le debugger:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     x $hash
    0  HASH(0x8032f028)
       'a' => HASH(0x80431588)
          'b' => HASH(0x804315b8)
             'c' => HASH(0x804315e8)
                'd' => undef

  5. #5
    Membre confirmé
    Avatar de cmcmc
    Homme Profil pro
    Inscrit en
    Juillet 2013
    Messages
    316
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Juillet 2013
    Messages : 316
    Points : 641
    Points
    641
    Par défaut
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    Taisha:~/tttmp $ perl -MData::Dumper -e 'sub to_hash { my $c; $c = { $_ => $c } for reverse @_; $c }; print Dumper(to_hash(qw(a b c d)))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    Taisha:~/tttmp $
    Sauf indication contraire tous les codes que je présente sont utilisables et testés (mais sans garantie d'aucune sorte)
    J'apporte beaucoup de soin à la rédaction de mes posts et apprécie les retours donc merci de s'il vous paraissent pertinents ou utiles
    Lazyness, Impatience and Hubris are good for you

  6. #6
    Expert confirmé

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2009
    Messages
    3 577
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 58
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Avril 2009
    Messages : 3 577
    Points : 5 753
    Points
    5 753
    Par défaut
    Merci cmcmc
    Plus j'apprends, et plus je mesure mon ignorance (philou67430)
    Toute technologie suffisamment avancée est indiscernable d'un script Perl (Llama book)
    Partagez vos problèmes pour que l'on partage ensemble nos solutions : je ne réponds pas aux questions techniques par message privé
    Si c'est utile, say

  7. #7
    Responsable Perl et Outils

    Avatar de djibril
    Homme Profil pro
    Inscrit en
    Avril 2004
    Messages
    19 820
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Avril 2004
    Messages : 19 820
    Points : 498 771
    Points
    498 771
    Par défaut
    Merci à vous trois.

    @philou, ce code peut toujours servir pour réutiliser ton code au niveau des sommaire en partant d'un hash !

  8. #8
    Membre confirmé
    Avatar de cmcmc
    Homme Profil pro
    Inscrit en
    Juillet 2013
    Messages
    316
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Juillet 2013
    Messages : 316
    Points : 641
    Points
    641
    Par défaut
    Citation Envoyé par cmcmc Voir le message
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    Taisha:~/tttmp $ perl -MData::Dumper -e 'sub to_hash { my $c; $c = { $_ => $c } for reverse @_; $c }; print Dumper(to_hash(qw(a b c d)))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    Taisha:~/tttmp $
    variante sans reverse
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    Taisha:~/tttmp $ perl -MData::Dumper -e 'sub to_hash { my $c = undef; $c = { pop @_ => $c } while @_; $c } print Dumper(to_hash(qw(a b c d)))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    Taisha:~/tttmp $
    variante récursive (ma préférée )
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    Taisha:~/tttmp $ perl -MData::Dumper -E 'sub to_hash { @_ ? { shift(@_) => to_hash(@_) } : undef } say Dumper(to_hash(qw(a b c d)))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    
    Taisha:~/tttmp $
    Sauf indication contraire tous les codes que je présente sont utilisables et testés (mais sans garantie d'aucune sorte)
    J'apporte beaucoup de soin à la rédaction de mes posts et apprécie les retours donc merci de s'il vous paraissent pertinents ou utiles
    Lazyness, Impatience and Hubris are good for you

  9. #9
    Membre confirmé
    Avatar de cmcmc
    Homme Profil pro
    Inscrit en
    Juillet 2013
    Messages
    316
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Juillet 2013
    Messages : 316
    Points : 641
    Points
    641
    Par défaut
    j'ai changé d'avis , je préfère celle là (récursive aussi) :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    Taisha:~/tttmp $ perl -MData::Dumper -E 'sub to_hash { $_[0] && { shift(@_) => to_hash(@_) } } say Dumper(to_hash(qw(a b c d)))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    
    Taisha:~/tttmp $
    Mais attention, elle ne supporte pas 0 comme clé
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    Taisha:~/tttmp $ perl -MData::Dumper -E 'sub to_hash { $_[0] && { shift(@_) => to_hash(@_) } } say Dumper(to_hash(qw(a 0 c d)))'
    $VAR1 = {
              'a' => '0'
            };
     
    Taisha:~/tttmp $
    en fait la version suivante serait plus robuste mais malheureusement la feuille produite contient 0 et non undef...
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    Taisha:~/tttmp $ perl -MData::Dumper -E 'sub to_hash { @_ && { shift(@_) => to_hash(@_) } } say Dumper(to_hash(qw(a 0 c d)))'  C-c C-z
    $VAR1 = {
              'a' => {
                       '0' => {
                                'c' => {
                                         'd' => 0
                                       }
                              }
                     }
            };
     
    Taisha:~/tttmp $
    Sauf indication contraire tous les codes que je présente sont utilisables et testés (mais sans garantie d'aucune sorte)
    J'apporte beaucoup de soin à la rédaction de mes posts et apprécie les retours donc merci de s'il vous paraissent pertinents ou utiles
    Lazyness, Impatience and Hubris are good for you

  10. #10
    Membre confirmé
    Avatar de cmcmc
    Homme Profil pro
    Inscrit en
    Juillet 2013
    Messages
    316
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Juillet 2013
    Messages : 316
    Points : 641
    Points
    641
    Par défaut
    Pour le sport, une version Inline::C
    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
    # -*- Mode: cperl; coding: utf-8; tab-width: 8; indent-tabs-mode: nil; cperl-indent-level: 4 -*-
    # to_hash_inline_C1.pl
    use strict;
    use warnings;
     
    use Inline C => <<'END_OF_C';
    SV* to_hash(SV* key1, ...) {
        Inline_Stack_Vars;
        int i;
        SV* ret = newSV(0);
     
        for (i = Inline_Stack_Items; i--; ) {
    	HV* hash = newHV();
    	hv_store_ent(hash, Inline_Stack_Item(i), ret, 0);
    	ret = newRV_noinc((SV*) hash);
        }
        return ret;
    }
    END_OF_C
        ;
     
     
    use Data::Dumper;
    print Dumper(to_hash(@ARGV));
    seul inconvénient, la liste d'entrée ne peut pas être vide à cause d'une limitation du compilateur C (lorsqu'on utilise une liste variable d'arguments, on est obligé de déclarer au moins 1 argument ).
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    Taisha:~/tttmp/to_hash $ perl to_hash_inline_C1.pl a b c d
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    Taisha:~/tttmp/to_hash $
    Sauf indication contraire tous les codes que je présente sont utilisables et testés (mais sans garantie d'aucune sorte)
    J'apporte beaucoup de soin à la rédaction de mes posts et apprécie les retours donc merci de s'il vous paraissent pertinents ou utiles
    Lazyness, Impatience and Hubris are good for you

  11. #11
    Expert confirmé

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2009
    Messages
    3 577
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 58
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Avril 2009
    Messages : 3 577
    Points : 5 753
    Points
    5 753
    Par défaut
    C'est sans doute la plus rapide ...
    Plus j'apprends, et plus je mesure mon ignorance (philou67430)
    Toute technologie suffisamment avancée est indiscernable d'un script Perl (Llama book)
    Partagez vos problèmes pour que l'on partage ensemble nos solutions : je ne réponds pas aux questions techniques par message privé
    Si c'est utile, say

  12. #12
    Membre confirmé
    Avatar de cmcmc
    Homme Profil pro
    Inscrit en
    Juillet 2013
    Messages
    316
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations forums :
    Inscription : Juillet 2013
    Messages : 316
    Points : 641
    Points
    641
    Par défaut
    Citation Envoyé par Philou67430 Voir le message
    C'est sans doute la plus rapide ...
    C'est ironique ?

    Elle est effectivement un poil plus rapide, mais pas spectaculairement, ce qui est normal parce qu'elle fait essentiellement la même chose que l'interpréteur... Ce n'était pas l'objet.

    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
    # -*- Mode: cperl; coding: utf-8; tab-width: 8; indent-tabs-mode: nil; cperl-indent-level: 4 -*-
    # to_hash_test.pl
    use strict;
    use warnings;
    use Data::Dumper;
    use Devel::Leak;
     
    use Inline C => <<'END_OF_C';
    SV* to_hash_inline(SV* key1, ...) {
        Inline_Stack_Vars;
        int i;
        SV* ret = newSV(0);
     
        for (i = Inline_Stack_Items; i--; ) {
    	HV* hash = newHV();
    	hv_store_ent(hash, Inline_Stack_Item(i), ret, 0);
    	ret = newRV_noinc((SV*) hash);
        }
        return ret;
    }
    END_OF_C
        ;
     
    sub to_hash_reverse { my $c; $c = { $_ => $c } for reverse @_; $c }
    sub to_hash_pop { my $c = undef; $c = { pop @_ => $c } while @_; $c }
    sub to_hash_recursive { no warnings 'recursion'; @_ ? { shift(@_) => to_hash_recursive(@_) } : undef }
    sub to_hash_altrec { no warnings 'recursion'; $_[0] && { shift(@_) => to_hash_altrec(@_) } }
    sub to_hash_while { my $c; my $i = @_; $c = { $_[$i] => $c } while $i--; $c }
     
    my %dispatch = ( inline    => \&to_hash_inline,
                     reverse   => \&to_hash_reverse,
                     pop       => \&to_hash_pop,
                     recursive => \&to_hash_recursive,
                     altrec    => \&to_hash_altrec,
                     while     => \&to_hash_while,
                   );
     
    my $name     = shift || q{inline};
    my $size     = shift || 100;
    my $cnt      = shift || 1;
    my $testleak = shift;
     
    exists $dispatch{$name} or die "usage : $0 [name [size [cnt [testleak]]]]\n",
      "  name: ", (join ", ", keys %dispatch);
     
     
    my @list = 1 .. $size;
    my $handle;
     
    my ($before, $after);
    $before = Devel::Leak::NoteSV($handle) if $testleak;
    while ($cnt--) { my $h = $dispatch{$name}->(@list) };
    $after  = Devel::Leak::CheckSV($handle) if $testleak;
    print "$before $after\n" if $testleak;
    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
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl inline 100 100000
     
    real	0m11.438s
    user	0m0.015s
    sys	0m0.015s
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl reverse 100 100000
     
    real	0m13.841s
    user	0m0.015s
    sys	0m0.015s
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl while 100 100000
     
    real	0m14.316s
    user	0m0.015s
    sys	0m0.015s
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl pop 100 100000
     
    real	0m14.041s
    user	0m0.047s
    sys	0m0.000s
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl recursive 100 100000
     
    real	0m18.402s
    user	0m0.047s
    sys	0m0.000s
    Taisha:~/tttmp/to_hash $ time perl to_hash_test.pl altrec 100 100000
     
    real	0m17.717s
    user	0m0.015s
    sys	0m0.015s
    Taisha:~/tttmp/to_hash $
    En fait j'ai un problème avec les deux versions récursives qui semblent causer des fuites de mémoire, que je n'ai pas encore eu le temps d'investiguer... J'ai codé la version inline pour me rafraîchir un peu sur le sujet.

    Oh, et pour mémoire, encore une version pure perl :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    Taisha:~/tttmp/to_hash $ perl -MData::Dumper -e 'sub to_hash { my $c; my $i = @_; $c = { $_[$i] => $c } while $i--; $c }; print Dumper(to_hash(qw/a b c d/))'
    $VAR1 = {
              'a' => {
                       'b' => {
                                'c' => {
                                         'd' => undef
                                       }
                              }
                     }
            };
    Taisha:~/tttmp/to_hash $
    Sauf indication contraire tous les codes que je présente sont utilisables et testés (mais sans garantie d'aucune sorte)
    J'apporte beaucoup de soin à la rédaction de mes posts et apprécie les retours donc merci de s'il vous paraissent pertinents ou utiles
    Lazyness, Impatience and Hubris are good for you

  13. #13
    Expert confirmé

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2009
    Messages
    3 577
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 58
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Avril 2009
    Messages : 3 577
    Points : 5 753
    Points
    5 753
    Par défaut
    Citation Envoyé par cmcmc Voir le message
    C'est ironique ?
    Juste humoristique, ce n'était pas moqueur
    Plus j'apprends, et plus je mesure mon ignorance (philou67430)
    Toute technologie suffisamment avancée est indiscernable d'un script Perl (Llama book)
    Partagez vos problèmes pour que l'on partage ensemble nos solutions : je ne réponds pas aux questions techniques par message privé
    Si c'est utile, say

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

Discussions similaires

  1. Convertir List en Vector
    Par michaelbob dans le forum Collection et Stream
    Réponses: 12
    Dernier message: 10/05/2011, 16h25
  2. Réponses: 1
    Dernier message: 15/01/2010, 16h20
  3. Convertir List<MonObjet> en List<Objet>
    Par Arthis dans le forum C#
    Réponses: 2
    Dernier message: 05/12/2008, 14h48
  4. Listes liées avec rechargement spécifique
    Par jpascal dans le forum Langage
    Réponses: 2
    Dernier message: 05/04/2008, 02h32
  5. [XStream] convertir liste de javabean en xml
    Par jeb001 dans le forum Format d'échange (XML, JSON...)
    Réponses: 2
    Dernier message: 04/01/2008, 09h19

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