C#.Net前台线程与后台线程的区别

简介: .Net的公用语言运行时(Common Language Runtime,CLR)能区分两种不同类型的线程:前台线程和后台线程。这两者的区别就是:应用程序必须运行完所有的前台线程才可以退出;而对于后台线程,应用程序则可以不考虑其是否已经运行完毕而直接退出,所有的后台线程在应用程序退出时都会自动结束。
.Net的公用语言运行时(Common Language Runtime,CLR)能区分两种不同类型的线程:前台线程和后台线程。这两者的区别就是:应用程序必须运行完所有的前台线程才可以退出;而对于后台线程,应用程序则可以不考虑其是否已经运行完毕而直接退出,所有的后台线程在应用程序退出时都会自动结束。

  .Net环境使用Thread建立的线程默认情况下是前台线程,即线程属性IsBackground= false,在进程中,只要有一个前台线程未退出, 进程就不会终止。主线程就是一个前台线程。
而后台线程不管线程是否结束,只要所有的前台线程都退出(包括正常退出和异常退出)后,进程就会自动终止。一般后台线程用于处理时间较短的任务,如在一个Web服务器中可以利用后台线程来处理客户端发过来的请求信息。而前台线程一般用于处理需要长时间等待的任务,如在Web服务器中的监听客户端请求的程序,或是定时对某些系统资源进行扫描的程序。
 
需要明白的概念性问题:
线程是寄托在进程上的,进程都结束了,线程也就不复存在了!
只要有一个前台线程未退出,进程就不会终止!即说的就是程序不会关闭!(即在资源管理器中可以看到进程未结束。)

测试代码:

 public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        /// <summary>
        /// 弹出窗体Form2
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button1_Click(object sender, EventArgs e)
        {
            Form2 _frm2 = new Form2();
            _frm2.Show();
        }

       
    }

 public partial class Form2 : Form
    {
        public Form2()
        {
            InitializeComponent();
        }

        Thread _Thread = null;

        private void Form2_Load(object sender, EventArgs e)
        {
            _Thread = new Thread(() => { while (true) { /*制造无限循环,等待用户关闭线程*/ } });

            _Thread.IsBackground = false;//false:设置为前台线程,系统默认为前台线程。
            //_Thread.IsBackground = true;//true:后台线程 

            _Thread.Start();
        }
    }
测试结果:注意在Debug模式下,看看vs是否关闭!!! 或者通过Realse模式,观看【资源管理器】中的应用程序线程是否关闭。

如设为前台线程,即IsBackground = False,关闭窗体2,在关闭窗体1,虽然窗体1关闭了,然而应用程序还是停留在资源管理器中。
如设为后台线程,即IsBackground = True, 关闭窗体1后,应用程序立刻从资源管理器中结束。

补充说明1:

private void Form2_Load(object sender, EventArgs e)
        {
            Thread _Thread = new Thread(() => 
            { 
                while (true) 
                { 
                    /*制造无限循环,等待用户关闭线程*/
                    this.BeginInvoke(new MethodInvoker(() => { this.Text = ""; }));
                    Thread.Sleep(3000);
                }
            });
            _Thread.IsBackground = true;
            _Thread.Start();  
        }

        //如果_Thread.IsBackground = true;为后台线程
        //场景1:打开Form1窗口后,立刻点击Button按钮弹出Form2窗口,此时Form2窗口中的线程已经启动了,
        //操作1:立刻关闭Form1窗口(主线程),此时不会出现任何的错误消息,同时资源管理器中ThreadDemo进程已经退出。
        //操作2:此时关闭Form2窗口,大概等待3秒后,程序会出现崩溃的消息,其原因是因为Form2窗口中的线程有操作UI的代码,而此时的Form2窗口已经关闭了(窗体对象已经释放)

    }
View Code

补充说明2:

 private void Form2_Load(object sender, EventArgs e)
        {
            Thread _Thread = new Thread(() =>
            {
                while (true)
                {
                    //if (this.IsHandleCreated)
                    /*制造无限循环,模型耗时的任务*/
                    this.BeginInvoke(new MethodInvoker(() => { this.Text = ""; }));
                    Thread.Sleep(3000);
                }
            });
            _Thread.IsBackground = false;
            _Thread.Start();
        }

        //如果_Thread.IsBackground = false;为前台线程
        //场景1:打开Form1窗口后,立刻点击Button按钮弹出Form2窗口,此时Form2窗口中的线程已经启动了,
        //操作(1):立刻关闭Form1窗口(主线程),大概等待3秒后,程序会出现崩溃的消息。
        //操作(2):此时关闭Form2窗口,现象和上面一样。
        //,其原因是因为Form2窗口中的线程有操作UI的代码而此时的Form2窗口已经关闭了(窗体对象已经释放)

        //此处如果程序不崩溃的话(if (this.IsHandleCreated)加上这句的话),则上面的两个操作可以在资源管理器中看到线程ThreadDemo一直存在进程中。

    }
