65

在 PowerShell 中使用,如何检查应用程序是否正在锁定文件?

我喜欢检查哪个进程/应用程序正在使用该文件,以便我可以关闭它。

4

10 回答 10

47

您可以使用SysInternals 工具 handle.exe执行此操作。尝试这样的事情:

PS> $handleOut = handle
PS> foreach ($line in $handleOut) { 
        if ($line -match '\S+\spid:') {
            $exe = $line
        } 
        elseif ($line -match 'C:\\Windows\\Fonts\\segoeui\.ttf')  { 
            "$exe - $line"
        }
     }
MSASCui.exe pid: 5608 ACME\hillr -   568: File  (---)   C:\Windows\Fonts\segoeui.ttf
...
于 2009-06-05T23:35:56.417 回答
20

您应该能够从常规命令行或 PowerShell使用openfiles 命令。

openfiles 内置工具可用于文件共享或本地文件。对于本地文件,您必须打开该工具并重新启动机器(同样,仅适用于第一次使用)。我相信打开此功能的命令是:

openfiles /local on

例如(适用于 Windows Vista x64):

openfiles /query | find "chrome.exe"

成功返回与 Chrome 关联的文件句柄。您还可以传入文件名以查看当前访问该文件的进程。

于 2009-06-05T21:11:56.537 回答
16

这可以帮助您:使用 PowerShell 找出锁定文件的进程。它解析每个进程的 System.Diagnostics.ProcessModuleCollection Modules 属性并查找锁定文件的文件路径:

$lockedFile="C:\Windows\System32\wshtcpip.dll"
Get-Process | foreach{$processVar = $_;$_.Modules | foreach{if($_.FileName -eq $lockedFile){$processVar.Name + " PID:" + $processVar.id}}}
于 2012-11-22T08:39:21.773 回答
10

您可以使用SysinternalHandle实用程序找到解决方案。

我不得不(稍微)修改代码以使用 PowerShell 2.0:

#/* http://jdhitsolutions.com/blog/powershell/3744/friday-fun-find-file-locking-process-with-powershell/ */
Function Get-LockingProcess {

    [cmdletbinding()]
    Param(
        [Parameter(Position=0, Mandatory=$True,
        HelpMessage="What is the path or filename? You can enter a partial name without wildcards")]
        [Alias("name")]
        [ValidateNotNullorEmpty()]
        [string]$Path
    )

    # Define the path to Handle.exe
    # //$Handle = "G:\Sysinternals\handle.exe"
    $Handle = "C:\tmp\handle.exe"

    # //[regex]$matchPattern = "(?<Name>\w+\.\w+)\s+pid:\s+(?<PID>\b(\d+)\b)\s+type:\s+(?<Type>\w+)\s+\w+:\s+(?<Path>.*)"
    # //[regex]$matchPattern = "(?<Name>\w+\.\w+)\s+pid:\s+(?<PID>\d+)\s+type:\s+(?<Type>\w+)\s+\w+:\s+(?<Path>.*)"
    # (?m) for multiline matching.
    # It must be . (not \.) for user group.
    [regex]$matchPattern = "(?m)^(?<Name>\w+\.\w+)\s+pid:\s+(?<PID>\d+)\s+type:\s+(?<Type>\w+)\s+(?<User>.+)\s+\w+:\s+(?<Path>.*)$"

    # skip processing banner
    $data = &$handle -u $path -nobanner
    # join output for multi-line matching
    $data = $data -join "`n"
    $MyMatches = $matchPattern.Matches( $data )

    # //if ($MyMatches.value) {
    if ($MyMatches.count) {

        $MyMatches | foreach {
            [pscustomobject]@{
                FullName = $_.groups["Name"].value
                Name = $_.groups["Name"].value.split(".")[0]
                ID = $_.groups["PID"].value
                Type = $_.groups["Type"].value
                User = $_.groups["User"].value.trim()
                Path = $_.groups["Path"].value
                toString = "pid: $($_.groups["PID"].value), user: $($_.groups["User"].value), image: $($_.groups["Name"].value)"
            } #hashtable
        } #foreach
    } #if data
    else {
        Write-Warning "No matching handles found"
    }
} #end function

例子:

