zoukankan      html  css  js  c++  java
  • 使用线程新建WPF窗体(公用进度条窗体)

    使用线程新建窗体

    项目中需要一个公用的进度条窗体.大家知道在wpf中,有两个线程,一个是UI线程,另一个是监听线程(一直监听用户的输入).如果我们后台有阻塞UI线程的计算存在,那么界面上的比如进度条什么动态的UI都会被因为阻塞而静止不动.

    所以我的想法就是新起一个线程来负责进度条窗体的显示与关闭,具体实现代码如下:

       private static Thread th;
            /// <summary>
            /// 显示进度条 线程
            /// </summary>
            /// <param name="msg">进度条界面需要显示的文字</param>
            /// <param name="sleep">增加进度条存在的延迟时间,单位为毫秒</param>
            public static void BeginProgressBar(string msg, int sleep)
            {
                try
                {
                    if (th == null || th.ThreadState == ThreadState.Aborted)
                    {
                        th = new Thread(new ThreadStart(() =>
                       {
                           WinProgressBar win = new WinProgressBar();
                           win.Tips = msg;
                           win.Topmost = true;
                           win.Show();
                           System.Windows.Threading.Dispatcher.Run();
                       }));
    
                        th.SetApartmentState(ApartmentState.STA);
                        th.IsBackground = true;
                    }
                    if (th.ThreadState != ThreadState.Background || th.ThreadState == ThreadState.Unstarted || th.ThreadState != ThreadState.Running)
                    {
                        th.Start();
                    }
                }
                catch (Exception)
                {
                    th = new Thread(new ThreadStart(() =>
                    {
                        WinProgressBar win = new WinProgressBar();
                        win.Tips = msg;
                        win.Topmost = true;
                        win.Show();
                        System.Windows.Threading.Dispatcher.Run();
                    }));
    
                    th.SetApartmentState(ApartmentState.STA);
                    th.IsBackground = true;
                    th.Start();
                }
                Thread.Sleep(sleep);
            }
    
            /// <summary>
            /// 结束进度条
            /// </summary>
            public static void EndProgressBar()
            {
                if (th.IsAlive)
                    th.Abort();
            }

    呵呵,大家仔细分析下代码看有什么问题没有?而你们又有什么好的解决方案没有?

    在使用过程中发现一个很严重的问题,出现一个进度条窗体,系统内存就会增大一点,到最后要么卡死,要么自动退出.

         原来在多线程程序运行中,由用户取消操作是一种非常常见的场景,比如用户使用windows资源管理器在当前文件夹中搜索文件时,可以通过点击其它文件夹而取消搜索。
    中途停止一个线程的执行,通常用Thread.Abort方法,但这种方式会造成程序涉及的数据完整性受到破坏,线程所占用的一些系统资源(比如文件句柄等)也可能无法完成。比较合理的方式是外界提出"取消操作"的请求,然后由线程自身来决定如何处理这一请求。
    在设计多线程程序时,可设置一个用于接收外部取消消息的属性,然后在线程函数中分阶段地检测这一属性,每个阶段的检查点由软件开发者确定,并且决定线程如何优雅退出。

    所以对上述代码进行了改进

    首先设置一个公共变量:

     public static bool IsClsoeProgeressBar = true;

    然后在进度条窗体中实时读这个数据

    /// <summary>
        /// WinProgressBar.xaml 的交互逻辑
        /// </summary>
        public partial class WinProgressBar : Window
        {
            public string Tips { get; set; }
    
            DispatcherTimer timer;
            public WinProgressBar()
            {
                InitializeComponent();
    
                timer = new DispatcherTimer();
                timer.Interval = TimeSpan.FromMilliseconds(200);
                timer.Tick += new EventHandler(timer_Tick);
                timer.IsEnabled = true;
    
                this.Topmost = true;
                //this.Loaded += delegate
                //{
                //    if (this.Tips != "")
                //        tbTips.Text = this.Tips;
                //};
            }
    
            void timer_Tick(object sender, EventArgs e)
            {
                if (Common.CommWindow.IsClsoeProgeressBar)
                {
                    if (this.Visibility == Visibility.Visible)
                    {
                        progressBar1.IsIndeterminate = false;
                        this.Hide();
                    }
                }
                else
                {
                    if (this.Visibility != Visibility.Visible)
                    {
                        progressBar1.IsIndeterminate = true;
                        this.Show();
                    }           
                }
            }
        }

    然后在系统加载的时候,创建这个进度条窗体

       try
                {
                    Thread th = new Thread(new ThreadStart(() =>
                    {
                        WinProgressBar win = new WinProgressBar();
    
                       // win.Tips = msg;
                        win.Show();
    
                        System.Windows.Threading.Dispatcher.Run();
    
                    }));
    
                    th.SetApartmentState(ApartmentState.STA);
                    th.IsBackground = true;
    
                    th.Start();
                }
                catch (Exception)
                {
                }

    我们只需改变公共变量来控制窗体的显示与隐藏

             /// <summary>
            /// 显示进度条 线程
            /// </summary>
      
            public static void BeginProgressBar()
            {
                IsClsoeProgeressBar = false;
            }
    
            /// <summary>
            /// 结束进度条
            /// </summary>
            public static void EndProgressBar()
            {
                //cts.Cancel();
                IsClsoeProgeressBar = true;
            }

    还有一个比较好的方法:

     #region Methods
    
            /// <summary>
            /// 一个耗时的任务
            /// </summary>
            private void BigTask()
            {
                Thread.Sleep(3000);
            }
    
            private void button1_Click(object sender, RoutedEventArgs e)
            {
                MyProgressBar(BigTask);
            }
    
            private void MyProgressBar(Action mywork)
            {
                var worker = new BackgroundWorker();
                var window = new BusyWindow();
                worker.DoWork += (s, e2) => { mywork(); };
                worker.RunWorkerCompleted += (s, e2) =>
                {
                    MessageBox.Show("任务已经完成");
                    window.Close();
                };
                worker.RunWorkerAsync();
                window.Show();
            }
    
            #endregion

    这个由网友热情的沙漠提供,也很不错,大家可以参考下

  • 相关阅读:
    C++引用之引用的使用
    C++引用之声明方法
    C++const与指针
    C++默认参数值函数
    LeanCloud 调研报告
    [译] 为何流处理中局部状态是必要的
    Z-Stack
    Think twice before starting the adventure
    Multi-pattern string match using Aho-Corasick
    C pointer again …
  • 原文地址:https://www.cnblogs.com/smiler/p/3292845.html
Copyright © 2011-2022 走看看