在 WPF 切换光标的时候,如果是通过本地资源的方法传入 stream 的,需要先复制到临时文件夹里面的文件,然后读取文件指针释放文件。如果此时的 temp 文件夹满了,那么复制文件的时候就无法继续了,于是就无法创建完成光标
最近有老师找我说软件无法使用了,我尝试调试他的电脑,发现任何修改光标的代码就无法继续,因为无法创建光标
大概的修改光标的代码是这样写的,从解决方案里面放一个光标文件,设置为资源通过访问解决方案文件 拿到资源
var uri = new Uri("pack://application:,,,/Text.cur");
var resource = Application.GetResourceStream(uri);
Cursor = new Cursor(resource.Stream);
看到的堆栈如下
在 System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath)
在 System.IO.Directory.InternalCreateDirectory(String fullPath, String path, Object dirSecurityObj, Boolean checkHost)
在 System.IO.Directory.InternalCreateDirectoryHelper(String path, Boolean checkHost)
在 System.IO.Directory.CreateDirectory(String path)
在 System.IO.FileHelper.CreateAndOpenTemporaryFile(String& filePath, FileAccess fileAccess, FileOptions fileOptions, String extension, String subFolder)
在 System.Windows.Input.Cursor.LoadFromStream(Stream cursorStream)
在 System.Windows.Input.Cursor..ctor(Stream cursorStream, Boolean scaleWithDpi)
在 System.Windows.Input.Cursor..ctor(Stream cursorStream)
在 FawlalnejajerelaWhallgemcurkear.MainWindow..ctor() 位置 D:\lindexi\程序\FawlalnejajerelaWhallgemcurkear\FawlalnejajerelaWhallgemcurkear\MainWindow.xaml.cs:行号 32
通过读源代码,发现在 LoadFromStream 方法里面是这样写的
private void LoadFromStream(Stream cursorStream)
{
string filePath = null;
try
{
// Generate a temporary file based on the memory stream.
// 从 temp 文件夹创建一个文件
using (FileStream fileStream = FileHelper.CreateAndOpenTemporaryFile(out filePath))
{
// 复制到文件
cursorStream.CopyTo(fileStream);
}
// 从文件里面读取光标
// create a cursor from the temp file
_cursorHandle = UnsafeNativeMethods.LoadImageCursor(IntPtr.Zero, filePath,
NativeMethods.IMAGE_CURSOR,
0, 0,
NativeMethods.LR_DEFAULTCOLOR |
NativeMethods.LR_LOADFROMFILE |
(_scaleWithDpi? NativeMethods.LR_DEFAULTSIZE : 0x0000));
if (_cursorHandle == null || _cursorHandle.IsInvalid)
{
throw new ArgumentException(SR.Get(SRID.Cursor_InvalidStream));
}
}
finally
{
// 尝试删除这个文件,因为光标已经读取了
FileHelper.DeleteTemporaryFile(filePath);
}
}
在 FileHelper.CreateAndOpenTemporaryFile 将会读取到一个 temp 文件夹里面的文件,但是如果这个文件无法访问,那么将不能继续
在我的设备上是很难做到让 temp 文件夹无法访问的,但是可以通过通过修改环境变量修改当前进程使用的系统 Temp 文件夹的路径设置一个无法访问的文件夹作为 temp 文件夹
做一个无法访问的文件夹很简单,只需要右击属性安装,去掉用户就可以了
运行代码就会发现提示对路径访问拒绝
System.UnauthorizedAccessException:“对路径“D:\lindexi\无法访问文件夹\WPF”的访问被拒绝。”
可以的解决方法有两个
- 通过环境变量修改 temp 文件夹作为程序自己内部的数据文件夹,这和 UWP 的相同,每个程序都可以有自己独立的 temp 文件夹,可以解决有一些逗比软件会更改整个 temp 文件夹或里面某些文件夹的访问权限或有逗比在 temp 文件夹写入了 65535 个文件让其他程序无法写入文件。从微软官方文档 可以知道 temp 文件夹的文件限制。
- 只对光标的修改将解决方案里面的文件修改为输出的文件,此时将会调用 LoadFromFile 方法,这个方法是读取文件不需要复制文件,相对性能比较快
上面提供的两个方法,第一个方法除了解决光标的问题,还可以解决其他问题。第二个方法可以提升一点性能,同时两个方法可以一起使用
这个问题提交给微软,欢迎小伙伴点击 Full temporary folder will crash cursor initialization 帮我点赞
通过修改环境变量修改当前进程使用的系统 Temp 文件夹的路径 - walterlv
本文会经常更新,请阅读原文: https://dotnet-campus.github.io//post/WPF-%E5%85%89%E6%A0%87%E5%88%9D%E5%A7%8B%E5%8C%96%E7%9A%84%E6%97%B6%E5%80%99-temp-%E6%96%87%E4%BB%B6%E5%A4%B9%E6%BB%A1%E4%BA%86%E6%97%A0%E6%B3%95%E5%88%9B%E5%BB%BA.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 lindexi (包含链接: https://dotnet-campus.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。