- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我知道我可以使用 Win32 API 来访问我自己的本地数据文件夹中的文件(例如,参见 this answered question),但我需要访问我的应用程序之外的文件(例如,来自图片库)和我正在尝试的库使用都是基于Win32文件HANDLE
s 和/或它们依赖于使用相对文件名。
因为获取图片库中的文件(或获取从选择器返回的文件/文件夹)的唯一方法是通过StorageFile
。对象,我怎样才能重用我现有的代码?我是否必须将其全部重写为异步并依赖 WinRT 存储 API?
最佳答案
从“周年更新”(又名“RS1”或构建 10.0.14393)开始,您可以获得 Win32 HANDLE
来自 StorageItem
(文件或文件夹)并从 HANDLE
中创建新的命名文件(返回 StorageFolder
) .您可以使用新的 IStorageFolderHandleAccess
和 IStorageItemHandleAccess
蜜蜂。
Note: These APIs have been accidentally placed inside the
WINAPI_PARTITION_DESKTOP
partition (they're not desktop-specific; they're available to UWPs). This will be addressed in future SDK updates.
StorageFile
或
StorageFolder
为界面。如果不支持该接口(interface),则意味着您的应用程序在低级操作系统上运行(或者存储项实际上并未由真实文件支持,而是伪文件)。您可以从 C++(C++/CX 或 WRL)或 C# 使用这些接口(interface)。
FolderPicker
的简单示例让用户在他们的磁盘上选择一个位置(返回一个代理
StorageFolder
对象),然后使用 Win32 API
ReadFile
和
WriteFile
从该位置读取和写入文件。
StorageHandleAccess.h
,复制 SDK 文件
WindowsStorageCOM.h
中的声明:
#pragma once
// These are copied from WindowsStorageCOM.h
// You can remove this header file once the real file has been updated
// to fix the WINAPI_PARTITION_DESKTOP block
typedef interface IOplockBreakingHandler IOplockBreakingHandler;
typedef interface IStorageItemHandleAccess IStorageItemHandleAccess;
typedef interface IStorageFolderHandleAccess IStorageFolderHandleAccess;
#ifdef __cplusplus
extern "C" {
#endif
typedef /* [v1_enum] */
enum HANDLE_OPTIONS
{
HO_NONE = 0,
HO_OPEN_REQUIRING_OPLOCK = 0x40000,
HO_DELETE_ON_CLOSE = 0x4000000,
HO_SEQUENTIAL_SCAN = 0x8000000,
HO_RANDOM_ACCESS = 0x10000000,
HO_NO_BUFFERING = 0x20000000,
HO_OVERLAPPED = 0x40000000,
HO_WRITE_THROUGH = 0x80000000
} HANDLE_OPTIONS;
DEFINE_ENUM_FLAG_OPERATORS(HANDLE_OPTIONS);
typedef /* [v1_enum] */
enum HANDLE_ACCESS_OPTIONS
{
HAO_NONE = 0,
HAO_READ_ATTRIBUTES = 0x80,
HAO_READ = 0x120089,
HAO_WRITE = 0x120116,
HAO_DELETE = 0x10000
} HANDLE_ACCESS_OPTIONS;
DEFINE_ENUM_FLAG_OPERATORS(HANDLE_ACCESS_OPTIONS);
typedef /* [v1_enum] */
enum HANDLE_SHARING_OPTIONS
{
HSO_SHARE_NONE = 0,
HSO_SHARE_READ = 0x1,
HSO_SHARE_WRITE = 0x2,
HSO_SHARE_DELETE = 0x4
} HANDLE_SHARING_OPTIONS;
DEFINE_ENUM_FLAG_OPERATORS(HANDLE_SHARING_OPTIONS);
typedef /* [v1_enum] */
enum HANDLE_CREATION_OPTIONS
{
HCO_CREATE_NEW = 0x1,
HCO_CREATE_ALWAYS = 0x2,
HCO_OPEN_EXISTING = 0x3,
HCO_OPEN_ALWAYS = 0x4,
HCO_TRUNCATE_EXISTING = 0x5
} HANDLE_CREATION_OPTIONS;
EXTERN_C const IID IID_IOplockBreakingHandler;
MIDL_INTERFACE("826ABE3D-3ACD-47D3-84F2-88AAEDCF6304")
IOplockBreakingHandler : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE OplockBreaking(void) = 0;
};
EXTERN_C const IID IID_IStorageItemHandleAccess;
MIDL_INTERFACE("5CA296B2-2C25-4D22-B785-B885C8201E6A")
IStorageItemHandleAccess : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE Create(
/* [in] */ HANDLE_ACCESS_OPTIONS accessOptions,
/* [in] */ HANDLE_SHARING_OPTIONS sharingOptions,
/* [in] */ HANDLE_OPTIONS options,
/* [optional][in] */ __RPC__in_opt IOplockBreakingHandler *oplockBreakingHandler,
/* [system_handle][retval][out] */ __RPC__deref_out_opt HANDLE *interopHandle) = 0;
};
EXTERN_C const IID IID_IStorageFolderHandleAccess;
MIDL_INTERFACE("DF19938F-5462-48A0-BE65-D2A3271A08D6")
IStorageFolderHandleAccess : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE Create(
/* [string][in] */ __RPC__in_string LPCWSTR fileName,
/* [in] */ HANDLE_CREATION_OPTIONS creationOptions,
/* [in] */ HANDLE_ACCESS_OPTIONS accessOptions,
/* [in] */ HANDLE_SHARING_OPTIONS sharingOptions,
/* [in] */ HANDLE_OPTIONS options,
/* [optional][in] */ __RPC__in_opt IOplockBreakingHandler *oplockBreakingHandler,
/* [system_handle][retval][out] */ __RPC__deref_out_opt HANDLE *interopHandle) = 0;
};
#ifdef __cplusplus
}
#endif
StorageFolder
,文件名和创建标志(打开或创建)并尝试打开(或创建)命名文件,从(到)文件读取(或写入)一些文本,并将一些输出写入调试控制台。
MainPage.xaml.cpp
文件(您应该只需要更新命名空间):
#include "pch.h"
#include "MainPage.xaml.h"
#include <ppltasks.h>
// TODO: Replace with your namespace
#error Replace this with your real namespace
using namespace FileHandleFromStorageFolder;
// Uncomment out this line and delete the next line once the SDK is fixed
//#include <WindowsStorageCOM.h>
#include "StorageHandleAccess.h"
// For ComPtr<>
#include <wrl\client.h>
// For HandleT<>
#include <wrl\wrappers\corewrappers.h>
__declspec(noreturn) inline void ThrowWithHRESULT(HRESULT hr, const wchar_t* message)
{
using namespace Platform;
throw ref new Exception(hr, ref new String(message));
}
__declspec(noreturn) inline void ThrowWithGetLastError(const wchar_t* message)
{
using namespace Platform;
throw ref new Exception(HRESULT_FROM_WIN32(GetLastError()), ref new String(message));
}
// Test is a simple test function. Pass in one of the HANDLE_CREATION_OPTIONS values
// (eg, HCO_CREATE_ALWAYS or HCO_OPEN_ALWAYS) and the function will try and either
// write to the file (if it's empty) or read from it (if it's not).
void Test(Windows::Storage::StorageFolder^ folder, const wchar_t* filename, HANDLE_CREATION_OPTIONS options)
{
using namespace Microsoft::WRL;
using namespace Microsoft::WRL::Wrappers;
// Get an IUnknown from the ref class, and then QI for IStorageFolderHandleAccess
ComPtr<IUnknown> abiPointer(reinterpret_cast<IUnknown*>(folder));
ComPtr<IStorageFolderHandleAccess> handleAccess;
HRESULT hr = abiPointer.As(&handleAccess);
if (FAILED(hr))
ThrowWithHRESULT(hr, L"Can't QI");
// Standard RAII wrapper for HANDLEs that represent files
HandleT<HandleTraits::FileHandleTraits>win32fileHandle;
// This is roughly equivalent of calling CreateFile2
hr = handleAccess->Create(filename, options,
HANDLE_ACCESS_OPTIONS::HAO_WRITE | HANDLE_ACCESS_OPTIONS::HAO_READ,
HANDLE_SHARING_OPTIONS::HSO_SHARE_NONE, HANDLE_OPTIONS::HO_NONE, nullptr,
win32fileHandle.GetAddressOf());
if (FAILED(hr))
ThrowWithHRESULT(hr, L"Can't access file");
// From here, it's standard Win32 code - nothing WinRT specific at all
LARGE_INTEGER size{ 0 };
if (FALSE == GetFileSizeEx(win32fileHandle.Get(), &size))
ThrowWithGetLastError(L"Can't get file size");
static const DWORD BUFFER_SIZE = 500;
char buffer[BUFFER_SIZE];
DWORD bytesUsed{ 0 };
if (size.QuadPart == 0)
{
const static auto str = "Hello, world\r\n";
if (FALSE == WriteFile(win32fileHandle.Get(), str, strlen(str), &bytesUsed, nullptr))
ThrowWithGetLastError(L"Can't write to file");
sprintf_s(buffer, ARRAYSIZE(buffer), "Wrote %d bytes to file\r\n", bytesUsed);
OutputDebugStringA(buffer);
}
else
{
if (FALSE == ReadFile(win32fileHandle.Get(), buffer, ARRAYSIZE(buffer) - 1, &bytesUsed, nullptr))
ThrowWithGetLastError(L"Can't read from file");
buffer[bytesUsed] = 0;
OutputDebugStringA(buffer);
}
}
// Trivial driver that gets a StorageFolder and then creates a file
// inside it, writes some text, then re-opens it to read text.
void TestWrapper()
{
using namespace Windows::Storage;
using namespace Windows::Storage::Pickers;
auto picker = ref new FolderPicker();
picker->FileTypeFilter->Append(L".txt");
picker->SuggestedStartLocation = PickerLocationId::Desktop;
concurrency::create_task(picker->PickSingleFolderAsync()).then([]
(StorageFolder^ folder)
{
if (folder != nullptr)
{
// Create and then read back a simple file
Test(folder, L"win32handletest.txt", HANDLE_CREATION_OPTIONS::HCO_CREATE_ALWAYS);
Test(folder, L"win32handletest.txt", HANDLE_CREATION_OPTIONS::HCO_OPEN_ALWAYS);
}
}
);
}
MainPage::MainPage()
{
InitializeComponent();
TestWrapper();
}
关于winapi - 如何在 UWP 中获取 StorageFile 或 StorageFolder 的 Win32 HANDLE?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42799235/
我正在对一个新的 Win8 商店应用程序进行单元测试,并注意到我想避免的竞争条件。所以我正在寻找一种方法来避免这种竞争情况。 我有一个类,当实例化时调用一个方法以确保它有一个本地 StorageFol
我正在使用 Windows Phone 8.1(RT) 应用程序,我想知道如何获取StorageFolder 中的文件数。 我知道我们可以使用 StorageFolder.GetFilesAsync(
我对 MSDN 和 SO 进行了大量研究,但似乎有很多关于这个主题的评论褒贬不一,没有直接的答案。我的 UWP 应用需要为用户下载一些项目。将其放入“下载”文件夹而不是文档或图片似乎是合乎逻辑的。 我
如果我有一个 Metro StorageFolder,我可以使用 获取文件异步(文件名)。 但是,我怎么知道文件是否首先存在?现在,我发现了一个异常,但这似乎不是正确的做法。 最佳答案 根据微软的说法
我正在尝试使数据库与 Windows 8 音乐库保持同步,但我尚未找到有效的解决方案。我知道 .NET 有一个 Windows 8 应用程序无法使用的 FileSystemWatcher。目前,我的想
在常规的 .NET Framework 中,有许多方法可以根据搜索模式检索目录中的文件子集,例如 DirectoryInfo.GetFiles(string searchPattern) . Wind
如何从用户友好的(本地化的)路径获取 StorageFolder? 文件夹可以有一个用户友好的(本地化的)名称。可以通过以下方式读取名称: StorageFolder.DisplayName 示例:文
StorageFolder.GetFilesAsync非常慢: 约 3500 个文件的文件夹需要 7 秒 回到 Windows Phone 8.0 Silverlight,我能够更快地获取 Camer
尝试在 InstalledLocation StorageFolder 中创建文件异步时访问被拒绝 StorageFolder storageFolder = Windows.ApplicationM
当在不同线程上多次调用 Windows.Storage.StorageFolder.GetFolderFromPathAsync 时,我收到 ArgumentExceptions。这是一个重现该问题的
我正在使用 DownloadsFolder 方法创建文件和文件夹。 我想获取父文件夹作为 StorageFolder 实例,这样我就可以列出和操作应用下载文件夹中的所有项目。 我已经从已知的 Stor
我有一个应用程序,我必须在其中存储和使用文件夹和文件的绝对 路径。我有一个非常简单的问题。当我像这样存储文件夹的路径:“ms-appdata:///local/my_folder”并尝试从此路径获取
这个问题在这里已经有了答案: 8 年前关闭。 Possible Duplicate: Check if a file exists in the project in WinRT 我正在使用 Stor
我想获取文件夹及其子文件夹中的所有文件。但是像这样的平面查询: var allFiles = await myFolder.GetFilesAsync(Windows.Storage.Search.C
FolderInformation 的方法/属性与 StorageFolder 的几乎相同。 我不明白为什么 WinRT api 也提供了 FolderInformation 和 StorageFol
我尝试从网络位置读取文件。但我不断收到 UnAuthorizedAccessException。 我通过 StorageFolder.GetFolderFromPathAsync 选择 Storage
我的 Win RT 应用程序在 Windows 8 beta 上与 VS2012RC 一起使用,现在在 Visual Studio 和 Windows 8 pro 的最终版本中出现了问题,即在 OnS
我知道我可以使用 Win32 API 来访问我自己的本地数据文件夹中的文件(例如,参见 this answered question),但我需要访问我的应用程序之外的文件(例如,来自图片库)和我正在尝
在 WP8.1 上有没有办法知道 StorageFolder-obj 属于 SD 还是 Phone?谢谢你。 最佳答案 例如,您可以通过检查 Path 来做到这一点StorageFolder(或 St
我是一名优秀的程序员,十分优秀!