socket简单实现ftp的文件传送(C++V2.0版,解决数据丢失问题)

我的百度博文:http://hi.baidu.com/lauo1988/blog/item/4060bc383dab37fbb311c74b.html

///

///以下代码已经解决了fpt文件传送过程中,数据丢失的问题(特别是大文件),为V2.0版源码

客户端程序

#include<iostream>
#include<winsock2.h>
#include<windows.h>
using namespace std;

static USHORT ser_port=10000;//服务器端口
static char ser_addr_str[256];//进行链接的服务器IP
static u_short host_port=6000;//本地服务器端口

static char buf[2048];
static char fname[256];

DWORD WINAPI Clienter(LPVOID param)
{
int err;
WSADATA wsaData;
err=WSAStartup(MAKEWORD(1,1),&wsaData);//开启网络服务
if(err!=0)//启动失败
{
   cerr<<"客户端启动失败!"<<endl;
   exit(0);
}

/*
   循环取得不同文件
*/
get_file:
SOCKET client=socket(AF_INET,SOCK_STREAM,0);//建立SOCKET
/*
   设定服务器的相关信息
*/
SOCKADDR_IN ser_addr;//服务器信息结构体
ser_addr.sin_addr.S_un.S_addr=inet_addr(ser_addr_str);//192.168.179.1   要访问的服务器地址
ser_addr.sin_family=AF_INET;
ser_addr.sin_port=htons(ser_port);//目标主机端口
/*
   与服务器链接
*/
err=connect(client,(SOCKADDR*)& ser_addr,sizeof(SOCKADDR));
if(err==-1)
{
   cout<<"服务器链接失败!"<<endl;
   goto end;
}
cout<<"请你输入请求的文件名:"<<endl;
memset(buf,0,sizeof(buf));
cin>>buf;
memcpy(fname,buf,256);
/*
   发送的是文件名
*/
loop:
send(client,buf,sizeof(buf),0);
recv(client,buf,sizeof(buf),0);
if(buf[0]==-2)
   goto loop;
if(buf[0]==-1)
{
   cerr<<"ftp server do not have the file:"<<fname<<endl;
   goto end;
}
else
{
   long long flen;
   int i=0,nlen;
   memcpy(&flen,buf,sizeof(long long));
   memcpy(&i,buf+sizeof(long long),4);
   memcpy(&nlen,buf+sizeof(int)+sizeof(long long),4);

   FILE *file=fopen(fname,"wb");
   if(file==NULL)
   {
    cerr<<fname<<" create error!"<<endl;
    goto end;
   }
   for(;i*1024+nlen!=flen;)
   {
    /*
     将讲到的内容写到文件中;
    */
    fwrite(buf+8+sizeof(long long),1,nlen,file);
    memset(buf,0,sizeof(buf));

    /*
     接收成功标识,以便继续传送
    */
    send(client,buf,64,0);

    /*
     再次接收
    */
    int flag=recv(client,buf,sizeof(buf),0);
    memcpy(&i,buf+sizeof(long long),4);
    memcpy(&nlen,buf+4+sizeof(long long),4);

    if(nlen==0)
    {
     break;
    }
   }
   fwrite(buf+sizeof(int)*2+sizeof(long long),1,nlen,file);
   fclose(file);
}
end:
closesocket(client);

/*
   命令行提示信息
*/
char cmd;
cout<<"是否继续(Y/N)?"<<endl;
getchar();//上次输入的回车
cmd=getchar();
if(!(cmd=='n'||cmd=='N'))
   goto get_file;

WSACleanup();
exit(0);
return 1;
}
int main()
{
cout<<"请你输入对方的主机IP:";
cin>>ser_addr_str;
cout<<"对方主机服务端端口:";
char buf[256];
cin>>buf;
ser_port=atoi(buf);
if(ser_port<=1024)//如果输入端口小于1024表明维持使用6000端口,当对话的另一方不使用同一主机时
   ser_port=10000;
cout<<"本机服务端端口:";
cin>>buf;
host_port=atoi(buf);
if(host_port<=1024)//同上
   host_port=6000;
CreateThread(0,0,Clienter,0,0,0);
while(1)
   Sleep(10000);
return 1;
}

///服务端程序

#include<iostream>
#include<WinSock2.h>
#include<windows.h>
using namespace std;
#pragma comment (lib,"ws2_32.lib")

USHORT host_port=10000;
SOCKET client;
SOCKET server;
sockaddr_in client_addr;
int addr_len;
char buf[2048];
char fname[256];
DWORD WINAPI ftp_server(LPVOID argv)//
{
start:
addr_len=sizeof(sockaddr);
client = accept(server,(sockaddr*)&client_addr,&addr_len);
cout<<inet_ntoa(client_addr.sin_addr)<<endl;

int b_len=0;
while(TRUE)
{
   memset(buf,0,sizeof(buf));
   int flag=
   recv(client,buf,sizeof(buf),0);
   if(strlen(buf)!=0)
   {
    cout<<"file_name:"<<buf<<endl;
    break;
   }
   cout<<strerror(flag)<<endl;
   cout<<buf<<endl;

   memset(buf,0,sizeof(buf));
   buf[0]=-2;
   send(client,buf,1,0);//重复请求
}
memcpy(fname,buf,strlen(buf)>256?255:strlen(buf));
fname[255]=0;
FILE* file=fopen(fname,"rb");
if(file==NULL)
{
   cerr<<"open error!"<<endl;
   memset(buf,0,sizeof(buf));
   buf[0]=-1;
   send(client,buf,1,0);
   goto end;
}
else
{
   fseek(file,0,2);
   long long flen=ftell(file);//得到文件最大长度;
   int begin,nlen;
   fseek(file,0,0);//回到文件头部

   if(flen==0)
   {
    cerr<<"文件长度为零!"<<endl;
    fclose(file);
    goto end;
   }
   /*
    对文件进行分块传送
   */
  
   for(int i=0;;)
   {
    memset(buf,0,sizeof(buf));
    memcpy(buf,&flen,sizeof(long long));
    memcpy(buf+sizeof(long long),&i,4);
    begin=ftell(file);
    fread(buf+sizeof(long long)+8,1,1024,file);

    nlen=ftell(file)-begin;/*文件读取的内容长度*/
    if(nlen==0)/*文件内容读取完全*/
    {
     break;
    } 
    memcpy(buf+sizeof(long long)+4,&nlen,4);
    send(client,buf,sizeof(buf),0);

    /*
     先等待,让客户端接收完毕
    */
    recv(client,buf,64,0);
   }
   fclose(file);
}
end:
closesocket(client);
goto start;

WSACleanup();
exit(0);
return 0;
}
int main(char argv[],int argc)
{
int flag;
WSADATA wd;
flag=WSAStartup(MAKEWORD(1,1),&wd);
if(flag!=0)
{
   cerr<<"version error!"<<endl;
   return 0;
}
/*
   创建一个服务器socket
*/
server=socket(AF_INET,SOCK_STREAM,0);
/* 
   设定服务器socket的端口、使用的协议族
*/
SOCKADDR_IN ser_addr;
ser_addr.sin_addr.S_un.S_addr=htonl(INADDR_ANY);
ser_addr.sin_port=htons(host_port);
ser_addr.sin_family=AF_INET;
/*
   与对应的端口相绑定
*/
bind(server,(sockaddr*)&ser_addr,sizeof(sockaddr));//x
/*
   将socket server设为监听模式
*/
listen(server,50);
/*
   开始对请求进行监听
*/

CreateThread(0,0,ftp_server,0,0,0);
while(1)
   Sleep(10000);
return 1;
}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值