Как правильно рассчитать скорость загрузки с Httpclient ReadAsStreamAsync? - PullRequest
0 голосов
/ 28 апреля 2018

Как правильно рассчитать скорость загрузки с Httpclient ReadAsStreamAsync? Я пытаюсь получить старую скорость загрузки, вычитая TotalBytesRead - OldTotalBytesRead, она работает как задумано, но проблема в истекшем времени, как я могу сделать это, чтобы он вычислял точно, когда истекшее время равно 1000 мс, в приведенном ниже коде вычисление скорости выполнения> истекшего времени, при котором скорость иногда неточна, что делает первое значение слишком большим, затем секунды маленькими или наоборот.

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Net.Http;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleDownloadApp
{
    class Program
    {

        [DllImport("Shlwapi.dll", CharSet = CharSet.Auto)]
        private static extern long StrFormatByteSize(
            long fileSize,
            [MarshalAs(UnmanagedType.LPTStr)] StringBuilder buffer,
            int bufferSize);

        static HttpClient _httpClient = new HttpClient();

        static long ContentLength = 0;
        static long TotalBytesRead = 0;
        static long OldTotalBytesRead = 0;

        static void Main(string[] args)
        {
            string url = "http://ipv4.download.thinkbroadband.com/10MB.zip";
            string savePath = "10MB.zip";

            Task.Run(() => Download(url, savePath)).Wait();
        }

        static async Task Download(string url, string saveAs)
        {
            using (HttpResponseMessage response = await _httpClient.GetAsync(url, HttpCompletionOption.ResponseHeadersRead))
            {
                response.EnsureSuccessStatusCode();

                ContentLength = response.Content.Headers.ContentLength ?? 0;

                byte[] buffer = new byte[1024];

                Stopwatch st = new Stopwatch();
                st.Start();

                using (Stream contentStream = await response.Content.ReadAsStreamAsync())
                using (var fileStream = new FileStream(saveAs, FileMode.Create, FileAccess.Write, FileShare.None, buffer.Length, true))
                {
                    while(true)
                    {
                        var bytesRead = await contentStream.ReadAsync(buffer, 0, buffer.Length);
                        if(bytesRead == 0)
                        {
                            Console.WriteLine("Download complete.");
                            break;
                        }

                        await fileStream.WriteAsync(buffer, 0, bytesRead);

                        TotalBytesRead += bytesRead;

                        var elapsedTime = st.ElapsedMilliseconds;
                        if (elapsedTime >= 1000)
                        {
                            st.Reset();
                            st.Start();

                            long bytesChanged = TotalBytesRead - OldTotalBytesRead;
                            OldTotalBytesRead = TotalBytesRead;

                            Console.WriteLine("Elapsed time: {0} || Download speed: {1} || {2}/{3}",
                                elapsedTime,
                                StrFormatByteSize(bytesChanged / elapsedTime),
                                StrFormatByteSize(TotalBytesRead),
                                StrFormatByteSize(ContentLength));
                        }
                    }
                }
            }
        }

        /// <summary>
        /// Converts a numeric value into a string that represents the number expressed as a size value in
        /// bytes, kilobytes, megabytes, or gigabytes, depending on the size.
        /// </summary>
        /// <param name="filelength">The numeric value to be converted.</param>
        /// <returns>the converted string</returns>
        public static string StrFormatByteSize(long filesize)
        {
            StringBuilder sb = new StringBuilder(11);
            StrFormatByteSize(filesize, sb, sb.Capacity);
            return sb.ToString();
        }
    }
}

результат загрузки 10 МБ файлов result (can't add photo yet)

...