首页 > 解决方案 > 在 for-each 循环中使用选择字符串

问题描述

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

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

$hostname_import = select-string .\test.txt -Pattern 'hostname ABC-.+'
$hostname = $hostname_import -replace '.+ ',''

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

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

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

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

巨大的帮助!

标签: powershellloopsselect-string

解决方案


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

它不支持传递目录路径以处理其中的文件(更不用说递归了),因此您可以将 a Get-ChildItem(可能带有-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
   }

请注意,在正则表达式模式中使用了捕获组 ( ),它允许通过输出的实例(...)从整体匹配中仅提取感兴趣的子字符串。这消除了对操作的需要;有关详细信息,请参阅底部。Microsoft.PowerShell.Commands.MatchInfoSelect-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实例组成)包含这些组捕获的内容,从 index 开始1;该属性再次.Value包含捕获的文本


推荐阅读