FtpGetFile WinINet 永远不会返回

Posted

技术标签:

【中文标题】FtpGetFile WinINet 永远不会返回【英文标题】:FtpGetFile WinINEt never returns 【发布时间】:2013-09-12 16:40:56 【问题描述】:

我遇到了一个奇怪的问题(很奇怪,让我说,呵呵)。在 FTP 下载 EXE 文件 (24 MB) 期间,如果连接中断,则 WinINEt 库的函数 FtpGetFile 似乎有错误并且永远不会返回。这会导致将来的文件传输失败(连接已打开)。 显然,我通过增加服务器传输的超时找到了一种解决方法,但我不喜欢它。我用谷歌搜索没有发现类似的问题(可能是我引入了错误的关键字)。

我在网上看了一些论坛,似乎每个人都不推荐使用 FtpGetFile,因为它有问题。

这出现在一个有很大延迟(并不总是)的网络场景中,但在良好的条件下它会消失(下载正确进行并且 FtpGetFile 总是返回)。

这是我如何使用该功能:

if( FtpGetFile(m_hFtpSession, strSourcePath.c_str(), strTargetPath.c_str(), 0, 0, FTP_TRANSFER_TYPE_BINARY, 0)==TRUE)

谁能证实?我应该重构我的代码并寻找更新吗?

谢谢

【问题讨论】:

你能解释一下-1吗??????这是拖钓吗? 【参考方案1】:

我找到了一种无需使用 FtpGetFile 即可下载文件的方法。我希望这段代码可以帮助某人:

bool RetrieveFile(const string& strSource, const string& strTarget) 

   /* The handle for the transfer */
   HINTERNET hTransfer = NULL;

   /*
    * Set default error
    */
   DWORD error = ERROR_SUCCESS;

   if( !isConnected ) 
     debug("%s(): ERROR not connected\n", __FUNCTION__);
     return false;
   

   /* Initiate access to a remote FTP connection */
   hTransfer = FtpOpenFile(hFtpSession, strSource.c_str(), GENERIC_READ,
       FTP_TRANSFER_TYPE_BINARY, 0);

   if(hTransfer) 
      std::ofstream myostream(strTarget.c_str(), std::ios::binary);
      if ( myostream.is_open() ) 
         static const DWORD SIZE = 1024;

         BYTE data[SIZE];
         DWORD size = 0;
         do 
            BOOL result = InternetReadFile(hTransfer, data, SIZE, &size);
            if ( result == FALSE ) 
               error = GetLastError();
               Debug("InternetReadFile(): %lu\n", error);
            
            myostream.write((const char*)data, size);
         
         while ((error == ERROR_SUCCESS) && (size > 0));

         // Close the stream
         myostream.close();
      
      else 
         Debug("Could not open '%s'.\n", strTarget.c_str());
         error = ERROR_FILE_NOT_FOUND; // Not necessarily not found, but it is to describe a file error which is different from ERROR_SUCCESS
      

      // Close
      const BOOL result = InternetCloseHandle(hTransfer);
      if ( result == FALSE ) 
         const DWORD error = GetLastError();
         debug("InternetClose(): %lu\n", error);
      

      /* Check error status of the process */
      return (error == ERROR_SUCCESS);
   

   DWORD dwInetError;
   DWORD dwExtLength = 1000;
   TCHAR *szExtErrMsg = NULL;
   TCHAR errmsg[1000];
   szExtErrMsg = errmsg;
   int returned = InternetGetLastResponseInfo( &dwInetError, szExtErrMsg, &dwExtLength );
   debug("dwInetError: %d  Returned: %d\n", dwInetError, returned);
   debug("Buffer: %s\n", szExtErrMsg);

   debug("%s() : ERROR to get '%s' file (errorCode=%d)\n", __FUNCTION__, strSource.c_str(), GetLastError());

   return false;

【讨论】:

以上是关于FtpGetFile WinINet 永远不会返回的主要内容,如果未能解决你的问题,请参考以下文章

为啥 IsDialogMessage() 可能永远不会返回?

RequestPermissionsAsync 永远不会返回

WinInet::InternetSetOption(...) 总是返回 0 并且 GetLastError() 返回 12018

为啥 eof() 永远不会返回 true?

WinInet 能否返回正在使用的 TLS

对 CancellationTokenSource.Cancel 的调用永远不会返回