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 :

[awk] Déplacer des lignes d'un fichier


Sujet :

Shell et commandes GNU

  1. #1
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut [awk] Déplacer des lignes d'un fichier
    Bonjour,


    Je possède
    input
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    AB-0004 XXX XXX
    A8-0005 XXX
    A8-0006 XXX
    AB-0007 XXX XXX
    AB-0008 XXX

    et j'aimarai obtenir

    Output
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    AB-0004 XXX XXX
    AB-0007 XXX XXX
    A8-0005 XXX
    A8-0006 XXX
    AB-0008 XXX


    J'aimerai pouvoir déplacer des lignes d'un fichier vers le bas en fonction du nombre de colonnes que ce soit avec un awk ou un sed, mais sans se baser sur le numéro de la ligne.

    Si quelqu'un avait une idée ! merci beaucoup !

  2. #2
    Expert confirmé
    Avatar de becket
    Profil pro
    Informaticien multitâches
    Inscrit en
    Février 2005
    Messages
    2 854
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations professionnelles :
    Activité : Informaticien multitâches
    Secteur : Service public

    Informations forums :
    Inscription : Février 2005
    Messages : 2 854
    Par défaut
    Salut,

    Quelque chose comme ceci devrait faire l'affaire :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    awk 'BEGIN {MAX=0}{if(NF > MAX){MAX=NF};arr[NF][length(arr[NF])+1] = $0} END { for ( i = MAX ; i > 0  ; i-- ) {  for ( j = length(arr[MAX][1] ) ; j>0 ; j-- ) { if ( length(arr[i][j]) > 0 ) { print arr[i][j] } }  }}' fichier.txt

  3. #3
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Merci beaucoup de votre aide ! Mais quand je pose le script écrit avec des incrémentations, il me pose une erreur au niveau du
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    awk' 
     
    BEGIN {MAX=0}{if(NF > MAX)
            {MAX=NF};
              arr[NF][length(arr[NF])+1] = $0}
     
    END { for ( i = MAX ; i > 0  ; i-- )
           {  for ( j = length(arr[MAX][1] ) ; j>0 ; j-- )
              { if ( length(arr[i][j]) > 0 )
                { print arr[i][j] }
              }
           }
         } ' input >  output

  4. #4
    Expert confirmé Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 294
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 294
    Par défaut
    Bonjour

    $ cat fichier.txt
    AB-0004 XXX XXX
    A8-0005 XXX
    A8-0006 XXX
    AB-0007 XXX XXX
    AB-0008 XXX
    AB-0004 XXX XXXiy
    A8-0005 XXXy
    A8-0006 XXXy
    AB-0007 XXX yXXX
    AB-0008 XXXyy
    
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    $ awk '{l=length($0);q[l]++;tab[l,q[l]]=$0;} END{n=asorti(q,o,"@ind_num_desc");for (i=1;i<=n;i++) for (j=1;j<=q[o[i]];j++) print tab[o[i],j];}' fichier.txt
    AB-0004 XXX XXXiy
    AB-0007 XXX yXXX
    AB-0004 XXX XXX
    AB-0007 XXX XXX
    AB-0008 XXXyy
    A8-0005 XXXy
    A8-0006 XXXy
    A8-0005 XXX
    A8-0006 XXX
    AB-0008 XXX

  5. #5
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Merci beaucoup de ton aide ! Mais l'argument 3 est reconnu comme invalide dans awk

    awk: fatal: 3 is invalid as number of arguments for asorti

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    awk '
    {l=length($0);
      q[l]++;
      tab[l,q[l]]=$0;} 
     
    END {n=asorti(q,o,"@ind_num_desc");
               for (i=1;i<=n;i++) 
                   for (j=1;j<=q[o[i]];j++) 
                   print tab[o[i],j];}' input > output

  6. #6
    Expert confirmé Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 294
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 294
    Par défaut
    C'est ton awk qui n'est peut-être pas GNU. Que retourne la commande suivante ? (ou toute autre commande donnant la version)


  7. #7
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Voici ma version

    GNU Awk 3.1.7
    Copyright (C) 1989, 1991-2009 Free Software Foundation.

    Mais existe-il une possibilité sans l'arobase

  8. #8
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Finalement j'ai trouvé le scrip suivant

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    awk '
    { if (NF == 7) lig7=lig7"\n"$0; else ligs=ligs"\n"$0; }
    END {print lig7 ligs}
    ' input > output

  9. #9
    Expert confirmé Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 374
    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 374
    Par défaut
    Version gawk:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    awk '{A[-NF][$0]=1};END{for (i in A) for (j in A[i]) print j}' fichier
    Version posix:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    awk '{A[-NF]=A[-NF]$0"\n"};END{for (i in A) {sub(/\n$/,"",A[i]);print A[i]}}' fichier

  10. #10
    Modérateur
    Avatar de N_BaH
    Profil pro
    Inscrit en
    Février 2008
    Messages
    7 672
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Février 2008
    Messages : 7 672
    Par défaut
    Citation Envoyé par judkil
    j'ai trouvé le scrip suivant
    et on en remercie pingouinux@ubuntu-fr
    N'oubliez pas de consulter les cours shell, la FAQ, et les pages man.

  11. #11
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Oui effetctivement merci pingouinux@ubuntu-fr !

  12. #12
    Expert confirmé Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 374
    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 374
    Par défaut
    Citation Envoyé par judkil Voir le message
    Finalement j'ai trouvé le scrip suivant

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    awk '
    { if (NF == 7) lig7=lig7"\n"$0; else ligs=ligs"\n"$0; }
    END {print lig7 ligs}
    ' input > output
    Hmm, cette version ne fonctionne que pour le cas de 7 champs sur la ligne...

  13. #13
    Expert confirmé
    Avatar de becket
    Profil pro
    Informaticien multitâches
    Inscrit en
    Février 2005
    Messages
    2 854
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations professionnelles :
    Activité : Informaticien multitâches
    Secteur : Service public

    Informations forums :
    Inscription : Février 2005
    Messages : 2 854
    Par défaut
    Citation Envoyé par disedorgue Voir le message
    Hmm, cette version ne fonctionne que pour le cas de 7 champs sur la ligne...
    Je me suis fait exactement la même réflexion et je trouve plus que limite de ne pas préciser que l'on utilise une version de linux qui date d'il y a une dizaine d'année.

  14. #14
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Dans la continuité, auriez vous une idée de comment appliquer une commande uniquement sur un bloque de ligne, par exemple uniquement de la ligne 3 à 5
    J'ai essaié avec ce script

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    awk'
    for ({$i>=3})
     { print $2, $1,
          }
     ' input > output

  15. #15
    Expert confirmé Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 294
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 294
    Par défaut
    Citation Envoyé par disedorgue Voir le message
    Version gawk:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    awk '{A[-NF][$0]=1};END{for (i in A) for (j in A[i]) print j}' fichier
    Version posix:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    awk '{A[-NF]=A[-NF]$0"\n"};END{for (i in A) {sub(/\n$/,"",A[i]);print A[i]}}' fichier
    Ces solutions font disparaître les lignes doublons. N'est-ce pas ? Espérons qu'il n'y ait pas de refrain dans le fichier.

    De plus, êtes-vous vraiment sûrs que "i in a" se passera dans l'ordre croissant des indices et pas par accès aléatoire ? Ordre numérique ou ordre alphabétique ?

  16. #16
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Merci beaucoup, je ne veux pas faire disparaitre de doublons. Je voudrais juste appliquer des instructions en language awk ( et non gawk) à partir de la ligne 3 à la fin, comme switcher deux colonnes


    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    awk'
    for ({$i>=3})
     { print $2, $1,
          }
     ' input > output

  17. #17
    Expert confirmé Avatar de Flodelarab
    Homme Profil pro
    Inscrit en
    Septembre 2005
    Messages
    5 294
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente (Poitou Charente)

    Informations forums :
    Inscription : Septembre 2005
    Messages : 5 294
    Par défaut
    Le comportement attendu de ta part est l'installation d'une version plus récente quand tu réalises que la tienne est dépassée. La solution proposée marche au moins à partir de GNU awk 4.1.4.

  18. #18
    Membre actif
    Femme Profil pro
    Étudiant
    Inscrit en
    Mai 2019
    Messages
    107
    Détails du profil
    Informations personnelles :
    Sexe : Femme
    Âge : 35
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2019
    Messages : 107
    Par défaut
    Je suis pas là pour discuter de versions, dans ces cas là pourquoi ne pas le faire en python3.

    Je suis sur un pc prêté par mon bureau. J'utilise awk via un VPN (accès à distance). Dans l'ordiateur source qui est à mon bureau, il y a une ancienne version voilà.
    Là je suis chez moi, je me connecte au serveur de mon bureau et je fais awk.

    Si tu as une solution en awk archaïque c'est cool, sinon si quelqu'un a une solution je suis ok

  19. #19
    Expert confirmé Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 374
    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 374
    Par défaut
    Citation Envoyé par Flodelarab Voir le message
    Ces solutions font disparaître les lignes doublons. N'est-ce pas ? Espérons qu'il n'y ait pas de refrain dans le fichier.

    De plus, êtes-vous vraiment sûrs que "i in a" se passera dans l'ordre croissant des indices et pas par accès aléatoire ? Ordre numérique ou ordre alphabétique ?
    Oui, c'est vrai , la première fait disparaître les doublons, mais pas la deuxième.

    Et pour le doute du parcours, si sont gawk le permet, on peut rajouter PROCINFO["sorted_in"]="@ind_num_asc" mais là, on est plus dans le monde posix mais gawk:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    gawk  '{A[-NF]=A[-NF]$0"\n"};END{PROCINFO["sorted_in"]="@ind_num_asc";for (i in A) {sub(/\n$/,"",A[i]);print A[i]}}' fichier
    mais dans ce cas, autant remettre un indice classique:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    gawk  '{A[NF]=A[NF]$0"\n"};END{PROCINFO["sorted_in"]="@ind_num_desc";for (i in A) {sub(/\n$/,"",A[i]);print A[i]}}' fichier
    PS: Le problème rencontré avec les solutions à base de boucle for classique du type for (i=index;i>var;i++) sont du au fait que l'on est pas obligé de rencontrer des lignes avec 1 champs, puis, 2, puis 3, certain nombre de champs peuvent être manquant.

  20. #20
    Expert confirmé Avatar de disedorgue
    Homme Profil pro
    Ingénieur intégration
    Inscrit en
    Décembre 2012
    Messages
    4 374
    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 374
    Par défaut
    Pour choisir des lignes, tu dois utilisr le compteur interne de awk qui est la variable NR:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    awk 'NR >= 3 && NR <=5 {print $2,$1}'
    Mais comme la demande est bancale, c'est dur de faire du propre (ici, on ne sait même pas si c'est avant ou après avoir traité le cas du nombre de champs)

+ Répondre à la discussion
Cette discussion est résolue.
Page 1 sur 2 12 DernièreDernière

Discussions similaires

  1. [awk] Déplacer des lignes vers le haut
    Par Shyma dans le forum Shell et commandes GNU
    Réponses: 10
    Dernier message: 24/07/2017, 17h32
  2. Shell + awk sommer des lignes de plusieurs fichiers
    Par erara22 dans le forum Shell et commandes GNU
    Réponses: 1
    Dernier message: 23/01/2012, 12h58
  3. Shell pour supprimer des lignes d'un fichier
    Par nelsa dans le forum Autres langages
    Réponses: 2
    Dernier message: 20/09/2004, 12h26
  4. Extraire des lignes d'un fichier en commande bash
    Par newnew dans le forum Linux
    Réponses: 3
    Dernier message: 27/07/2004, 16h22
  5. Réponses: 4
    Dernier message: 24/04/2003, 22h28

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