C#使用socket实现FTP、POP3、SMTP的客户端 (一)

概述

  • socket本质是编程接口,是对TCP/IP的封装。
  • TCP/IP是传输层的协议。
  • FTP、POP3、SMTP都是应用层的协议,是基于TCP/IP协议的。

所以,我们使用socket实现上述几种协议的客户端,其实是对借助socket对TCP/IP数据传输的封装基础,再往上封装一层的。
(简单说,以FTP为例,就是将FTP中的上传或者下载这类“一次”操作,分解成“多次”的通过socket进行数据的传输罢了。)

FTP客户端


界面图:

FTP客户端图

控件由以下组成:

  • 五个textbox: tb_IP, tb_port, tb_username, tb_password, tb_path
  • 三个listbox: lsb_local, lsb_server, lsb_status
  • 四个button: btn_conn, btn_setPath, btn_upload, btn_download

该FTP客户端主要实现了建立连接、上传、下载三个button的功能。


头文件:

using System;
using System.Windows.Forms;
using System.Net.Sockets;
using System.IO;
using System.Text.RegularExpressions;

Sockets包是肯定要的,IO主要是为了使用NetworkStream类来方便socket的读写,RegularExpressions主要用的是它的split()函数


全局变量:

#region  Private variable
private TcpClient cmdServer;
private TcpClient dataServer;
private NetworkStream cmdStrmWtr;
private StreamReader cmdStrmRdr;
private NetworkStream dataStrmWtr;
private StreamReader dataStrmRdr;
private String cmdData;
private byte[] szData;
private const String CRLF = "\r\n";
#endregion

都知道,FTP协议的实现需要建立两个连接,一个21号(通常用21号)端口传输命令,一个随机端口传输数据。所以有两个NetworkStream。

必须注意的是,FTP服务器的命令端口(通常用21号)是保持连接的,数据端口只有在命令端口收到来自Client的请求时才会暂时打开,传输完之后又关闭。

(不了解FTP底层的建议百度“使用telnet执行ftp交互”,代码的实现主要都是通过FTP的命令实现的。)

主要用到的FTP命令如下:

命令 描述
USER <用户名> 登录FTP的用户名
PASS <密码> 登录FTP的密码
QUIT 断开连接
. .
PASV 进入被动模式,返回server的数据端口,等待client连接
ABOR 断开数据端口的连接
. .
LIST 查看服务器文件(从数据端口返回结果)
STOR <文件名> 请求上传
RETR <文件名> 请求下载

(推荐一个FTPServer(迷你FTP服务器)的工具,可以用它来快速创建FTP的服务器端,方便做测试,简单粗暴。有没有毒不敢保证,反正界面简洁,比FileZilla Server轻便一点。百度一下就有了。)


全局函数:

#region  Private Functions

/// <summary>
/// 获取命令端口返回结果,并记录在lsb_status上
/// </summary>
private String getSatus()
{

    String ret = cmdStrmRdr.ReadLine();
    lsb_status.Items.Add(ret);
    lsb_status.SelectedIndex = lsb_status.Items.Count - 1;
    return ret;
}

  • 2
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值