有没有办法检查文件是否正在使用?
我正在用 C# 编写一个程序,需要重复访问 1 个图像文件。 大多数情况下它可以工作,但如果我的计算机运行速度很快,它会在文件保存回文件系统之前尝试访问该文件并抛出错误:
“文件正在被另一个进程使用”
我想找到解决这个问题的方法,但是我所有的谷歌搜索都只能通过使用异常处理来创建检查。 这违背了我的宗教信仰,所以我想知道是否有人有更好的方法?
I'm writing a program in C# that needs to repeatedly access 1 image file. Most of the time it works, but if my computer's running fast, it will try to access the file before it's been saved back to the filesystem and throw an error:
"File in use by another process"
I would like to find a way around this, but all my Googling has only yielded creating checks by using exception handling. This is against my religion, so I was wondering if anyone has a better way of doing it?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(20)
更新了此解决方案的注释:对于只读文件,使用
FileAccess.ReadWrite
检查将会失败,因此该解决方案已修改为使用FileAccess.Read
检查>。原文:
我在过去几年中一直使用此代码,并且没有遇到任何问题。
理解您对使用异常的犹豫,但您无法一直避免它们:
Updated NOTE on this solution: Checking with
FileAccess.ReadWrite
will fail for Read-Only files so the solution has been modified to check withFileAccess.Read
.ORIGINAL:
I've used this code for the past several years, and I haven't had any issues with it.
Understand your hesitation about using exceptions, but you can't avoid them all of the time:
您可能会遇到线程竞争状况,有记录的示例将其用作安全漏洞。 如果您检查该文件是否可用,但随后尝试使用它,您可能会在此时抛出该文件,恶意用户可能会利用该文件在您的代码中强制和利用。
最好的选择是 try catch / finally 尝试获取文件句柄。
You can suffer from a thread race condition on this which there are documented examples of this being used as a security vulnerability. If you check that the file is available, but then try and use it you could throw at that point, which a malicious user could use to force and exploit in your code.
Your best bet is a try catch / finally which tries to get the file handle.
使用它来检查文件是否被锁定:
出于性能原因,我建议您在同一操作中读取文件内容。 以下是一些示例:
您自己尝试一下:
Use this to check if a file is locked:
For performance reasons I recommend you read the file content in the same operation. Here are some examples:
Try it out yourself:
我最近遇到了这个问题并发现了这个: https:// /learn.microsoft.com/en-us/dotnet/standard/io/handling-io-errors。
在此,Microsoft 描述了以下方法来检查 IOException 是否由锁定的文件引起:
I recently came across this issue and found this: https://learn.microsoft.com/en-us/dotnet/standard/io/handling-io-errors.
Here, Microsoft describes the following method for checking if an
IOException
was due to a locked file:只需按预期使用异常即可。 接受该文件正在使用并重试,重复直到您的操作完成。 这也是最有效的,因为您不会浪费任何周期在执行操作之前检查状态。
使用下面的函数,例如
2 秒后超时的可重用方法
Just use the exception as intended. Accept that the file is in use and try again, repeatedly until your action is completed. This is also the most efficient because you do not waste any cycles checking the state before acting.
Use the function below, for example
Reusable method that times out after 2 seconds
您可以返回一个任务,该任务在流可用时立即为您提供该流。 这是一个简化的解决方案,但它是一个很好的起点。 它是线程安全的。
您可以照常使用此流:
You can return a task which gives you a stream as soon as it becomes available. It's a simplified solution, but it is a good starting point. It's thread safe.
You can use this stream as usual:
上面接受的答案存在一个问题,如果文件已打开并使用 FileShare.Read 模式进行写入,或者文件具有只读属性,则代码将无法工作。 此修改后的解决方案工作最可靠,需要记住两件事(对于已接受的解决方案也是如此):
记住上述内容,这会检查文件是否锁定用于写入或锁定以防止读取:
The accepted answers above suffer an issue where if file has been opened for writing with a FileShare.Read mode or if the file has a Read-Only attribute the code will not work. This modified solution works most reliably, with two things to keep in mind (as true for the accepted solution also):
Keeping the above in mind, this checks if the file is either locked for writing or locked to prevent reading:
也许您可以使用 FileSystemWatcher 并监视更改事件。
我自己没有使用过这个,但可能值得一试。 如果文件系统观察程序对于这种情况来说有点重,我会选择 try/catch/sleep 循环。
Perhaps you could use a FileSystemWatcher and watch for the Changed event.
I haven't used this myself, but it might be worth a shot. If the filesystemwatcher turns out to be a bit heavy for this case, I would go for the try/catch/sleep loop.
据我所知,这里有一些代码与接受的答案具有相同的功能,但代码较少:
但是我认为按以下方式执行此操作会更稳健:
Here is some code that as far as I can best tell does the same thing as the accepted answer but with less code:
However I think it is more robust to do it in the following manner:
除了工作 3 行代码之外,仅供参考:如果您想要完整信息 - Microsoft 开发中心上有一个小项目:
https://code.msdn.microsoft.com/windowsapps/How-to-know-the-process-704839f4< /a>
现在发现于:
https://github.com/TacticalHorse/LockFinder/blob/master/LockFinder。 cs
来自简介:
它通过连接到“重新启动管理器会话”来工作。
对于您的特定需求,它可能有点过度设计......
但如果这就是您想要的,请继续获取 vs-project。
Aside from working 3-liners and just for reference: If you want the full blown information - there is a little project on Microsoft Dev Center:
https://code.msdn.microsoft.com/windowsapps/How-to-know-the-process-704839f4
Now found at:
https://github.com/TacticalHorse/LockFinder/blob/master/LockFinder.cs
From the Introduction:
It works by connecting to the "Restart Manager Session".
It might be a little overengineered for your particular needs...
But if that is what you want, go ahead and grab the vs-project.
希望这可以帮助!
Hope this helps!
我知道的唯一方法是使用 Win32 独占锁 API,它速度不太快,但有示例。
大多数人,为了解决这个问题,只需尝试/捕获/睡眠循环即可。
the only way I know of is to use the Win32 exclusive lock API which isn't too speedy, but examples exist.
Most people, for a simple solution to this, simply to try/catch/sleep loops.
根据我的经验,您通常想要执行此操作,然后“保护”您的文件以执行一些奇特的操作,然后使用“受保护”的文件。 如果您只想像这样使用一个文件,则可以使用 Jeremy Thompson 的答案中解释的技巧。 但是,如果您尝试对大量文件执行此操作(例如,当您编写安装程序时),您就会受到很大的伤害。
解决此问题的一种非常优雅的方法是利用以下事实:如果正在使用某个文件,您的文件系统将不允许您更改文件夹名称。 将文件夹保留在同一个文件系统中,它会像魅力一样工作。
请注意,您应该了解可以利用此漏洞的明显方式。 毕竟,文件不会被锁定。 另请注意,还有其他原因可能导致您的
移动
操作失败。 显然,正确的错误处理(MSDN)可以在这里提供帮助。对于单个文件,我会坚持 Jeremy Thompson 发布的锁定建议。
In my experience, you usually want to do this, then 'protect' your files to do something fancy and then use the 'protected' files. If you have just one file you want to use like this, you can use the trick that's explained in the answer by Jeremy Thompson. However, if you attempt to do this on lots of files (say, for example when you're writing an installer), you're in for quite a bit of hurt.
A very elegant way this can be solved is by using the fact that your file system will not allow you to change a folder name if one of the files there it's being used. Keep the folder in the same file system and it'll work like a charm.
Do note that you should be aware of the obvious ways this can be exploited. After all, the files won't be locked. Also, be aware that there are other reasons that can result in your
Move
operation to fail. Obviously proper error handling (MSDN) can help out here.For individual files I'd stick with the locking suggestion posted by Jeremy Thompson.
我曾经需要将 PDF 上传到在线备份存档。 但如果用户在其他程序(例如 PDF 阅读器)中打开文件,备份将会失败。 在匆忙中,我尝试了该线程中的一些最佳答案,但无法使它们发挥作用。 对我有用的是尝试将 PDF 文件移动到它自己的目录。 我发现如果文件在另一个程序中打开,这将会失败,并且如果移动成功,则不需要恢复操作,因为如果将文件移动到单独的目录,则需要恢复操作。 我想发布我的基本解决方案,以防它对其他人的特定用例有用。
I once needed to upload PDFs to an online backup archive. But the backup would fail if the user had the file open in another program (such as PDF reader). In my haste, I attempted a few of the top answers in this thread but could not get them to work. What did work for me was trying to move the PDF file to its own directory. I found that this would fail if the file was open in another program, and if the move were successful there would be no restore-operation required as there would be if it were moved to a separate directory. I want to post my basic solution in case it may be useful for others' specific use cases.
您可以使用我的库从多个应用程序访问文件。
您可以从 nuget 安装它:Install-Package Xabe.FileLock
如果您想了解有关它的更多信息,请检查
https://github.com/tomaszzmuda/Xabe.FileLock
fileLock.Acquire 方法将返回 true仅当可以锁定该对象独占的文件时。
但上传文件的应用程序也必须在文件锁定下进行。
如果对象不可访问,则方法返回 false。
You can use my library for accessing files from multiple apps.
You can install it from nuget: Install-Package Xabe.FileLock
If you want more information about it check
https://github.com/tomaszzmuda/Xabe.FileLock
fileLock.Acquire method will return true only if can lock file exclusive for this object.
But app which uploading file must do it in file lock too.
If object is inaccessible metod returns false.
尝试将文件移动/复制到临时目录。 如果可以的话,它没有锁,您可以安全地在临时目录中工作而不会获得锁。 否则只需尝试在 x 秒内再次移动它。
Try and move/copy the file to a temp dir. If you can, it has no lock and you can safely work in the temp dir without getting locks. Else just try to move it again in x seconds.
这样的事情会有帮助吗?
Would something like this help?
我使用此解决方法,但在使用 IsFileLocked 函数检查文件锁定和打开文件之间有一个时间跨度。 在这段时间内,其他线程可以打开该文件,因此我将收到 IOException。
所以,我为此添加了额外的代码。 就我而言,我想加载 XDocument:
你觉得怎么样? 我可以改变一些事情吗? 也许我根本不必使用 IsFileBeingUsed 函数?
谢谢
I use this workaround, but i have a timespan between when i check the file locking with IsFileLocked function and when i open the file. In this timespan some other thread can open the file, so i will get IOException.
So, i added extra code for this. In my case i want load XDocument:
What do you think? Can i change some thing? Maybe i did not have to use IsFileBeingUsed function at all?
Thanks
我很想知道这是否会引发任何 WTF 反应。 我有一个从控制台应用程序创建并随后启动 PDF 文档的流程。 然而,我正在处理一个弱点,如果用户多次运行该进程,生成相同的文件而不首先关闭先前生成的文件,应用程序将抛出异常并死亡。 这种情况相当频繁,因为文件名基于销售报价编号。
我决定依靠自动递增的文件版本控制,而不是以这种不优雅的方式失败:
也许可以更加小心
catch
块,以确保我捕获正确的 IOException(s) 。 我可能还会在启动时清除应用程序存储,因为这些文件无论如何都是临时的。我意识到这超出了操作员简单检查文件是否正在使用的问题的范围,但这确实是我到达这里时想要解决的问题,所以也许对其他人有用。
I'm interested to see if this triggers any WTF reflexes. I have a process which creates and subsequently launches a PDF document from a console app. However, I was dealing with a frailty where if the user were to run the process multiple times, generating the same file without first closing the previously generated file, the app would throw an exception and die. This was a rather frequent occurrence because file names are based on sales quote numbers.
Rather than failing in such an ungraceful manner, I decided to rely on auto-incremented file versioning:
Probably some more care can be given to the
catch
block to ensure I'm catching the correct IOException(s). I'll probably also clear out the app storage on startup since these files are intended to be temporary anyways.I realize this goes beyond the scope of the OP's question of simply checking if the file is in use but this was indeed the problem I was looking to solve when I arrived here so perhaps it will be useful to someone else.