Como mover partes e linhas repetidas de código c ++ usando awk ou sed?

Nov 25 2020

Eu tenho um grande pedaço de código C ++ com milhares de linhas como esta:

   case 14: //OrderSelect
      Execute_OrderSelect();
   break;
   case 15: // OrderGetDouble
      Execute_OrderGetDouble();
   break;
   case 16:   //OrderGetInteger
      Execute_OrderGetInteger();
   break;

Minha tarefa é torná-los assim:

   case 14: Execute_OrderSelect();     break;   // OrderSelect
   case 15: Execute_OrderGetDouble();  break;   // OrderGetDouble
   case 16: Execute_OrderGetInteger(); break;   // OrderGetInteger

Observe que tanto Execute ... quanto comentários podem ser qualquer string.


Suponho que esquematicamente poderíamos escrever o original assim:

AAA NN BBB
CCC
DDD

e tentar transformá-lo em: AAA NN CCC DDD BBB.

Tentei sem sucesso com todos os tipos de sedexpressões e o melhor que pude fazer foi a operação trivial de combinar o Execute...()com o break;, mas não fui capaz de mover o comentário. Estou pensando que estou usando a ferramenta errada para isso e talvez awkseja uma opção melhor ou mais simples de usar?

Aqui estão algumas awkvariáveis:

FNR    The input record number in the current input file.
FS     The input field separator, a space by default.
NF     The number of fields in the current input record.
NR     The total number of input records seen so far.
OFMT   The output format for numbers, "%.6g", by default.
OFS    The output field separator, a space by default.
ORS    The output record separator, by default a newline.
RS     The input record separator, by default a newline.
RT     The  record terminator. Gawk sets RT to the input 
       text that matched the character or regular expression 
       specified by RS.
RSTART The index of the first character matched by match(); 0 if no match

Como posso tornar meu dia mais brilhante?


Perguntas relacionadas:

  • Forma AWK ou sed para colar linhas não adjacentes
  • mudar a posição de uma linha em um arquivo usando sed
  • Como selecionar linhas entre dois padrões de marcadores que podem ocorrer várias vezes com awk / sed

Respostas

2 EdMorton Nov 25 2020 at 05:34

Aqui estão os ossos, massagem adequada;

$ cat tst.awk /^[[:space:]]*case[[:space:]]/ { comment = "" if ( match($0,"//") ) {
        comment = substr($0,RSTART) $0 = substr($0,1,RSTART-1) } caseLineNr = 1 } caseLineNr { if ( caseLineNr++ > 1 ) { sub(/^[[:space:]]+/,"") } sub(/[[:space:]]+$/,"")
    printf "%s\t", $0

    if ( /^break[[:space:]]*;/ ) {
        print comment
        caseLineNr = 0
    }
}

$ awk -f tst.awk file
   case 14:     Execute_OrderSelect();  break;  //OrderSelect
   case 15:     Execute_OrderGetDouble();       break;  // OrderGetDouble
   case 16:     Execute_OrderGetInteger();      break;  //OrderGetInteger