用Windows API函数(CreateFile/ReadFile/WriteFile/CloseHandle)完成文件拷贝程序(初级版)

时间:2022-11-24 16:15:39

文件拷贝程序

程序类型:Console

参数:源文件名   目的文件名

要求:1.只能使用Windows API函数(CreateFile/ReadFile/WriteFile/CloseHandle)完成

这个程序是借鉴网上的一位程序员的,根据我自己的需要进行了修改,但是网址我找不到了,现向他致敬。我修改后的代码如下(代码1):

 #include<stdio.h>
 #include<windows.h>
 #include "iostream"
 //#include "DLLShiXian"
 using namespace std;
 void copyfuc(char *Sourcefile,char *Destinationfile)
 {
     HANDLE hSourceFile = CreateFile(Sourcefile, GENERIC_READ, , NULL, OPEN_ALWAYS, , NULL);
     HANDLE hDestFile = CreateFile(Destinationfile, GENERIC_WRITE, , NULL, CREATE_ALWAYS, , NULL);
     if( hSourceFile == INVALID_HANDLE_VALUE || hDestFile == INVALID_HANDLE_VALUE)
     {

         cout<<"无法复制文件!"<<endl;
         return;
     }
     DWORD dwRemainSize=GetFileSize(hSourceFile,NULL);
     ];
     )
     {
         DWORD dwActualRead=;
         ReadFile(hSourceFile,buffer,,&dwActualRead,NULL);
         dwRemainSize-=dwActualRead;
         DWORD dwActualWrote=;
         while(dwActualWrote<dwActualRead)
         {
             DWORD dwOnceWrote = ;
             WriteFile(hDestFile,buffer+dwActualWrote,dwActualRead-dwActualWrote,&dwOnceWrote,NULL);
             dwActualWrote+=dwOnceWrote;
         }
     }

     CloseHandle(hSourceFile);
     CloseHandle(hDestFile);
 }

 void main()
 {

     ];
     cout<<"请输入源文件的路径及文件名"<<endl;
     cin>>Sourcefile;
     ];
     cout<<"请输入目标文件的路径及文件名"<<endl;
     cin>>Destinationfile;
     copyfuc(Sourcefile,Destinationfile);

 }

然后自我感觉Writefile函数中无需考虑lpNumberOfBytesWritten和nNumberOfBytesToWrite是否相等的问题,便把代码简化了一下,复制文件时没有产生错误,如下(代码2):

#include<windows.h>
#include "iostream"
using namespace std;
void copyfuc(char *Sourcefile,char *Destinationfile)//定义一个子程序,参数分别是源文件名和目标文件名
{
    HANDLE hSourceFile=CreateFile(Sourcefile, GENERIC_READ, , NULL, OPEN_ALWAYS, , NULL);//用createfile创建或者打开源文件
    HANDLE hDestFile=CreateFile(Destinationfile, GENERIC_WRITE, , NULL, CREATE_ALWAYS, , NULL);//用createfile创建或者打开目标文件
    if( hSourceFile==INVALID_HANDLE_VALUE||hDestFile==INVALID_HANDLE_VALUE)//打开或者创建文件失败,则返回错误信息
    {

        cout<<"无法复制文件!"<<endl;
    return;
     }
    DWORD dwRemainSize=GetFileSize(hSourceFile,NULL);//获取源文件大小
    ];//设置缓冲区
    )//复制文件,每次最多复制1024个字节
    {
        DWORD dwActualRead=;//实际读取字节数,也就是缓冲区中文件大小
        ReadFile(hSourceFile,buffer,,&dwActualRead,NULL);//读取源文件到缓冲区
        dwRemainSize-=dwActualRead;//剩余文件大小
        DWORD dwActualWrote=;//使用结束后清0,等待存储下一次读取的字节数
        DWORD dwOnceWrote=;//用来保存实际写入字节数
        WriteFile(hDestFile,buffer,dwActualRead,&dwOnceWrote,NULL);//把缓冲区内容写入目标文件
    }
       CloseHandle(hSourceFile);  
       CloseHandle(hDestFile);
}

void main()
{

    ];
    cout<<"请输入源文件的路径及文件名"<<endl;
    cin>>Sourcefile;
    ];
    cout<<"请输入目标文件的路径及文件名"<<endl;
    cin>>Destinationfile;
    copyfuc(Sourcefile,Destinationfile);
    return;
} 

但是我在深入学习他的过程中找到一篇文章http://a547705232.blog.163.com/blog/static/1724915952010102644045562/里面说了这么一句话:

lpNumberOfBytesWritten: 输出参数,指向存储真实写入的数据大小的变量,可能已经到达文件尾等原因,其值可能与nNumberOfBytesToWrite不同。如果lpOverlapped为NULL,则该参数不能为NULL。

这个可能是原作者嵌套一个循环来确保缓冲区中的内容全部写入目标文件的原因吧(24-29行),我又查看了其他的文档和资料,也看了MSDN,没有发现同样的描述,假如以后知道了再贴上来。

  感觉这个程序还是不全面,以后再完善,比如加上一个对话框来选择源文件,一个对话框来选择目标文件,假如目标文件不存在可以新建,若文件很大,添加文件复制进度条等等功能。

关于createfile的具体用法请戳这儿:http://www.cppblog.com/yishanhante/articles/19545.html,http://blog.csdn.net/mafuli007/article/details/7370279

另推荐一个BLOG,里面有很多Windows API编程实例,可以看一下:http://blog.csdn.net/caimouse/article/details/11815095

当然你也可以查看MSDN(英文):http://msdn.microsoft.com/en-us/library/aa365747(VS.85).aspx

http://msdn.microsoft.com/en-us/library/windows/desktop/aa365467(v=vs.85).aspx