FTP服务的LIST指令获取的文件列表信息的解析

类别:VC语言 点击:0 评论:0 推荐:

客户端使用LIST命令指定获取服务器端FTP共享目录(或者下面的子目录),服务器端将通过数据端口将该指定目录下的文件列表(包括子目录)信息发送给客户端。本文对该文件列表信息进行分析和解析。

文件列表信息分为UNIX格式和DOS格式两种。笔者是比照了Serv-U和微软自带的FTP服务器写出本文的,也许别的服务器另有新的方式或者格式也说不定,欢迎大家补充。

首先不妨来看一下UNIX格式和DOS格式下的文件列表信息都是怎么样的:

 //MS-DOS文件列表格式解析
 //02-23-05  09:24AM                 2245 readme.ESn
 //05-25-04  08:56AM             19041660 VC.ESn


  UNIX文件列表格式解析
  UNIX文件格式:
  Serv-U:
  -rwxrw-r--   1 user     group        3014 Nov 12 14:57 cwinvnc337.ESn
  -rwxrw-r--   1 user     group       20480 Mar  3 11:25 inmcsvr更新说明.ESn
  -rwxrw-r--   1 user     group         450 Apr 13 11:39 对话框中加入工具条.ESn
  Windows自带FTP:
  -rwxrwxrwx   1 owner    group        19041660 May 25  2004 VC.ESn
  -rwxrwxrwx   1 owner    group             450 Apr  6 15:04 对话框中加入工具条.ESn

注:由于未发现Serv-U支持DOS格式,因此DOS格式只列了微软自带的。

下面我们对以上的格式进行分析:

首先,文件列表信息中,每个文件的信息之间用回车换行符(\r\n)分隔。因此分解时第一步就是用\r\n进行截取。然后是对每一个文件信息的解析。

每一个文件信息中,分为多个信息段,各个信息段之间用一个空格符间隔。UNIX格式和DOS格式的信息段的数量的顺序是不同的。下面将分别分析。

先看看DOS格式,拿出一条文件信息来讲:02-23-05  09:24AM                 2245 readme.ESn


第一段为05-25-04,一个空格后,为第二段08:56AM,一个空格后,为    19041660,由于文件长度不一定,预留的位置比较长,因此前面用空格填充了。
解析的时候,逐段用空格截取,记住,截取完第一段后,剩下的内容先用TrimLeft去除左侧的空格,然后继续截取就可以了。
因此,DOS格式共分四段,其中第一段为日期,第二段为时间,第三段为文件长度,第四段为文件名称。

对了,如果只需要获取文件名称,你也不能从后面截取,因为文件名称是允许带空格的。:》
另外,如果列举的是个目录,那么,第三段就不是文件长度了,而是固定为:<DIR>


再看UNIX格式,也拿出一条文件信息来讲:
-rwxrw-r--   1 user     group        3014 Nov 12 14:57 cwinvnc337.ESn
unix我不熟,每一段的意义不太清楚。但以上的格式分解为:第一段为-rwxrw-r--,第二段为1,第三段为user,第四段为group,第五段为文件长度,第六段为月,第七段为日,第八段为时间,第九段为文件名称。
需要注意的是:如果格式串的第一个字符为d,表示为一个目录信息,比如drwxrw-r--
另外,第八段有可能不是时间,而是年份,比如2005,从上面的例子中你可以发现。


对于不同的FTP服务器,LIST获取的信息不尽相同,但段的顺序和意义是不变的。只是表示文件的长度的段的长度有所不同。
以下是笔者在实际项目中的解析函数,做的不是很好,但希望对大家有所帮助吧。

/***************************************************
 Function:  CRecvFileMan::PraseFileList_MSDOS
 Description: 解析MSDOS风格的文件列表
 Table Accessed: 
 Table Updated: 
 Parameter:  CString sFileList - MSDOS风格文件列表
 Return:   无返回值
 Others:   
***************************************************/
void CRecvFileMan::PraseFileList_MSDOS(CString sFileList)
{
 CString sLen;
 CString sFile;
 CString sOneFile;
 int nIdx = 0;
 while(1)
 {
  nIdx = sFileList.Find("\r\n");
  if(nIdx == -1)
   break;
  sOneFile = sFileList.Left(nIdx);
  sFileList = sFileList.Mid(nIdx + 2);
  sLen = GetSegmentInfo(sOneFile,2);
  if(sLen == "<DIR>")
  {
   continue;
  }
  sFile = GetSegmentInfo(sOneFile,0);
  //根据解析的文件信息,形成下载文件类对象
  CRecvFile *pFile = new CRecvFile(sFile,atoi(sLen));
   m_arTransFile.Add(pFile);
 };
}

/***************************************************
 Function:  CRecvFileMan::PraseFileList_UNIX
 Description: 解析UNIX风格的文件列表
 Table Accessed: 
 Table Updated: 
 Parameter:  CString sFileList - UNIX风格文件列表,以回车换行分隔
 Return:   无返回值
 Others:   
***************************************************/
void CRecvFileMan::PraseFileList_UNIX(CString sFileList)
{
 CString sLen;
 CString sFile;
 CString sOneFile;
 int nIdx = 0;
 while(1)
 {
  nIdx = sFileList.Find("\r\n");
  if(nIdx == -1)
   break;
  sOneFile = sFileList.Left(nIdx);
  sFileList = sFileList.Mid(nIdx + 2);
  if(sOneFile.GetAt(0) == 'd')//第一个字母是d,表示是目录,忽略
   continue;
  sLen = GetSegmentInfo(sOneFile,4);
  sFile = GetSegmentInfo(sOneFile,3);
  //根据解析的文件信息,形成下载文件类对象
   CRecvFile *pFile = new CRecvFile(sFile,atoi(sLen));
   m_arTransFile.Add(pFile);
 };
}

/***************************************************
 Function:  CRecvFileMan::GetSegmentInfo
 Description: 得到文件列表中某个文件描述的指定段的信息
 Table Accessed: 
 Table Updated: 
 Parameter:  CString &sFileInfo - 指定的文件描述信息。即是传入,也是传出参数
           将获取段号信息后剩余的信息返回,以便进一步截取其它信息
     int nSegment - 指定的段号,从0开始
 Return:   CString - 指定段号的信息
 Others:   
***************************************************/
CString CRecvFileMan::GetSegmentInfo(CString &sFileInfo,int nSegment)
{
 int nIdx = -1;
 int nSeg = 0;
 CString sInfo = "";
 sFileInfo.TrimLeft();
 while(nSeg < nSegment + 1)//逐段切隔
 {
  nIdx = sFileInfo.Find(" ");//以空格为切隔
  if(nIdx < 0)//如果已没有空格,即最后一段信息
  {
   if(nSeg == nSegment)//如果最后一段正好是需要的段
   {
    sInfo = sFileInfo;//返回剩余信息
   }
   else
    sInfo = "";
   break;
  }
  else
  {
   sInfo = sFileInfo.Left(nIdx);//得到段信息
   sFileInfo = sFileInfo.Mid(nIdx+1);//切隔信息
   sFileInfo.TrimLeft();//过滤左侧的空格符
  }
  nSeg++;
 }
 return sInfo;
}

本文地址:http://com.8s8s.com/it/it29.htm