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

Shell et commandes GNU Discussion :

Tri rapide sur un tableau d'entiers


Sujet :

Shell et commandes GNU

  1. #41
    Expert éminent sénior Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 243
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 243
    Points : 13 459
    Points
    13 459
    Par défaut
    Je viens de finir l'épreuve dont nous avons parlé, en bash, hum, en awk, et il faut expliquer comment se passer de tableau.

    Le jeu utilise un "read" pour donner les "entrées" au programme en bash (ce qui pose de nombreux problèmes pour lesquels j'ouvrirai peut-être une discussion, comme le fait que "read" supprime le premier espace entré ou qu'un texte avec des contre-obliques est saccagé)
    Voici la base donnée en point de départ:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    read N
    for (( i=0; i<N; i++ )); do
        read Pi
    done
    Il n'est pas question de tableau.

    Pour ne pas avoir à traiter les variables une à une ou par tableau, il convient de renvoyer vers la sortie... de la boucle !!! Pas celle du programme. Et de retraiter.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    read N
    for (( i=0; i<N; i++ )); do
        read Pi
        echo $Pi
    done |sort -n |awk 'blablabla'
    Et voilà une épreuve réussie sans effort.
    Cette réponse vous apporte quelque chose ? Cliquez sur en bas à droite du message.

  2. #42
    Expert éminent sénior Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 286
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Ingénieur intégration
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Décembre 2012
    Messages : 4 286
    Points : 12 742
    Points
    12 742
    Par défaut
    Pour le problème de la contre-oblique, je ne sais pas, mais pour l'espace en début de ligne, il suffit soit d'utiliser la variable que read initialise par défaut ($REPLY), soit de redéfinir l'IFS à rien:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    $ read
       essai pour REPLY
    $ echo "$REPLY"
       essai pour REPLY
    $ read VAR
       essai pour VAR sans IFS
    $ echo "$VAR"
    essai pour VAR sans IFS
    $ IFS='' read VAR
       essai pour var avec IFS
    $ echo "$VAR"
       essai pour var avec IFS
    Cordialement.

  3. #43
    Expert éminent sénior Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 243
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 243
    Points : 13 459
    Points
    13 459
    Par défaut
    Pas mal du tout. J'avais mis IFS=';'. Mais rien c'est encore mieux.
    Pour les contre-obliques, j'ai rajouté l'option -r. "read -r" n'interprète pas l'échappement.

    Mais j'ai encore des problèmes pas forcément identifiés. Comme par exemple la version de awk. a2 peut s'écrire, selon les langages, et manifestement selon les awk, a**2 ou a^2.

    Sur encore une autre épreuve, l'algorithme est bon mais plante sur la validation finale sur un texte "lorem ipsum etc...". Et je suis sûr que c'est dû à un effet de bord de bash. Un caractère spécial doit avoir un comportement spécial.

    To be continued ...
    Cette réponse vous apporte quelque chose ? Cliquez sur en bas à droite du message.

  4. #44
    Expert éminent sénior
    Avatar de Sve@r
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Février 2006
    Messages
    12 690
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Oise (Picardie)

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

    Informations forums :
    Inscription : Février 2006
    Messages : 12 690
    Points : 30 985
    Points
    30 985
    Billets dans le blog
    1
    Par défaut
    Citation Envoyé par Flodelarab Voir le message
    Mais j'ai encore des problèmes pas forcément identifiés. Comme par exemple la version de awk. a2 peut s'écrire, selon les langages, et manifestement selon les awk, a**2 ou a^2.
    Et t'as essayé a*a ??? A mon avis ça devrait fonctionner dans toutes les versions de awk...
    Mon Tutoriel sur la programmation «Python»
    Mon Tutoriel sur la programmation «Shell»
    Sinon il y en a pleins d'autres. N'oubliez pas non plus les différentes faq disponibles sur ce site
    Et on poste ses codes entre balises [code] et [/code]

  5. #45
    Membre averti
    Homme Profil pro
    [SciComp]
    Inscrit en
    Août 2013
    Messages
    134
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : [SciComp]

    Informations forums :
    Inscription : Août 2013
    Messages : 134
    Points : 323
    Points
    323
    Par défaut
    Bonsoir,

    Une propal en full bash :
    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
    declare -a tab
    dist=10000000
    read N
    for (( i=0; i<N; i++ )); do
        read Pi
        tab[$Pi]=1
    done
     
    if [ ! $N -eq ${#tab[@]} ]; then dist=0; fi
     
    if [ $dist -gt 0 ]; then
        d=10000000
        k=0
        for pi in ${!tab[*]}; 
        do
            xc=$pi
            if [ $k -eq 1 ]; then 
                d=$(($xc-$xp))
                if [ $d -lt $dist ]; then 
                  dist=$d
                fi
            fi
            xp=$xc
            k=1
        done
     
    fi
    echo $dist
    qui passe le test de rapidité (probablement de justesse, c'est assez long).
    Je suis preneur de toute remarque concernant les mauvaises pratiques et les astuces qui permettraient de rendre ce code plus beau et rapide.

    Cordialement,
    xflr6

  6. #46
    Expert confirmé
    Homme Profil pro
    Développeur informatique en retraite
    Inscrit en
    Avril 2008
    Messages
    2 101
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Côtes d'Armor (Bretagne)

    Informations professionnelles :
    Activité : Développeur informatique en retraite

    Informations forums :
    Inscription : Avril 2008
    Messages : 2 101
    Points : 5 849
    Points
    5 849
    Par défaut
    Citation Envoyé par xflr6 Voir le message
    Je suis preneur de toute remarque concernant les mauvaises pratiques et les astuces qui permettraient de rendre ce code plus beau et rapide.
    Tu dois pouvoir gagner un tout petit petit peu en n'utilisant qu'une seule variable à la place de xc et pi

    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
    declare -a tab
    dist=10000000
    read N
    for (( i=0; i<N; i++ )); do
        read Pi
        tab[$Pi]=1
    done
     
    if [ ! $N -eq ${#tab[@]} ]; then dist=0; fi
     
    if [ $dist -gt 0 ]; then
        d=10000000
        k=0
        for xc in ${!tab[*]}; # énumérer sur xc et non pi
        do
            # xc=$pi # inutile
            if [ $k -eq 1 ]; then 
                d=$(($xc-$xp))
                if [ $d -lt $dist ]; then 
                  dist=$d
                fi
            fi
            xp=$xc
            k=1
        done
     
    fi
    echo $dist

  7. #47
    Expert éminent sénior Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 286
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Ingénieur intégration
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Décembre 2012
    Messages : 4 286
    Points : 12 742
    Points
    12 742
    Par défaut
    Très joli

    Peut-être plus rapide en faisant tout les if dans let:
    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
    eclare -a tab
    dist=10000000
    read N
    for (( i=0; i<N; i++ )); do
        read Pi
        tab[$Pi]=1
    done
     
    if [ ! $N -eq ${#tab[@]} ]; then dist=0; fi
     
    if [ $dist -gt 0 ]; then
        d=10000000
        k=0
        for xc in ${!tab[*]}; # énumérer sur xc et non pi
        do
            ((xc-xp < dist ? dist=xc-xp : 0))
            xp=$xc
            k=1
        done
     
    fi
    echo $dist
    A voir...
    Cordialement.

+ Répondre à la discussion
Cette discussion est résolue.
Page 3 sur 3 PremièrePremière 123

Discussions similaires

  1. Tri rapide d'un tableau
    Par lefort dans le forum Télécharger
    Réponses: 0
    Dernier message: 30/11/2010, 15h24
  2. TRI RAPIDE D'un tableau de pointeur dure!
    Par pierrotibus dans le forum C
    Réponses: 16
    Dernier message: 05/09/2007, 20h47
  3. [Collection] Tri sur un tableau d'entier
    Par Grand sorcier dans le forum Collection et Stream
    Réponses: 2
    Dernier message: 17/07/2006, 16h07
  4. Tri Rapide sur un CLIST
    Par ensisoft dans le forum MFC
    Réponses: 9
    Dernier message: 13/12/2005, 23h22
  5. Réponses: 2
    Dernier message: 08/04/2004, 16h30

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