View Code

 

如果想让上面的程序不崩溃可以使用:

  if (this.IsHandleCreated)
                    {
                        /*制造无限循环,模型耗时的任务*/
                        this.BeginInvoke(new MethodInvoker(() => { this.Text = ""; }));
                    }
View Code

 if (!this.IsDisposed ) 也是可以的。判断当前窗体是否已经释放。(备注:释放的窗体是不能操作界面UI元素的。)

目录
相关文章
|
3月前
|
数据采集 XML JavaScript
C# 中 ScrapySharp 的多线程下载策略
C# 中 ScrapySharp 的多线程下载策略
|
10天前
|
Java 物联网 编译器
C#一分钟浅谈:.NET Core 与 .NET 5 区别
本文对比了 .NET Core 和 .NET 5,从历史背景、主要区别、常见问题及易错点等方面进行了详细分析。.NET Core 侧重跨平台支持和高性能,而 .NET 5 在此基础上统一了 .NET 生态系统,增加了更多新特性和优化。开发者可根据具体需求选择合适的版本。
32 7
|
7天前
|
开发框架 网络协议 .NET
C#/.NET/.NET Core优秀项目和框架2024年10月简报
C#/.NET/.NET Core优秀项目和框架2024年10月简报
|
1月前
|
监控 网络安全 调度
Quartz.Net整合NetCore3.1,部署到IIS服务器上后台定时Job不被调度的解决方案
解决Quartz.NET在.NET Core 3.1应用中部署到IIS服务器上不被调度的问题,通常需要综合考虑应用配置、IIS设置、日志分析等多个方面。采用上述策略,结合细致的测试和监控,可以有效地提高定时任务的稳定性和可靠性。在实施任何更改后,务必进行充分的测试,以验证问题是否得到解决,并监控生产环境的表现,确保长期稳定性。
47 1
|
13天前
|
开发框架 安全 .NET
C#面:Server.UrlEncode、HttpUtility.UrlDecode的区别
通过上述详细的解释和实例分析,相信大家对 `Server.UrlEncode` 和 `HttpUtility.UrlDecode` 的区别有了更深刻的理解,并能在实际开发中灵活运用。
23 0
|
1月前
|
网络协议 网络性能优化 C#
C# 一分钟浅谈:UDP 与 TCP 协议区别
【10月更文挑战第8天】在网络编程中,传输层协议的选择对应用程序的性能和可靠性至关重要。本文介绍了 TCP 和 UDP 两种常用协议的基础概念、区别及应用场景,并通过 C# 代码示例详细说明了如何处理常见的问题和易错点。TCP 适用于需要可靠传输和顺序保证的场景,而 UDP 适用于对延迟敏感且可以容忍一定数据丢失的实时应用。
34 1
|
1月前
|
开发框架 前端开发 API
C#/.NET/.NET Core优秀项目和框架2024年9月简报
C#/.NET/.NET Core优秀项目和框架2024年9月简报
|
1月前
|
C# 开发者
【捞底干货】C#中equals和==运算符的区别
【捞底干货】C#中equals和==运算符的区别
28 1
|
1月前
|
边缘计算 开发框架 人工智能
C#/.NET/.NET Core优秀项目和框架2024年8月简报
C#/.NET/.NET Core优秀项目和框架2024年8月简报
|
2月前
|
C# 索引
C# 一分钟浅谈:接口与抽象类的区别及使用
【9月更文挑战第2天】本文详细对比了面向对象编程中接口与抽象类的概念及区别。接口定义了行为规范,强制实现类提供具体实现;抽象类则既能定义抽象方法也能提供具体实现。文章通过具体示例介绍了如何使用接口和抽象类,并探讨了其实现方式、继承限制及实例化差异。最后总结了选择接口或抽象类应基于具体设计需求。掌握这两者有助于编写高质量的面向对象程序。
117 5