0

我试图弄清楚如何在下载许多文件的文件夹时计算当前的下载速度(它不是我正在下载的单个文件)。而且我无法正确地做到这一点,现在经历了好几个小时,它变得太混乱了。下载速度有时会太高,有时会为 0。

我正在使用 curl 和 C++。

在我的下载功能中,程序递归下载每个文件,直到所有文件都被下载。

这就是我设置 curl 以在下载期间调用 TraceProgress 函数的方式:

curl_easy_setopt( curl, CURLOPT_PROGRESSFUNCTION, TraceProgress );
curl_easy_setopt( curl, CURLOPT_PROGRESSDATA, &response );
curl_easy_setopt( curl, CURLOPT_NOPROGRESS, 0 );

这是剩余的代码:

    double totalDownloadableSize = 0; // total size of the download, set prior to starting download of the first file
    double downloadedSizeTillNow = 0; // total size downloaded till now - keep adding file size that just completed downloading
    double currentDownloadingSize = 0; // size that we are downloading - downloadedSizeTillNow + bytes downloaded of the current file (in TraceProgress Function)
    double oldDownloadNow = 0; // size of the old download bytes of that particular file


    string fileDownloading = "";
    string tempFileDownloading = "";

    time_t startSeconds;
    time_t oldSeconds;

    int downloadIterationCounter = 0;




    int TraceProgress( void *clientp, double dltotal, double dlnow, double ultotal, double ulnow )
    {

        // add size downloaded till now of this file to the total size downloaded till now
        currentDownloadingSize = downloadedSizeTillNow + dlnow;

        double rem = ( ( totalDownloadableSize - currentDownloadingSize ) / 1024 ) / 1024;

        // get current time in seconds
        time_t currentSeconds = time (NULL);

        // get elapsed time since last itiration
        time_t secondsElapsedSinceLastItiration = currentSeconds - oldSeconds;

        double downloadSinceLastIteration;
        if ( oldDownloadNow < dlnow )// so that we don't get wrong data when download file changes
        {
            downloadSinceLastIteration = dlnow - oldDownloadNow;
        }
        else
        {
            downloadSinceLastIteration = dlnow;
        }

        // calculate current download speed : (dlnow - oldNow) / (current time - oldTime)
        double currentDownloadSpeed = downloadSinceLastIteration / (double)secondsElapsedSinceLastItiration;


        // if downloading file name is not same as it was in the last call to this function
        // change the display text and save the name in the temp. This approach will avoid unnecessory
        // text change calls.
        if ( fileDownloading.compare( tempFileDownloading ) != 0 )
        {
             tempFileDownloading = fileDownloading;
             string dlfilename = "Downloading:  " + fileDownloading;
             SetWindowText( hDownloadingSTATIC, dlfilename.c_str() );// set text to static control
        }


        if ( downloadIterationCounter == 4 )
        {
             std::ostringstream strs_dn;
             strs_dn << (unsigned int)( rem );
             std::string downloadNow = strs_dn.str();


             string remSize = "Remaining:  " + downloadNow + " MB";
             SetWindowText( hRemainingDownloadSTATIC, remSize.c_str() );// set text to static control


             double idownloadSpeed = currentDownloadSpeed / 1024;

             std::ostringstream strs_dnSp;
             strs_dnSp << (unsigned int)( idownloadSpeed );
             std::string downloadSpeed = strs_dnSp.str();

             string downSize = "Download Speed:  " + downloadSpeed + " KB/s";
             SetWindowText( hDownloadSpeedSTATIC, downSize.c_str() );// set text to static control


             oldSeconds = currentSeconds;// save in old
             oldDownloadNow = dlnow;// save in old


             downloadIterationCounter = 0;
        }
        else
        {
             downloadIterationCounter++;
        }

        return 0;
    }

任何帮助表示赞赏。非常感谢。

4

1 回答 1

1

dlnow - oldDownloadNow可能是错的。你应该只是使用dlnow。你根本不需要oldDownloadNow,除非你想显示下载速度的变化率

我被 `dltotal` 和 `dlnow` 的名字弄糊涂了。`dltotal` 是下载的*预期*总字节数,`dlnow` 是到目前为止下载的字节数。所以确实需要 `oldDownloadNow`,而 `dlnow - oldDownloadNow` 是当前的增量。

这个片段

    if ( oldDownloadNow < dlnow )
    {
        downloadSinceLastIteration = dlnow - oldDownloadNow;
    }
    else
    {
        downloadSinceLastIteration = dlnow;
    }

有错误:oldDownloadNow == dlnow表示自上次以来没有下载任何内容。在这种情况下,当前的瞬时下载速度为零。片段应替换为

    downloadSinceLastIteration = dlnow - oldDownloadNow;

没有任何检查。如果libcurl突然决定oldDownloadNow > dlnow,这将适时显示负下载速度,这是正确的做法。永远不要故意隐藏错误。

另外,time()分辨率太粗。使用某种细粒度的计时器。

于 2012-12-09T16:08:18.777 回答