用VC++编制FTP客户端应用程序FTP协议将使用两条单独的TCP连接,一条专用于发送FTP命令,另一条则专用于传递数据。初始建立连接时,服务器在21号端口上接收来自客户端的命令连接。当需要传送数据时(文件列表、文件数据等),客户端向服务器发出Port命令,并进入监听状态,等待来自服务器的数据连接请求。
首先我们利用VC++ 6.0的AppWizard创建一个基于对话框的应用程序,命名为FtpClientDemo。为程序生成五个基于CAsyncSocket的新类,这里列出主要代码。
CCommandSocket类的主要代码
void CCommandSocket::OnReceive(int nErrorCode)
{
//这个函数使得服务器的应答消息显示在编辑框上
char buffer=new char[4096];
memset(buffer,0,4096);
this-〉Receive(buffer,1024,0);
//接收应答消息
MessageList+=buffer;
m_ReturnMessage-〉SetWindowText(MessageList);
delete buffer;
}
CFileSocket类的主要代码
void CFileSocket::OnReceive(int nErrorCode)
{
//函数将收到的文件数据写到文件中
if(File= =NULL)
{ File=new CFile();
File-〉Open(FileName,CFile::modeWrite|CFile::modeCreate);
}
charbuffer=new char[4096];
memset(buffer,0,4096);
this-〉Receive(buffer,4096,0);
ReceiveString=buffer;
File-〉Write(ReceiveString,ReceiveString.GetLength( ));
delete buffer;
}
CReceiveSocket类的主要代码
void CReceiveSocket::OnReceive(int nErrorCode)
{
//接收服务器传来的文件列表消息
CString ReceiveString,Temp;
charbuffer=new char[4096];
memset(buffer,0,4096);
this-〉Receive(buffer,4096,0); //接收消息
ReceiveString+=buffer;
delete buffer;
//将文件列表从收到的消息字符串中分离出来,并显示在列表框中
while(!ReceiveString.IsEmpty())
{ int p=ReceiveString.Find("\r\n");
if(p!=-1)
{ Temp=ReceiveString.Left(p);
ReceiveString=ReceiveString.Right(ReceiveString.GetLength()-p-2);
DisplayMessage-〉AddString(Temp);
}
}
}
CPortSocket类主要代码
void CPortSocket::OnAccept(int nErrorCode)
{
//根据不同的标志选择相应的数据连接类,以接受服务器端的数据连接请求
if(Flag= =LISTFILE)
//若程序要求对目录进行列表,则采用CReceiveSocket类
{DataSocket=new CReceiveSocket(FileList);
this-〉Accept(DataSocket);
}
else if(Flag= =DOWNLOAD)
//若程序要求下载文件,则生成CFileSocket类的对象
{FileSocket=new CFileSocket(FileName);
this-〉Accept(FileSocket);
}
}
主对话框类CFtpClient- DemoDlg的主要代码
void CFtpClientDemoDlg::OnFileList()
//响应“文件列表”按钮、列表目录
{ CString Temp;
if(ControlSocket= =NULL)
{
//连接到FTP服务器
ControlSocket=new CCommandSocket(&&m_ReturnMessage);
ControlSocket-〉Create();
m_Server.GetWindowText(Temp);
ControlSocket-〉Connect(Temp,21);
//FTP服务器在21号端口接收连接
}
m_User.GetWindowText(Temp);
Temp="USER "+Temp+"\r\n";
ControlSocket-〉Send(Temp,Temp.GetLength(),0);
//发User命令,验证用户
m_Pass.GetWindowText(Temp); //m_Pass为“口令”编辑框的对应控制
Temp="PASS "+Temp+"\r\n";
ControlSocket-〉Send(Temp,Temp.GetLength(),0);
//发Pass命令,校验口令
LisentPort(LISTFILE);
//数据连接的对象为目录列表
ControlSocket-〉Send("LIST \r\n",7 ,0);
//发List命令,要求列表目录
}
void CFtpClientDemoDlg::OnDownLoad( )
//下载文件
{
CString String;
LisentPort(DOWNLOAD);
//获得要下戴文件的文件名
m_LocalFile.GetWindowText(String);
// m_LocalFile为“文件名”编辑框的对应控制
String="RETR "+String+"\r\n";
ControlSocket-〉Send(String,String.GetLength( ),0);
//发RETR命令,下载文件
}
void CFtpClientDemoDlg::LisentPort(UINT Flag)
{
//根据要求选择不同的数据连接对象
if(LisentSocket!=NULL)
//清空LisentSocket
{ LisentSocket-〉Close();
delete LisentSocket;
LisentSocket=NULL;
}
if(Flag= =LISTFILE)
//如果为目录列表数据连接对象
{ LisentSocket=new CPortSocket(LISTFILE);
LisentSocket-〉SetListBox(&&m_FileList);
//传列表框到CLisentSocket类中
}
else if(Flag= =DOWNLOAD)
//如果为文件传输数据连接对象
{ CString String;
m_LocalFile.GetWindowText(String);
LisentSocket=new CPortSocket(DOWNLOAD);
LisentSocket-〉SetFileName(String);
//传文件名到CLisentSocket类中
}
LisentSocket-〉Create();
//建立Socket并进行监听,等待FTP服务器进行数据连接
LisentSocket-〉Listen();
//取得数据连接Socket的IP地址和监听端口,并把它们作为Port命令的参数
SOCKADDR_IN listing_address, control_address;
int addr_size;
addr_size = sizeof(listing_address);
LisentSocket-〉GetSockName((SOCKADDR )&&listing_address, &&addr_size); //
取IP地址
ControlSocket-〉GetSockName((SOCKADDR )&&control_address, &&addr_size); /
/取端口
unsigned char port = (unsigned char )&&(listing_address.sin_port);
unsigned char host = (unsigned char )&&(control_address.sin_addr);
CString strBuffer;
strBuffer.Format("PORT %i,%i,%i,%i,%i,%i\r\n",(int)host[0], (int)host[1], (
int)host[2], (int)host[3],(int)port[0], (int)port[1]);
ControlSocket-〉Send(strBuffer,strBuffer.GetLength(),0);
//发送Port命令,进行数据连接
}
以上代码在VC++ 6.0、Windows 98上运行通过。