6

我在 Windows 8 C# 应用程序中有以下代码,它从服务器获取图像并存储它:

        private async Task httpFetcher()
    {
        HttpClient httpClient = new HttpClient();
        HttpRequestMessage request = new HttpRequestMessage(
HttpMethod.Get, "http://www.example.com/fakeImageRotator.php"); // FOR EXAMPLE
        HttpResponseMessage response = await httpClient.SendAsync(request,
            HttpCompletionOption.ResponseHeadersRead);

        Uri imageUri;
        BitmapImage image = null;

        try
        {
            var imageFile = await ApplicationData.Current.LocalFolder.CreateFileAsync(
         "test.png", CreationCollisionOption.ReplaceExisting);
            var fs = await imageFile.OpenAsync(FileAccessMode.ReadWrite);
            DataWriter writer = new DataWriter(fs.GetOutputStreamAt(0));
            writer.WriteBytes(await response.Content.ReadAsByteArrayAsync());
            await writer.StoreAsync();
            writer.DetachStream();
            await fs.FlushAsync();
            writer.Dispose();

            if (Uri.TryCreate(imageFile.Path, UriKind.RelativeOrAbsolute, out imageUri))
            {
                image = new BitmapImage(imageUri);
            }

        }
        catch (Exception e)
        {
            return;
        }

        image1.Source = image;
    }

看来我在此特定行上随机出现错误:

                var imageFile = await ApplicationData.Current.LocalFolder.CreateFileAsync(
         "test.png", CreationCollisionOption.ReplaceExisting);

它并不总是发生,所以我不确定如何查明问题。所有错误详细信息都在这里:

UnauthorizedAccessException 被捕获

访问被拒绝。(来自 HRESULT 的异常:0x80070005 (E_ACCESSDENIED))在 System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) 在 System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) 在 System.Runtime.CompilerServices.TaskAwaiter`1.GetResult () 在 TestApp.MainPage.d__4.MoveNext() 在 d:\TestApp\TestApp\MainPage.xaml.cs:line 86

4

3 回答 3

6

更新 - “拒绝访问”错误是由多种原因引起的。

第一个原因与下载图像有关。似乎下载代码中的某些内容正在打开文件。我已经简化了下面的下载代码。

第二个原因与BitmapImage保持打开文件的对象有关。有关详细信息,请参阅此帖子:删除以前在 WinRT 中的 DataTemplate 中使用的图像文件时访问被拒绝

解决第二个问题的一种方法是使用 astream而不是 aUri来初始化BitmapImage.

这是一个适合我的版本(您的原始代码也在此处,但已注释掉):

var imageFile = await ApplicationData.Current.LocalFolder.CreateFileAsync(
  "test.png", CreationCollisionOption.ReplaceExisting);
/*
var fs = await imageFile.OpenAsync(FileAccessMode.ReadWrite);
DataWriter writer = new DataWriter(fs.GetOutputStreamAt(0));
writer.WriteBytes(await response.Content.ReadAsByteArrayAsync());
await writer.StoreAsync();
writer.DetachStream();
await fs.FlushAsync();
writer.Dispose();

if (Uri.TryCreate(imageFile.Path, UriKind.RelativeOrAbsolute, out imageUri))
{
    image = new BitmapImage(imageUri);
}
*/
var fs = await imageFile.OpenStreamForWriteAsync();
await response.Content.CopyToAsync(fs);
await fs.FlushAsync();
// you may want to have this Dispose as part of a 
// finally block (try/ catch/ finally)
fs.Dispose();

var bs = await imageFile.OpenAsync(Windows.Storage.FileAccessMode.Read);
image = new BitmapImage();
image.SetSource(bs);
...
image1.Source = image;
于 2013-06-08T22:39:26.900 回答
1

当我将 pdf 文档下载到 LocalFolder 文件并尝试显示它时,我遇到了同样的问题。我无法可靠地说明为什么会发生这种情况,但是这个小技巧帮助我解决了这个问题:

而不是使用

try
{
  StorageFile storage = await ApplicationData.Current.LocalFolder.CreateFileAsync(
    fileName, CreationCollisionOption.ReplaceIfExists);

  //work with file
}
catch (Exception) { ... }

我现在使用这个:

StorageFile storage = null;
try 
{ 
  //First try to get the file
  storage = await ApplicationData.Current.LocalFolder.GetFileAsync(fileName); 
}
catch (Exception) 
{ 
  //Ignore this exception
}

try
{
  //If the storage file is still null, create it
  if (storage == null)
    storage = await ApplicationData.Current.LocalFolder.CreateFileAsync(
      fileName, CreationCollisionOption.OpenIfExists);

  //Work with file
}
catch (Exception)
{
  //Process exception
}
于 2013-07-02T10:10:58.703 回答
1

如果您可以选择使用 CreationCollisionOption。GenerateUniqueName而不是 ReplaceExisting,尝试使用它。

至少它为我解决了问题。

于 2014-10-23T14:49:00.470 回答