- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
一直在阅读有关异步和任务的内容,并一直在尝试通过 PInvoke 将 CopyFileEx 方法转换为有进展的任务模式。我对进度部分有疑问。
CopyFileEx 有一个名为 CopyProgressRoutine 的回调,它有一个名为 lpData 的参数,它接受一个指针。我想我可以用它来传递我的 IProgress 界面,这样我就可以报告进度。然而,事实证明我必须使用一个结构,而不是一个类。有什么想法可以使它正常工作,或者我是否正朝着完全错误的方向前进?
public class ProgressReportAsync
{
public int PercentDone { get; set; }
public string InfoText { get; set; }
public void setProgress(long _progress, long _total)
{
PercentDone = Convert.ToInt32((_progress * 100) / _total);
InfoText = PercentDone + "% complete."; ;
}
}
class FileCopyAsync
{
class UserCallbackArg
{
public CancellationToken ct;
public IProgress<ProgressReportAsync> prg;
public UserCallbackArg(CancellationToken _ct, IProgress<ProgressReportAsync> _prg)
{
ct = _ct;
prg = _prg;
}
public UserCallbackArg() { }
}
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool CopyFileEx(string lpExistingFileName, string lpNewFileName, CopyProgressRoutine lpProgressRoutine, Object lpData, ref bool pbCancel, CopyFileFlags dwCopyFlags);
private delegate CopyProgressResult CopyProgressRoutine(long TotalFileSize, long TotalBytesTransferred,
long StreamSize, long StreamBytesTransferred, uint dwStreamNumber, CopyProgressCallbackReason dwCallbackReason,
IntPtr hSourceFile, IntPtr hDestinationFile, IntPtr lpData);
[Flags]
enum CopyFileFlags : uint
{
COPY_FILE_FAIL_IF_EXISTS = 0x00000001,
COPY_FILE_RESTARTABLE = 0x00000002,
COPY_FILE_OPEN_SOURCE_FOR_WRITE = 0x00000004,
COPY_FILE_ALLOW_DECRYPTED_DESTINATION = 0x00000008,
COPY_FILE_COPY_SYMLINK = 0x00000800,
COPY_FILE_NO_BUFFERING = 0x00001000
}
enum CopyProgressResult : uint
{
PROGRESS_CONTINUE = 0,
PROGRESS_CANCEL = 1,
PROGRESS_STOP = 2,
PROGRESS_QUIET = 3
}
enum CopyProgressCallbackReason : uint
{
CALLBACK_CHUNK_FINISHED = 0x00000000,
CALLBACK_STREAM_SWITCH = 0x00000001
}
private static bool m_bCancel;
private CopyProgressResult CopyProgressHandler(long total, long transferred, long streamSize, long StreamByteTrans, uint dwStreamNumber, CopyProgressCallbackReason reason, IntPtr hSourceFile, IntPtr hDestinationFile, IntPtr lpData)
{
switch (reason)
{
case CopyProgressCallbackReason.CALLBACK_CHUNK_FINISHED:
UserCallbackArg ucarg = (UserCallbackArg)Marshal.PtrToStructure(lpData, typeof(UserCallbackArg));
IProgress<ProgressReportAsync> prg = ucarg.prg;
ProgressReportAsync prgReport = new ProgressReportAsync();
prgReport.setProgress(transferred, total);
prg.Report(prgReport);
if (ucarg.ct.IsCancellationRequested)
{
m_bCancel = true;
}
return m_bCancel ? CopyProgressResult.PROGRESS_CANCEL : CopyProgressResult.PROGRESS_CONTINUE;
default:
return CopyProgressResult.PROGRESS_CONTINUE;
}
}
public FileCopyAsync() { }
public Task DoWorkAsync(string _from, string _to, CancellationToken ct, IProgress<ProgressReportAsync> prg)
{
return TaskEx.Run(() =>
{
bool copyResult;
if (File.Exists(_to))
{
//throw new Exception("File already exists: " + _to);
}
if (!File.Exists(_from))
{
throw new FileNotFoundException(_from);
}
FileInfo fi = new FileInfo(_from);
m_bCancel = false;
UserCallbackArg ucarg = new UserCallbackArg(ct, prg);
GCHandle handle = GCHandle.Alloc(ucarg, GCHandleType.Pinned);
IntPtr ptr = handle.AddrOfPinnedObject();
if (fi.Length > (1024 * 1024 * 100))
{
//Greater than 100mb then no buffer flag added
copyResult = CopyFileEx(_from, _to, new CopyProgressRoutine(CopyProgressHandler), ptr, ref m_bCancel, (CopyFileFlags.COPY_FILE_RESTARTABLE & CopyFileFlags.COPY_FILE_FAIL_IF_EXISTS & CopyFileFlags.COPY_FILE_NO_BUFFERING));
}
else
{
copyResult = CopyFileEx(_from, _to, new CopyProgressRoutine(CopyProgressHandler), ptr, ref m_bCancel, (CopyFileFlags.COPY_FILE_RESTARTABLE & CopyFileFlags.COPY_FILE_FAIL_IF_EXISTS));
}
if (!copyResult)
{
int error = Marshal.GetLastWin32Error();
if (m_bCancel && (error == 1235))
{
return;
}
else
{
Win32Exception ex = new Win32Exception(error);
throw new Win32Exception(error);
}
}
});
}
}
最佳答案
我认为最简单的解决方案是将 CopyProgressHandler 回调移动到您的用户参数类。在这种情况下,您可以使用 ucarg.CopyProgressHandler 作为您的 CopyProgressRoutine 并在您存储在用户参数类中的 IProgress 引用上调用方法。也许您也可以将 m_bCancel 标志移动到该类。
通过这种方法,您可以避免对数据进行编码。
关于c# - 将 CopyFileEx 转换为任务模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9932046/
是否可以从线程调用 CopyFileEx 和 CopyCallback/ProgressRoutine 函数(ProgressBar.Position 将同步)? 我可以在线程中声明 CopyCall
我从一开始就一直在使用 CopyFileEx,它一直运行良好。如果文件未锁定,它将复制该文件。如果文件被锁定,则不会,GetLastError 将报告错误。一切都很好,花花公子。 但是,有些文件被部分
我有一个 DWORD dwCopyFlags,其中包含函数 CopyFileEx 的复制标志。有没有办法解析 dwCopyFlags 以检查它是否包含某种类型的标志,例如 COPY_FILE_FAIL
我正在编写一些需要调用 CopyFileEx 函数的 C++ 代码。与大多数其他 WIN32 函数一样,CopyFileEx 的文档说: 如果函数失败,返回值为零。要获取扩展的错误信息,请调用 Get
我最近不得不从使用 File.Copy() 改为使用 CopyFileEx,我正在努力寻找如何使用它。 经过大量谷歌搜索后,我找到了 this很好用的包装器,但我需要的是获取当前文件的复制字节的进度,
Windows API 函数 CopyFileEx 允许指定一个回调函数,当文件的一部分被复制到例如向用户提供有关进度的反馈。 这在我的程序中运行良好,但我现在有一个额外的要求:我需要暂时暂停复制过程
一直在阅读有关异步和任务的内容,并一直在尝试通过 PInvoke 将 CopyFileEx 方法转换为有进展的任务模式。我对进度部分有疑问。 CopyFileEx 有一个名为 CopyProgress
我正在尝试为来自 here 的 CopyFileEx 使用 FileUtilities.CopyFile 包装器.但是 CopyFileCallbackAction 直到文件被复制后才会被调用(我试过
我使用 CreateDialogParam 创建了一个带有进度条和取消按钮的对话框,以在复制多个文件(使用 CopyFileEx)时显示状态。 如何正确使用 CopyFileEx 取消进程,从按下对话
我正在使用 Microsoft 的 CopyFileEx 方法来复制文件。我正在从一台服务器复制到另一台服务器,文件位于共享文件夹中,我正在使用 UNC。我收到错误 999。我在 Microsoft
我已经在网上和 stackoverflow 上搜索过了。 我想将多个文件从多个来源复制到多个目的地。我还没有理会 UnauthorizedAccessExceptions - 这些文件很可能无论如何都
谁能给我一个在 Qt 中使用带有进度回调的 CopyFileEx 的工作示例? 我发现了一些划痕并试图合并它但没有成功。我什至无法将 CopyProgressRoutine 函数作为 CopyFile
我想从要复制的源文件中调用一个 copyFile 函数。 VBA FileCopy 函数不允许在复制时打开源文件。我正在考虑改用 CopyFile 或 CopyFileEx(显示进度条)。 因此,如果
我做了一个复制文件的应用程序,通常是大文件。我想显示复制的进度,但我不能让它发挥作用。有人可以告诉我我做错了什么吗?现在它的工作非常有趣。 % 比 49%,回到 0,然后回到 40,然后又回到 0,然
我最近和一个 friend 聊天,他告诉我 CopyFile和 CopyFileEx在 C++ Metro 应用程序中是不允许的(这似乎是正确的)。为什么是这样?为什么他们提出了一个奇怪的替代品,名为
我在此处使用 CopyFileEx 的 FileUtilities.CopyFile 包装器 http://msdn.microsoft.com/en-us/magazine/cc163851.asp
Windows Server 2003 有一个限制,它会阻止您复制与您拥有的 RAM 量成比例的超大文件。限制在于 CopyFile 和 CopyFileEx 函数,它们由 xcopy、Explore
我是一名优秀的程序员,十分优秀!