socket实现FTP客户端

通过socket实现FTP客户端(MFC版)

使用socket我们可以自己编写一个FTP的客户端。开始之前先了解FTP的两种工作模式大致的过程。

被动模式的流程

被动模式下,FTP客户端使用一个非知名的随机端口N(>1024)去连接FTP服务器的21端口建立控制连接,然后通过控制通道发送PASV命令到服务器,服务器收到后会开启一个随机端口P(>1024)并开始监听,同时告诉客户端连接我的端口P,客户端随后使用N+1端口连接服务器的端口P,建立数据连接。

主动模式的流程

主动模式下,FTP客户端使用一个非知名的随机端口N(>1024)去连接FTP服务器的21端口建立控制连接,然后通过控制通道发送PORT N+1命令到服务器,同时开始监听N+1端口,服务器收到后主动使用20端口与客户端的N+1端口连接,建立数据连接。

客户端与服务器交互的常用命令

这里的命令是指底层数据包封装的命令,并不是指用户在控制台使用的与服务器交互的命令,比如dir、get、put等等。

  • 客户端使用USER、PASS命令登录 FTP 服务器。

  • 客户端使用PASV命令告知服务器使用被动模式,获取服务器数据连接的端口号,与服务器建立数据连接。

  • 客户端使用PORT命令告知服务器使用主动模式,附带自己监听的数据端口,让服务器主动与其建立数据连接。

  • 客户端使用SIZE命令获取文件的大小。

  • 客户端使用RETR、STOR命令下载、上传文件。

  • 客户端使用QUIT命令退出连接。

    使用socket实现FTP客户端

    创建socket

    WSADATA WSAData;
      if (WSAStartup(MAKEWORD(2,2),&WSAData)!=0)//建立套接字绑定
      {
         
          MessageBox("初始化Winsock失败!");
          return;
      }
      SockCtrl=socket(AF_INET,SOCK_STREAM,0);//创建连接套接字
      if (SockCtrl==INVALID_SOCKET)
      {
         
        MessageBox("创建控制Socket失败!");
        WSACleanup();
        return;
      }
    

    建立控制连接

    nConnect=connect(SockCtrl,(sockaddr*)&serveraddr,sizeof(serveraddr));//建立FTP控制连接
      if (nConnect==SOCKET_ERROR)
      {
         
          MessageBox("控制连接建立失败!");
          closesocket(SockCtrl);
          WSACleanup();
          return;
      }
      if (!RecvRespond())
          return;
      else
      {
         //判断连接应答码
        if (RespondCode==220)
        {
         
           m_tips+="Server:  ";
           m_tips+=Respond;
        }
        else
        {
         
            MessageBox("控制连接响应错误!");
            closesocket(SockCtrl);
            WSACleanup();
            return;
        }
      }
    

    定义SendCommand函数,用于客户端发送命令

    //发送命令
    bool CFtpClientDlg::SendCommand(void)
    {
         
       int nSend;
     nSend=send(SockCtrl,Command,strlen(Command),0);
     if (nSend==SOCKET_ERROR)
     {
         
        MessageBox("Socket发送失败!");
        closesocket(SockCtrl);
        WSACleanup();
        return false;
     }
    
     return true;
    }
    

    定义RecvRespond函数,用于客户端接收响应码

    //接收响应码
    bool CFtpClientDlg::RecvRespond(void)
    {
         
          int nRecv;
      memset(Respond,0,MAX_SIZE);
      nRecv=recv(SockCtrl,Respond,MAX_SIZE,0);//通过连接接收响应
      if (nRecv==SOCKET_ERROR)
      {
         
         MessageBox("Socket接收失败!");
         closesocket(SockCtrl);
         WSACleanup();
         return false;
      }
      //从响应中解析响应码
      char* ReplyCodes=new char[3];
      memset(ReplyCodes,0,3);
      memcpy(ReplyCodes,Respond,3);
      RespondCode=atoi(ReplyCodes);
      return true;
    }
    

    验证用户名、口令

    sprintf(Command,"USER %s\r\n",m_user); //Command为事先定义的缓冲区
        if (!SendCommand())
          return;
    
      m_tips+="Client:  ";//m_tips为编辑框控件变量,用于显示交互过程
      m_tips+=Command;
    
      if (!RecvRespond())
          return;
      else
      {
         //判断USER应答码
         if (RespondCode==220 || RespondCode==331)
         {
         
            m_tips+="Server: ";
            m_tips+=Respond;
         }
         else
         {
         
            MessageBox("USER响应错误!");
            closesocket(SockCtrl);
            WSACleanup();
            return;
         }
      }
    
      if (RespondCode==331)//User name okay,need password
      {
          //合成PASS命令
          memset(Command,0,MAX_SIZE);
          sprintf(Command,"PASS %s\r\n",m_pwd);
          if (!SendCommand())
              return;
          m_tips+="Client: PASS ******\r\n";
          if (!RecvRespond())
              return;
          else
          {
         //判断PASS响应码
              if (RespondCode==230)
              {
         
                 m_tips+="Server: ";
                 m_tips+=Respond;
                 m_CONNECT.EnableWindow(FALSE);
                 m_QUIT.EnableWindow
  • 3
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
ftp客户端ftpclient纯C语言winsock实现socket编程 /* *本程序是2009年计算机网络课程设计作品。 *本程序参考rfc959标准。能准确与遵守此标准的服务器进行信息交互。 *本人不保留任何版权。 *本程序仅供学习研究测试使用。因使用本程序所有或部分代码所产生的任何后果,本人均不负任何法律责任。 *2009年7月13日 */ #include #include #include /*system()*/ #include #pragma comment(lib,"ws2_32.lib") #define ONUM 512 #define MNUM 512 #define FNUM 512 #define pt struct host { char ip[20]; unsigned short port; }; SOCKET ts; fd_set readfds; struct timeval timeval; struct host host; char renum[4]; char ordertemp[ONUM]; char order[ONUM]; char ordercp[ONUM]; char mess[MNUM]; char file[FNUM]; char setpath[FNUM]; char setpathf[FNUM]; int i,door,r,sys,seti;/*i for;door switch;r receive num;sys system state;set set state*/ char *p;/*strtok*/ int printmess(); void input(char ordertemp[]); int ftp(); int user(); int pass(); int command(); int list(); SOCKET createDataSocket(); int set(); int retr(); int stor(); int stor() { char filename[256]; SOCKET ds; int wi; int r2,r3,r4,bsnum,brnum; FILE *fp=NULL; set(); memset(filename,'\0',256); memset(order,'\0',ONUM); for(i=5;ibsnum); }/*while*/ printf("\n"); switch(printmess()) { case 425: case 426: case 451: case 551: case 552:closesocket(ds);fclose(fp);return -1; case 250: case 226:fclose(fp);return 0; default:return 0; } }/*stor*/ int retr() { char filename[256]; unsigned long fsize,wfsize; int r2,wi; SOCKET ds; FILE *fp=NULL; memset(filename,'\0',256); memset(order,'\0',ONUM); for(i=5;i<=200&&ordercp[i]!='\0';i++)order[i-5]=ordercp[i]; strcpy(filename,setpathf); strcat(filename,"\\\\"); strcat(filename,order); ds=createDataSocket(); if(ds==-1)return -1; memset(order,'\0',ONUM); strcpy(order,"type i\r\n");/*type i 二进制 type a ASCII*/ send(ts,order,strlen(order),0); switch(printmess()) { case 421:closesocket(ds);closesocket(ts);sys=0;return 421; case 530:closesocket(ds);sys=1;return -1; case 500: case 501: case 504: case 226:closesocket(ds);return -1; case 200:break; default:closesocket(ds);return -1; } strcat(ordercp,"\r\n"); send(ts,ordercp,strlen(ordercp),0); switch(printmess()) { case 421:closesocket(ds);closesocket(ts);sys=0;return 421; case 530:closesocket(ds);sys=1;return -1; case 450: case 500: case 501: case 550:closesocket(ds);return -1; case 125: case 150:break; default:closesocket(ds);return -1; } set(); system(setpath); fp=fopen(filename,"wb"); if(!fp){printf("create file fail!\n");closesocket(ds);printmess();return -1;} wi=1;fsize=0;wfsize=0; while(wi) { memset(file,'\0',FNUM); r=recv(ds,file,FNUM,0); if(r==SOCKET_ERROR) { printf("file recv error!\n"); closesocket(ds); fclose(fp); return -1; } fsize=fsize+r; if(r==0){wi=0;break;} printf("receive %8d bytes! have received %16d bytes!\r",r,fsize); r2=fwrite(file,sizeof(char),r,fp); fflush(fp);/*这里是关键*/ wfsize=wfsize+r2; printf("write %8d bytes!have written %20d bytes!\r",r2,wfsize); }/*while*/ printf("\n"); switch(printmess()) { case 425: case 426: case 451:closesocket(ds);fclose(fp);return -1; case 250: case 226: closesocket(ds); wi=1; while(wi)if(fclose(fp)==0)wi=0;return 0; default:return 0; } }/*retr()*/ 以下代码请下载本程序。VC++6.0编译通过

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值