Qual é o equivalente do gato de Bash no PowerShell?

7

Eu quero cat um arquivo e saido o número da linha de cada linha que ele gera.

No entanto, no PowerShell, cat gera uma matriz. Por isso, a pergunta se torna efetivamente: Como eu imprimo o índice de cada item enquanto ele está sendo enviado para o console ...?

Eu tentei algo assim:

$k = cat foo.js
$k | foreach { $index = $k.IndexOf($_) + 1; write "$index : $_"; } | more

Isso me deu alguns resultados estranhos. Alguns números de linha repetidos. O que é uma maneira elegante e mais confiável de fazer isso?

    
por deostroll 10.05.2017 / 14:57

7 respostas

5

Eu quero catar um arquivo e gerar o número da linha de cada linha que ele gera.

Use o seguinte comando:

$counter = 0; get-content .\test.txt | % { $counter++; write-host "'t$counter' $_" }

Como apontado nos comentários:

  • Pode ser melhor usar write-output em vez de write-host , pois isso permite o processamento adicional da saída.
  • echo é um alias para write-output

Assim, o comando acima se torna:

$counter = 0; get-content .\test.txt | % { $counter++; echo "'t$counter' $_" }

Exemplo de saída:

> type test.txt
foo
//approved
bar
// approved
foo
/*
approved
*/
bar

> $counter = 0; get-content .\test.txt | % { $counter++; echo "'t$counter' $_" }
        1 foo
        2 //approved
        3 bar
        4 // approved
        5 foo
        6 /*
        7 approved
        8 */
        9 bar
>

Exemplo de saída do Cygwin cat -n para comparação:

$ cat -n test.txt
     1  foo
     2  //approved
     3  bar
     4  // approved
     5  foo
     6  /*
     7  approved
     8  */
     9  bar
$
    
por 10.05.2017 / 15:25
11

Você pode abusar de Select-String para isso:

Select-String -Pattern .* -Path .\foo.txt | select LineNumber, Line

Exemplo de saída:

LineNumber Line
---------- ----
         1 a   
         2     
         3 b   
         4     
         5 c   
    
por 10.05.2017 / 17:37
1

IndexOf() corresponderá à primeira ocorrência do valor, portanto, os números de linha duplicados usando o código original significa que você tem várias linhas no arquivo idênticas. Tente o seguinte:

$k = Get-Content -Path foo.js
$l = 0 
while ($l -lt $k.length) {
    "{0,5}:  {1}" -f $l,$k[$l]
    $l++
}
    
por 10.05.2017 / 15:10
1

Qualquer coisa que você tenha que pensar por muito tempo no prompt não é elegante. Então elegância seria obter exatamente o que você precisa, colocar em um script e chamar o script quando necessário. Para resolver este problema com mais elegância e poder do que eu sozinho, usei o roteiro de Jeffrey Hicks aqui: link

Exemplo: Get-NumberedContent. \ README.txt amostra de saída:

369 | The Java(TM) Runtime Environment (JRE) and the JavaFX(TM) runtime are 
370 | products of Sun Microsystems(TM), Inc. 
371 | 
372 | Copyright © 2008 Sun Microsystems, Inc. 
373 | 4150 Network Circle, Santa Clara, California 95054, U.S.A. 
374 | All rights reserved.

Script abaixo, caso o link falhe:

#Requires -version 2.0

# Jeffery Hicks
# http://jdhitsolutions.com/blog
# follow on Twitter: http://twitter.com/JeffHicks
# "Those who forget to script are doomed to repeat their work."

#  ****************************************************************
#  * DO NOT USE IN A PRODUCTION ENVIRONMENT UNTIL YOU HAVE TESTED *
#  * THOROUGHLY IN A LAB ENVIRONMENT. USE AT YOUR OWN RISK.  IF   *
#  * YOU DO NOT UNDERSTAND WHAT THIS SCRIPT DOES OR HOW IT WORKS, *
#  * DO NOT USE IT OUTSIDE OF A SECURE, TEST SETTING.             *
#  ****************************************************************


