【问题标题】:select-string use in a for-each loop在 for-each 循环中使用选择字符串
【发布时间】:2021-09-28 07:33:21
【问题描述】:

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

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

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

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

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

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

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

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

帮助很大!

【问题讨论】:

    标签: powershell loops select-string


    【解决方案1】:

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

    它不支持传递 directory 路径以处理其中的文件(更不用说 递归),因此您将拥有管道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
       }
    

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

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


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

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

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

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

    仅提取 正则表达式匹配的行的 部分,请使用 .Matches 属性(如果 @ 987654348@ 用于文字子串匹配的开关也通过了),如上所示。

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

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

    【讨论】:

      【解决方案2】:

      听起来您需要一个Get-ChildItem,它将获取文件夹中的所有文件,然后将输出通过管道传输到Foreach-Object 循环,该循环将引用每个文件。

      请注意,我添加了-recurse 开关,以防您还需要获取子文件夹文件。

      试试下面的:

      Get-Childitem -path "C:\PathToFolder" -recurse | Foreach-Object {
          $hostname_import = select-string $_.Fullname -Pattern 'hostname ABC-.+'
          $hostname = $hostname_import -replace '.+ ',''
          
          # The rest of your logic goes here
      }
      

      【讨论】:

      • 这行得通,但是每个输入文件调用一次Select-String 效率很低;最好将Get-ChildItem 输出直接传送到Select-String。此外,要仅对匹配行的 text 进行操作,请使用 $hostname_import.Line
      猜你喜欢
      • 2013-06-02
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-02-19
      • 2015-11-12
      • 1970-01-01
      • 1970-01-01
      • 2013-07-07
      相关资源
      最近更新 更多