PowerShell的grep -f等效命令

222
我正在寻找PowerShell中与grep --file=filename相等的命令。如果您不了解grep,则filename是一个文本文件,其中每行都有一个正则表达式模式,您想要匹配它们。

也许我错过了一些明显的东西,但是Select-String似乎没有这个选项。

9个回答

243
Select-String中,-Pattern参数支持模式数组。因此,你要找的是:
Get-Content .\doc.txt | Select-String -Pattern (Get-Content .\regex.txt)

这将通过使用位于regex.txt中的每个正则表达式(每行一个)来搜索文本文件doc.txt

8
此外,PowerShell的选项卡自动完成会使命令正确地大写,因此输入不难。 - joon
4
几年迟到了,但我要补充一点,PowerShell 是不区分大小写的,因此命令的大小写不是问题。 - dee-see
1
@Vache 当然 - 我认为我上面的评论是回复另一个评论,但似乎那个评论已经消失了。 - joon
2
@dee-see 有点晚了,但是我想补充一下Windows不区分大小写,所以命令、路径的大小写不是问题。 :) - pixis

127
PS) new-alias grep findstr
PS) C:\WINDOWS> ls | grep -I -N exe

105:-a---        2006-11-02     13:34      49680 twunk_16.exe
106:-a---        2006-11-02     13:34      31232 twunk_32.exe
109:-a---        2006-09-18     23:43     256192 winhelp.exe
110:-a---        2006-11-02     10:45       9216 winhlp32.exe

PS) grep /?

30
我不明白这个为什么会有这么多票。它甚至没有回答问题。。 - Frode F.
30
我喜欢这个答案的原因是 findstr 的工作方式最像 Linux 上的 grepSelect-String 适用于处理对象,但有时你只需要匹配字符串。 - Elijah W. Gagne
11
值得注意的是,findstr 不是 PowerShell 的本地命令,而是命令提示符的命令。 - anishpatel
1
这个findstr命令已经对我可用,并且它的工作方式就像Unix的GREP。 - Lennon
6
@FrodeF。这是因为当我们搜索“PowerShell grep等效命令”时,这个问题会第一个出现。 :) - skaveesh
显示剩余2条评论

49

虽然我不熟悉 grep,但使用 Select-String 您可以做到:

Get-ChildItem filename.txt | Select-String -Pattern <regexPattern>

你也可以使用 Get-Content 来实现这个功能:

(Get-Content filename.txt) -match 'pattern'

6
将其扩展为例如 dir *.cs -Recurse | sls "TODO" | select -Unique "Path"。感谢这个很好的指示。 - Jeroen
很好知道别名是 sls,这样写起来更快。 - рüффп

13

我曾经在使用PowerShell查找文件中的文本时遇到了相同的问题。为了尽可能接近Linux环境,我使用了以下命令。

希望这能对某些人有所帮助:

PowerShell:

PS) new-alias grep findstr
PS) ls -r *.txt | cat | grep "some random string"

说明:

ls       - lists all files
-r       - recursively (in all files and folders and subfolders)
*.txt    - only .txt files
|        - pipe the (ls) results to next command (cat)
cat      - show contents of files comming from (ls)
|        - pipe the (cat) results to next command (grep)
grep     - search contents from (cat) for "some random string" (alias to findstr)

是的,这也可以正常工作:

PS) ls -r *.txt | cat | findstr "some random string"

这很不错,但我还想知道它被发现的文件名。 - Ken Seehart
要获取文件夹/文件信息,你需要将该信息作为管道传递,cat 会丢失这些信息并将所有内容转换为字符串。 你可以运行以下命令来实现你所需的功能:ls -r *.txt | select-string -pattern "some text" | Format-Table LineNumber, Filename, Line。 同样的命令,使用别名:ls -r *.txt | sls -pattern "some text" | ft LineNumber, Filename, Line - Serj
要获取文件夹/文件信息,您需要将该信息作为管道传递,cat 会丢失该信息并将所有内容转换为字符串。 您可以运行以下命令来实现所需的功能 ls -r *.txt | select-string -pattern "some text" | Format-Table LineNumber, Filename, Line。 使用别名的相同命令:ls -r *.txt | sls -pattern "some text" | ft LineNumber, Filename, Line - Serj
混合使用Linux和Windows命令:在此示例中,Linux命令“ls -la”用于列出目录中的文件,然后PowerShell命令“findstr”用于过滤包含“git”的单词的结果:“wsl ls -la | findstr" git"”。这也可以通过Windows“dir”命令与Linux grep命令混合完成:“dir | wsl grep git”。 - Serj

10

所以我在这个链接中找到了一个相当不错的答案: https://www.thomasmaurer.ch/2011/03/powershell-search-for-string-or-grep-for-powershell/

但本质上是:

Select-String -Path "C:\file\Path\*.txt" -Pattern "^Enter REGEX Here$"

这一行PowerShell命令结合了目录文件搜索(*或者你可以指定一个文件)和文件内容搜索,类似于grep。输出结果将会类似于:

这个命令可以在一个力所能及的地方进行快速搜索,并返回相关的结果。

doc.txt:31: Enter REGEX Here
HelloWorld.txt:13: Enter REGEX Here

2
他想从文件中加载模式。 - Rag
如果他所问的是那样,那么还有其他类似于我的答案,因此我觉得问题可能没有表达出期望的答案。但是在重新阅读后,我同意他可能想要将正则表达式从文件中导入到选择字符串中。 - cody.tv.weber

3

我发现PowerShell中的“filter”和“alias”可能是一种在管道输出中使用grep的方法(grep文件应该是类似的):

首先定义一个过滤器:

filter Filter-Object ([string]$pattern) {
    Out-String -InputObject $_ -Stream | Select-String -Pattern "$pattern"
}

然后定义别名:


    New-Alias -Name grep -Value Filter-Object

最后,在你的个人资料中加入前面的筛选器和别名:

$Home[My ]Documents\PowerShell\Microsoft.PowerShell_profile.ps1

重新启动您的PS,这样就可以使用它了:

alias | grep 'grep'


参考文献

  1. 别名: Set-Alias 这里 New-Alias 这里

  2. Filter (特殊函数) 这里

  3. 个人资料 (就像 bash 的 .bashrc): 这里

  4. out-string (这是关键) 这里:
    在 PowerShell 中,输出是基于对象的 这里,因此关键
    是将对象转换为字符串并对字符串进行过滤。

  5. Select-String 这里:
    在字符串和文件中查找文本


1
这个问题已经有了答案,但我想补充一点,在Windows中有一个名为WSL的Windows子系统用于Linux。例如,如果您想检查是否有名为 Elasticsearch 并处于运行状态的服务,则可以在powershell中执行类似下面的代码片段:net start | grep Elasticsearch

0
但是 select-String 似乎没有这个选项。
正确。PowerShell 不是 *nix shells 工具集的克隆。
然而,自己构建类似的工具并不难:
$regexes = Get-Content RegexFile.txt | 
           Foreach-Object { new-object System.Text.RegularExpressions.Regex $_ }

$fileList | Get-Content | Where-Object {
  foreach ($r in $regexes) {
    if ($r.IsMatch($_)) {
      $true
      break
    }
  }
  $false
}

1
这是一个糟糕的答案。Select-String -Pattern somestring 更加简洁。 - Kellen Stuart

0

可能吗?

[regex]$regex = (get-content <regex file> |
foreach {
          '(?:{0})' -f $_
        }) -join '|'

Get-Content <filespec> -ReadCount 10000 |
 foreach {
           if ($_ -match $regex)
             {
              $true
              break
             }
         }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接