Function Get-NumberedContent {
#Requires -version 2.0

<#
.Synopsis
    Display file contents in a numbered fashion.
.Description
    This function will display the contents of a text file as numbered output. If the file is 
    a script file, commented lines will be displayed in Green. Unlike Get-Content, the output
    is written to the console using Write-Host. This function is primarily meant as a console 
    based file viewer.It does not write to the pipeline unless you use the -PassThru parameter
    in which case you will get no colorized output. 

    For script files, or any file for that matter, you can specify a character ad the comment
    character. The default comment character is the #. Any line that begins with that chara-
    cter will be treated as a comment. You can skip comments by using -NoComment. Otherwise 
    the line will print in a green font. You can override the fontcolor using -CommentColor. 

    Use -NoBlank to suppress output of any blank lines. You can also combine -NoBlank and 
    -NoComment to get a very short numbered line output.

    Line 0 will display the full filename and path

.Parameter Filepath
    The filename and path.
.Parameter CommentCharacter
    The character to use as the comment character. The default is #. The parameter has an 
    alias of "Char".
.Parameter CommentColor
    The font color to use for commented lines. The default is green. This parameter has an
    alias of "Color"
.Parameter NoComments
    If the file is a script file, -NoComments will suppress any lines that begin with the 
    appropriate comment character.
.Parameter NoBlanks
    Suppress output of any blank lines. Line tabs and spacing will be maintained but blank
    lines will not be displayed.
.Parameter Passthru
    Write the output to the pipeline. 
.Example
    PS C:\> Get-NumberedContent c:\scripts\test.ps1

    Display line numbered content of Test.ps1 using the default comment character (#) and the
    default comment color, Green.
.Example
    PS C:\> Get-NumberedContent c:\scripts\update.vbs -nocomment -char "'"

    Display the results of update.vbs without and lines that start with the comment character
    for VBS scripts. This expression is using the parameter alias CHAR for -CommentCharacter.
.Example
    PS C:\> get-numberedcontent c:\files\report.ext -noblanks -pass | out-file NumReport.txt

    Display the contents of c:\files\report.txt without any blank lines and pass to the pipeline.
    The pipelined output is then sent to the Out-File cmdlet.
.Example
    PS C:\> dir c:\TEST\*.CSV | get-numberedcontent -commentCharacter ";" -commentColor "Red"  -noblanks

    Get the content for every CSV file in the Test directory. Commented lines that start with ;
    will be displayed in a red color and blank lines will be suppressed.

.Inputs
    Accepts strings as pipelined input
.Outputs
    None

.Link
   Get-Content


.Notes
 NAME:      Get-NumberedContent
 VERSION:   2.0
 AUTHOR:    Jeffery Hicks
            http://jdhitsolutions.com/blog
 LASTEDIT:  10/13/2009 


#>


[CmdletBinding()]

    param (
        [Parameter(
         ValueFromPipeline=$True,
         Position=0,
         Mandatory=$True,
         HelpMessage="The filename and path of a text file.")] 
         [string]$Filename,

         [Parameter(
         ValueFromPipeline=$False,
         Mandatory=$False,
         HelpMessage="The comment character for a specific file type.")] 
         [Alias("Char")]
         [string]$CommentCharacter="#",

         [Parameter(
         ValueFromPipeline=$False,
         Mandatory=$False,
         HelpMessage="The comment character color. Default is Green.")] 
         [ValidateSet("Black","DarkBlue","Blue","DarkGreen","Green","DarkCyan","Cyan",
         "DarkRed","Red","Magenta","White","DarkGray","Gray","DarkYellow","Yellow")] 
         [Alias("Color")]
         [string]$CommentColor="Green",

         [Parameter(
         ValueFromPipeline=$False,
         Mandatory=$False,
         HelpMessage="Suppress comment lines for script files.")] 
         [switch]$NoComment,

         [Parameter(
         ValueFromPipeline=$False,
         Mandatory=$False,
         HelpMessage="Suppress blank lines.")] 
         [switch]$NoBlank,

         [Parameter(
         ValueFromPipeline=$False,
         Mandatory=$False,
         HelpMessage="Write object to the pipeline instead of the console.")] 
         [switch]$Passthru

         )

Begin {
    if ($NoComment) { Write-Debug "No comments"}
    if ($NoBlank) {Write-Debug "No blank lines"}
    Write-Debug "Comment character is #CommentCharacter"
    Write-Debug "Comment color is $CommentColor"
    if ($passthru) {Write-Debug "Passthru"}

} #end Begin

Process {

    if ($_) {
        $Filename=$_
    $FullName=$_.Fullname
    }
    else {
    $Fullname=$Filename
    }

    write-debug "Testing $filename"
    If (Test-Path $filename) {
        $counter = -1

        write-debug "Getting content"
        $content=get-content $Filename

        #get the total number of lines and then the length
        #of that number so the number of leading zeros can be
        #calculated more accurately
        write-debug "Calculating number of lines"
        $c=($content.count).ToSTring().Length

        write-debug "Padding line numbers to $c places"
        write-debug "Processing content"
        $content | foreach { 
            #default font color
            $fcolor="White"

            #determine if line is a blank
            if ($_.Trim().Length -gt 0) {
                $Empty=$False
                write-debug "Line is not empty"
            }
            else {
                write-debug "Line is empty"
                $Empty=$True
             }


             #determine if line is a comment

            $isComment=$False

             if ($_.Trim().StartsWith($CommentCharacter))  {
                   write-debug "Comment line found"
                   $fcolor=$CommentColor
                   $isComment=$True
                }


            if (($NoBlank -AND $Empty) -OR ($NoComment -AND $IsComment )) {
                write-debug "Skipping line"
              }

            else {

         $counter++

                if ($counter -eq 0) {
                    $line = "{0:d$($c)} | {1}" -f $counter,$FullName.ToUpper() 
            $fcolor="White"               
                }

                else {

                    #write a line number with leading zeros the | bar and then the line of text from the file
                    #trimming off any trailing spaces
                    $line = "{0:d$($c)} | {1}" -f $counter,$_.TrimEnd()
                }

                if ($Passthru) {
                    write $line
                }
                else {
                   Write-Host $line -foregroundcolor $fcolor
                }
            } #else not a blank line


         } #end ForEach
    } #end if Test-Path
    else {
        Write-Warning "Failed to find $filename"
        }
  } #end Process

 End {
  Write-Debug "Ending and exiting"

 }

 } #end function

Set-Alias gnc Get-NumberedContent
    
por 10.05.2017 / 16:01
1

Semelhante ao código de DavidPostill, mas com o número justificado à direita, como cat -n

$cnt=0;gc .\test.txt|%{$cnt++;"{0,6} {1}" -f $cnt,$_}

Ou com o mesmo resultado:

select-string -path .\test.txt "^" |%{"{0,6} {1}" -f $_.LinenUmber,$_.Line}

Exemplo de saída:

PS> $cnt=0;gc .\test.txt |%{$cnt++;"{0,6} {1}" -f $cnt,$_}
     1 foo
     2 //approved
     3 bar
     4 // approved
     5 foo
     6 /*
     7 approved
     8 */
     9 bar
    
por 10.05.2017 / 19:26
1

cat no PowerShell é na verdade um alias para Get-Content . Você pode ver isso em Get-Alias cat . Muitos dos comandos nix simples ofereciam equivalentes de PS para facilitar a entrada dos usuários no PowerShell. Eles não são espelhos perfeitos, mas tentam.

Além disso, não é necessário fazer nenhum trabalho sofisticado com Get-Content é a saída para calcular os números de linha. Isso já é feito para você pelo cmdlet.

Get-Content C:\temp\pingtst.csv | ForEach-Object{"$($_.readcount):  $_"} 

Concedido que a saída não seja perfeita e esteja alinhada à esquerda, mas você pode corrigir isso rolando suas próprias funções e cmdlets. O PowerShell trabalha com desempenho máximo com objetos, por isso, transformar o arquivo em outra coisa ficaria assim:

PS C:\Windows\system32> Get-Content C:\temp\pingtst.csv | Select-Object ReadCount,@{Name="Line";Expression={"$_"}}

ReadCount Line      
--------- ----      
        1 localhost 
        2 localhost0
        3 localhost1
        4 localhost2

Lembre-se de que há muito mais opções para ajudar, como -Head , -Tail , -TotalCount etc., que podem adicionar funcionalidade a esse cmdlet aparentemente simples.

Embora eu tenha certeza que não é exatamente o que você estava esperando. O ponto é que Get-Content já conhece os números das linhas, portanto não há necessidade de contagens ou nada disso.

    
por 11.05.2017 / 03:01
1
Get-Content -Path D:\in\demo.txt | % { "{0,5} {1}" -f $PSItem.Readcount, $PSItem }

Ou talvez o seguinte, para garantir uma linha (na saída do PowerShell) por linha (no arquivo).

Get-Content -Path D:\in\demo.txt -ReadCount 1 | % { "{0,5} {1}" -f $PSItem.Readcount, $PSItem }
    
por 11.05.2017 / 03:57