1

以下のように、backgroundworker do_dowork イベントによって返される進捗状況を示すプログレスバーがあります。

       if (ftpSourceFilePath.Scheme == Uri.UriSchemeFtp)
            {
                FtpWebRequest objRequest = (FtpWebRequest)FtpWebRequest.Create(ftpSourceFilePath);
                NetworkCredential objCredential = new NetworkCredential(userName, password);
                objRequest.Credentials = objCredential;
                objRequest.Method = WebRequestMethods.Ftp.DownloadFile;
                FtpWebResponse objResponse = (FtpWebResponse)objRequest.GetResponse();
                StreamReader objReader = new StreamReader(objResponse.GetResponseStream());
                int len = 0;
                int iProgressPercentage = 0;
                FileStream objFS = new FileStream((cd+"\\VolareUpdate.rar"), FileMode.Create, FileAccess.Write, FileShare.Read);

                    while ((len = objReader.BaseStream.Read(buffer, 0, buffer.Length)) > 0)
                        {

                            objFS.Write(buffer, 0, len);
                            iRunningByteTotal += len;
                            double dIndex = (double)(iRunningByteTotal);
                            double dTotal = (double)buffer.Length;
                            double dProgressPercentage =  (dIndex / dTotal);

                            iProgressPercentage = (int)(dProgressPercentage);

                            if (iProgressPercentage > 100)
                            {
                                iProgressPercentage = 100;

                            }

                            bw.ReportProgress(iProgressPercentage);


                        }
}

ただし、プログレスバーは更新されません。検索中に、UIスレッドがブロックされていると言われ、ループの外に進行状況を渡すとうまくいくと思いました。それから私はこれに変わります

if (ftpSourceFilePath.Scheme == Uri.UriSchemeFtp)
                {
                    FtpWebRequest objRequest = (FtpWebRequest)FtpWebRequest.Create(ftpSourceFilePath);
                    NetworkCredential objCredential = new NetworkCredential(userName, password);
                    objRequest.Credentials = objCredential;
                    objRequest.Method = WebRequestMethods.Ftp.DownloadFile;
                    FtpWebResponse objResponse = (FtpWebResponse)objRequest.GetResponse();
                    StreamReader objReader = new StreamReader(objResponse.GetResponseStream());
                    int len = 0;
                    int iProgressPercentage = 0;
                    FileStream objFS = new FileStream((cd+"\\VolareUpdate.rar"), FileMode.Create, FileAccess.Write, FileShare.Read);
    while ((len = objReader.BaseStream.Read(buffer, 0, buffer.Length)) > 0)
                        {

                            objFS.Write(buffer, 0, len);
                            iRunningByteTotal += len;
                            double dIndex = (double)(iRunningByteTotal);
                            double dTotal = (double)buffer.Length;
                            double dProgressPercentage =  (dIndex / dTotal);

                            iProgressPercentage = (int)(dProgressPercentage);

                            if (iProgressPercentage > 100)
                            {
                                iProgressPercentage = 100;

                            }
                           // System.Threading.Thread.Sleep(2000);
                              iProgressPercentage++;
                         //    SetText("F", true);




                        }

                         bw.ReportProgress(iProgressPercentage);
                         progressBar.Refresh();

}

しかし、それでも役に立ちませんでした。workerprogresschanged イベントにブレークポイントを設定すると、progressbar.value が表示されますが、更新されません。私はprogressbar.update(0を試しました。また、ループ内でしばらくスレッドをスリープさせようとしましたが、それでも解決しませんでした。提案/ヘルプをいただければ幸いです。

4

4 に答える 4

0

返信ありがとうございます。以下のようなデリゲートメソッドを使用してこれを達成しました。

  delegate void SetProgressBar(int percentage, bool logic);
        private void SetProgress(int percentage, bool logic)
        {
            if (progressBar.InvokeRequired)  
            {
                SetProgressBar d = new SetProgressBar(SetProgress);
                this.Invoke(d, new object[] { percentage, logic });

            }
            else
            {

                progressBar.Style = ProgressBarStyle.Blocks;
                progressBar.Value = percentage;
                progressBar.Text = percentage.ToString() + "%";

            }


        }

そして私のドワークイベント

private void bw_DoWork(object sender, DoWorkEventArgs e)
 {
            BackgroundWorker worker = sender as BackgroundWorker;


  while ((len = objReader.BaseStream.Read(buffer, 0, buffer.Length)) != 0)
                        {
                           .....

                          .......
                            int iProgressPercentage = (int)(dProgressPercentage * 100);
                            bw.ReportProgress(iProgressPercentage);

                        }
}

私の Progress changed イベントでは、上記のデリゲート メソッドをトリガーします。

   private void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {


             SetProgress(e.ProgressPercentage, true);


        }

これは問題なくうまく機能します...すべての貢献に感謝します。

于 2013-11-12T06:53:46.343 に答える
0
private void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {
            progressBar1.Value = e.ProgressPercentage;
        }
于 2013-11-06T04:48:27.843 に答える
0

Application.DoEvents() への呼び出しが欠落していると思います

ループのような長時間実行されるタスクで UI を変更したい場合は、呼び出す必要があります

Application.DoEvents()

これには、UI に加えられた変更が反映されます。

于 2013-11-06T05:34:35.853 に答える