c# - 进程无法访问该文件,因为它正在被另一个进程在第二次调用时使用
问题描述
我有一个 Azure 函数 (HttpTrigger),它在我的 Azure 存储中创建一个 tar.gz 文件。
当我触发它一次时,该文件已成功创建。但如果我再次点击它,我会遇到错误:
[信息] 该进程无法访问文件“D:\local\Temp\myDir\tarArchive.tar.gz”,因为它正被另一个进程使用。
我必须重新启动 Azure 门户中的功能才能创建另一个文件。
这是我的代码:
FileStream fs = new FileStream(myDir+"/firstFile", FileMode.Create);
fs.Write(bytesToCompress, 0, bytesToCompress.Length);
fs.Dispose();
FileStream sfs = new FileStream(myDir + "/secondfile", FileMode.Create);
sfs.Dispose();
DirectoryInfo DirectoryOfFilesToBeTarred = new DirectoryInfo(myDir);
FileInfo[] filesInDirectory = DirectoryOfFilesToBeTarred.GetFiles();
string tarArchiveName = myDir + "/tarArchive.tar.gz";
using (Stream targetStream = new GZipOutputStream(File.Create(tarArchiveName)))
{
using (TarArchive tarArchive = TarArchive.CreateOutputTarArchive(targetStream, TarBuffer.DefaultBlockFactor))
{
foreach(FileInfo fileToBeTarred in filesInDirectory)
{
log.Info(fileToBeTarred.FullName);
TarEntry entry = TarEntry.CreateEntryFromFile(fileToBeTarred.FullName);
tarArchive.WriteEntry(entry, true); // Error thrown here
}
}
}
我认为当再次调用该函数时 fileToBeTarred 仍在使用中(我错了吗?),但我试图从这个 FileInfo 创建一个流以便 Dispose() 它,但没有解决我的问题。我也尝试过 Delete() 它,但没有任何效果。
有人看到我没看到的吗?
谢谢你的帮助
更新
这是 Wim Coenen 给出的修正代码
using (Stream fileStream = File.Create(tarArchiveName))
using (Stream targetStream = new GZipOutputStream(fileStream))
using (TarArchive tarArchive = TarArchive.CreateOutputTarArchive(targetStream, TarBuffer.DefaultBlockFactor))
{
foreach (FileInfo fileToBeTarred in filesInDirectory)
{
log.Info(fileToBeTarred.FullName);
TarEntry entry = TarEntry.CreateEntryFromFile(fileToBeTarred.FullName);
tarArchive.WriteEntry(entry, true); // Error thrown here
}
}
以及错误日志(e = Exception object)e.message =>
2018-08-01T11:59:46.887 [信息] 进程无法访问文件“D:\local\Temp\myDir\tarArchive.tar.gz”,因为它正被另一个进程使用。
e.ToString() =>
2018-08-01T11:59:47.152 [Info] System.IO.IOException:进程无法访问文件“D:\local\Temp\myDir\tarArchive.tar.gz”,因为它正被另一个进程使用。
在 System.IO.__Error.WinIOError(Int32 错误代码,字符串可能全路径)
在 System.IO.FileStream.Init(字符串路径、FileMode 模式、FileAccess 访问、Int32 权限、Boolean useRights、FileShare 共享、Int32 bufferSize、FileOptions 选项、SECURITY_ATTRIBUTES secAttrs、String msgPath、Boolean bFromProxy、Boolean useLongPath、Boolean checkHost)
在 System.IO.FileStream..ctor(字符串路径、FileMode 模式、FileAccess 访问、FileShare 共享)
在 ICSharpCode.SharpZipLib.Tar.TarArchive.WriteEntryCore(TarEntry sourceEntry,布尔递归)
在 ICSharpCode.SharpZipLib.Tar.TarArchive.WriteEntry(TarEntry sourceEntry,布尔递归)
在 UploadFileFromBCText.Function1.d__4.MoveNext()
解决方案
看起来您不是在 Azure 存储中创建它,而是在 Function App 实例的本地磁盘上创建它。这可能不是一个好主意,因为实例是短暂的,不会长期保存您的文件。
相反,请查看Azure Blob 存储输出绑定- 其目的是将文件存储在 Azure 存储中,而不使用低级文件 API 或 SDK。
推荐阅读
- html - 如何将光标/文本位置指示器添加到
- html - 传递 AJAX 参数并从应用程序的 expressjs 端访问它
- angular - 使用 Angular 和 httpclient 执行谷歌应用程序脚本
- java - Spring Boot Diriest Context Out of Memory 异常
- validation - NestJS:验证每个地图
使用类验证器的值 - reactjs - keyof 和 React Redux 操作
- c# - C#/Unity - 无法将可编写脚本的对象保存到磁盘
- microsoft-teams - 使用 Power Automate Flow 自动开始 Teams 会议
- node.js - 除非我记录承诺,否则为什么我的异步代码无法正常工作?
- r - 使用 pivot_longer 正确地从宽到长获取数据