在 for-each 循环中使用 select-string

Ste*_*tin 3 powershell loops select-string

我正在构建一个脚本,该脚本将访问大量文件并拉出特定字符串以便分配为变量。

所有文件中的字符串都是相似的,这不是问题。我能够使这个过程作为一个单独的事件运行(定义一个源文件)

$hostname_import = select-string .\test.txt -Pattern 'hostname ABC-.+'
$hostname = $hostname_import -replace '.+ ',''
Run Code Online (Sandbox Code Playgroud)

上面将输出目标文件中标识的特定主机名(第二个功能是修剪单词主机名和空格),然后我可以使用它继续根据需要使用变量来执行各种操作。

我遇到的问题是在 foreach 循环中执行此函数,以便我可以获取初始文件 - 执行选择字符串函数(或类似函数),然后按预期在循环内执行数据修改。

对于上下文 - 我正在查看配置文件 - 并根据这些配置构建一个单独的文件来报告结果 - 报告构建的一部分需要设备的主机名。

--编辑 1: 在咨询了我的橡皮鸭之后,我将尝试将这些文件作为 CSV 导入,以便可能找到解决方案。

  • 感谢@Otter 让我度过了难关。我以前有一个使用此函数的脚本,但由于某种原因我无法按预期执行该函数。我之前的脚本和这个脚本之间的区别是 $_.Fullname

巨大的帮助!

mkl*_*nt0 6

Select-String可以通过其或参数直接处理多个文件,作为路径数组和/或作为通配符表达式。-Path-LiteralPath

它不支持传递目录路径来处理其中的文件(更不用说递归地),因此您必须将 a Get-ChildItem(可能是 with -Recurse)的结果通过Select-String管道传递给调用。

以下示例使用后一种技术,循环遍历*.config当前目录子树中的所有文件:

Get-ChildItem -File -Recurse -Filter *.config |
  Select-String -Pattern 'hostname ABC-(.+)' |
   ForEach-Object {
     $sourceFilePath = $_.Path
     $hostName = $_.Matches[0].Groups[1].Value
   }
Run Code Online (Sandbox Code Playgroud)

(...)请注意正则表达式模式中捕获组 ( ) 的使用,它允许通过输出Microsoft.PowerShell.Commands.MatchInfo的实例从整体匹配中仅提取感兴趣的子字符串Select-String。这消除了操作的需要-replace;有关详细信息,请参阅底部部分。

请注意,每个文件可能会报告多个匹配项;如果您知道只有一个(或者只对第一个感兴趣),请添加-ListSelect-String调用中以加快操作速度。


如何仅提取匹配行/行部分的文本(字符串):

当您直接字符串上下文(例如)中使用Select-String输出对象(类型为)时,如果给出了文件参数,则生成的字符串表示形式将不仅仅包含行 文本因为输入文件路径会添加到行文本前面,后跟; 例如:Microsoft.PowerShell.Commands.MatchInfo-replace:
C:\path\to\file.config:hostname ABC-foo

仅将行文本直接提取为字符串,请使用.Line属性

  • 注意:在 PowerShell (Core) 7+ 中,您现在可以通过传递开关来要求直接Select-String输出字符串(匹配行)。-Raw

仅提取正则表达式匹配的行的一部分,请使用该属性.Matches(如果还-SimpleMatch传递了文字子字符串匹配的开关,则不适用),如上所示。

  • .Matches是实例的集合(如果传递了开关,System.Text.RegularExpressions.Match则只能有多个元素),每个实例的属性都包含与整体模式匹配的文本。-AllMatches.Value

  • 如果正则-Pattern表达式包含捕获组( (...)),则每个Match实例的.Groups集合(本身由实例组成Match)包含这些组捕获的内容,从索引开始1;该属性再次.Value包含捕获的文本