PS C:\tmp> . .\Get-LockingProcess.ps1
PS C:\tmp> Get-LockingProcess C:\tmp\foo.txt

Name                           Value
----                           -----
ID                             2140
FullName                       WINWORD.EXE
toString                       pid: 2140, user: J17\Administrator, image: WINWORD.EXE
Path                           C:\tmp\foo.txt
Type                           File
User                           J17\Administrator
Name                           WINWORD

PS C:\tmp>
于 2015-07-24T17:25:31.680 回答
7

我也在寻找解决方案,但遇到了一些问题。

  1. 不想使用外部应用程序
  2. 打开文件需要本地 ON 属性,这意味着必须将系统配置为在执行之前使用它。

经过广泛的搜索,我找到了。

https://github.com/pldmgg/misc-powershell/blob/master/MyFunctions/PowerShellCore_Compatible/Get-FileLockProcess.ps1

感谢保罗·迪马乔

这似乎是纯 powershell 和 .net / C#

于 2020-05-19T13:20:43.157 回答
1

在 PsGallery 中发布了一个 PowerShell 模块,以发现和终止对文件或文件夹具有打开句柄的进程。它将函数暴露于:1) 找到锁定进程,以及 2) 终止锁定进程。该模块在首次使用时自动下载handle.exe。

Find-LockingProcess()
检索具有打开到指定路径的文件句柄的进程信息。
示例:Find-LockingProcess -Path $Env:LOCALAPPDATA
示例:Find-LockingProcess -Path $Env:LOCALAPPDATA | 获取进程

Stop-LockingProcess()
杀死所有打开指定路径的文件句柄的进程。
示例:Stop-LockingProcess -Path $Home\Documents

PsGallery 链接:https ://www.powershellgallery.com/packages/LockingProcessKiller 安装运行:
Install-Module -Name LockingProcessKiller

于 2021-08-26T03:51:03.877 回答
0

我在仅使用 PowerShell 和 .NET 框架类的锁定文件检测中看到了一个不错的解决方案:

function TestFileLock {
    ## Attempts to open a file and trap the resulting error if the file is already open/locked
    param ([string]$filePath )
    $filelocked = $false
    $fileInfo = New-Object System.IO.FileInfo $filePath
    trap {
        Set-Variable -name filelocked -value $true -scope 1
        continue
    }
    $fileStream = $fileInfo.Open( [System.IO.FileMode]::OpenOrCreate,[System.IO.FileAccess]::ReadWrite, [System.IO.FileShare]::None )
    if ($fileStream) {
        $fileStream.Close()
    }
    $obj = New-Object Object
    $obj | Add-Member Noteproperty FilePath -value $filePath
    $obj | Add-Member Noteproperty IsLocked -value $filelocked
    $obj
}
于 2012-09-15T18:20:59.067 回答
0

您可以在 handle.exe 上找到您的路径。

我使用过 PowerShell,但您可以使用另一个命令行工具。

具有管理权限:

handle.exe -a | Select-String "<INSERT_PATH_PART>" -context 0,100

向下搜索“线程:...”,您应该会看到使用您的路径的进程名称。

于 2020-10-27T22:51:17.590 回答
-1

我喜欢命令提示符 (CMD) 的功能,它也可以在 PowerShell 中使用:

tasklist /m <dllName>

请注意,您不能输入 DLL 文件的完整路径。光是名字就够了。

于 2015-04-01T23:39:07.620 回答
-5

如果你稍微修改一下上面的函数,它将返回 True 或 False(你需要以完整的管理员权限执行),例如用法:

PS> TestFileLock "c:\pagefile.sys"

function TestFileLock {
    ## Attempts to open a file and trap the resulting error if the file is already open/locked
    param ([string]$filePath )
    $filelocked = $false
    $fileInfo = New-Object System.IO.FileInfo $filePath
    trap {
        Set-Variable -name Filelocked -value $true -scope 1
        continue
    }
    $fileStream = $fileInfo.Open( [System.IO.FileMode]::OpenOrCreate, [System.IO.FileAccess]::ReadWrite, [System.IO.FileShare]::None )
    if ($fileStream) {
        $fileStream.Close()
    }
    $filelocked
}
于 2012-09-18T11:40:57.110 回答