使用线程新建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

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

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏数据和云

高频错误:ORA-01555深入剖析

黄玮(Fuyuncat) 资深Oracle DBA,个人网站www.HelloDBA.com,致力于数据库底层技术的研究,其作品获得广大同行的高度评价. ORA...

2818
来自专栏mini188

学习笔记:delphi多线程知识

最近一直在温习旧的知识,刚好学习了一下Java的线程安全方面的知识,今天想起之前一直做的Delphi开发,所以还是有必要温习一下,看看这些不同的编程语言有什么不...

2016
来自专栏张善友的专栏

ASP.NET MVC扩展库

很多同学都读过这篇文章吧 ASP.NET MVC中你必须知道的13个扩展点,今天给大家介绍一个ASP.NET MVC的扩展库,主要就是针对这些扩展点进行。这个项...

1747
来自专栏Leetcode名企之路

【java】CountDownLatch运用场景(1)

CountDownLatch也叫闭锁,使得一(多)个主线程必须等待其他线程完成操作后再执行。 使用的方式是:CountDownLatch内部维护一个计数器,主线...

601
来自专栏NetCore

一个让人遗忘的角落—Exception(二)

在上一篇中"一个被人遗忘的角落--Exception(一)"中,跟大家简单介绍了一下Exception,也使大家充分的了解了Exception管理在一个项目中的...

1739
来自专栏Albert陈凯

Spark详解07广播变量BroadcastBroadcast

Broadcast 顾名思义,broadcast 就是将数据从一个节点发送到其他各个节点上去。这样的场景很多,比如 driver 上有一张表,其他节点上运行的 ...

3296
来自专栏大内老A

WCF技术剖析之三十三:你是否了解WCF事务框架体系内部的工作机制?[上篇]

WCF事务编程主要涉及到这么三个方面:通过服务(操作)契约确定TransactionFlow的策略;通过事务绑定实现事务流转;通过服务操作行为控制事务的自动登记...

1948
来自专栏Jack的Android之旅

android蓝牙4.0的知识要点

这次主要讲解蓝牙4.0的基本要点,作为自己的备忘录记录下来吧。首先普及一下蓝牙4.0基于Gatt协议来实现。而蓝牙4.0以下的是传统蓝牙,基于socket方式来...

744
来自专栏王亚昌的专栏

epoll使用详解

 Linux平台上传统的I/O复用模型有select和poll模型,但二者在解决大量并发请示时却表现不佳。与select/poll相比,epoll的优点体现在以...

631
来自专栏dotnet core相关

Define Constraints That Are Minimal and Sufficient 设定不多不少的约束

主管不在,然后暂时没什么任务,把第二节看了,然后整理一下,下班之后就能继续去打球了。

512

扫描关注云+社区