1

我正在尝试创建一个附加(或创建,然后附加)到现有文件的脚本。该文件正在与另一个系统连接,因此为了避免在我追加文件时正在读取文件的情况,我正在对其创建一个非共享锁。脚本似乎可以完美运行,但有一个我无法理解的异常。这是一个代码示例:

$ErrorActionPreference = "Stop"
try
{
    [System.IO.FileStream]$file1 = [System.IO.File]::Open("test.txt", [System.IO.FileMode]::Append,[System.IO.FileAccess]::Write, [System.IO.FileShare]::None)
}
catch
{
    Write-Host "Unable to create a lock on this file"
}

有一个条件不会触发 catch 块 - 如果文件已经通过 powershell 控制台锁定,并且如果此代码是从脚本运行而不是直接在窗口中键入的。如果我首先在 Excel 中打开文件,catch 块总是有效的。如果我通过在控制台中键入此代码来执行此代码,它也始终有效。但是,如果我首先通过在控制台中键入文件来锁定文件,然后运行尝试执行相同操作的脚本,则不会引发异常并且不会触发 catch 块。

我在这里想念什么?

更新:我做了额外的故障排除,这是我的发现。首先,如何重现这个:

  • 创建一个名为 test.txt 的文件
  • 创建一个名为 的脚本test1.ps1,将上面的代码放入其中
  • 在控制台中运行此命令:

    [System.IO.FileStream]$file2 = [System.IO.File]::Open("test.txt", [System.IO.FileMode]::Append,[System.IO.FileAccess]::Write, [System.IO.FileShare]::None)
    
  • 现在执行test1.ps1脚本

并且脚本的 catch 块不会触发......在我的工作环境中!我在家里测试过,它按预期工作。所以我回去工作并再次测试它,这次使用PC本地驱动器上的文件并且它工作。使它无法工作的一个区别是我正在处理的文件位于我公司的 DFS 命名空间中。我在我的 PC 上映射的本地网络驱动器上尝试了它,它运行良好。

因此,无论出于何种原因,PowerShell 都不会像预期的那样对 DFS 命名空间上的 PowerShell 锁定的文件做出反应。我尝试过的所有其他应用程序都会检测到锁定:记事本和 Excel 拒绝读取文件,当我在 PDF 文档上测试它时,Acrobat 也不会打开它。但是PowerShell没有抱怨,甚至处理所有后续方法都没有错误——没有任何效果;CopyTo()Flush()方法不会失败,但文件没有改变。

4

1 回答 1

2

当您将上述代码作为脚本运行时,文件/句柄会在脚本(或更准确地说,PowerShell 进程)终止时自动关闭。Open()您需要保持脚本运行以保持句柄打开,例如通过在调用后添加无限循环:

$ErrorActionPreference = "Stop"
try {
    $writeStream = [IO.File]::Open($FilePath,[IO.FileMode]::Append,[IO.FileAccess]::Write, [IO.FileShare]::None)
    do {
        Start-Sleep -Milliseconds 100
    } while ($true)
} catch {
    Write-Host "Unable to create a lock on this file"
}
于 2016-10-11T00:37:25.560 回答