当前位置: 移动技术网 > IT编程>开发语言>c# > c# 面试必备线程基础知识点

c# 面试必备线程基础知识点

2020年11月21日  | 移动技术网IT编程  | 我要评论
线程的知识太多,知识点有深有浅,往深的研究会涉及操作系统、cpu、内存,往浅了说就是一些语法。没有一定的知识积累,很难把线程的知识写得全面,当然我也没有这个能力。所以想到一个点写一个点,尽量总结一些有

线程的知识太多,知识点有深有浅,往深的研究会涉及操作系统、cpu、内存,往浅了说就是一些语法。没有一定的知识积累,很难把线程的知识写得全面,当然我也没有这个能力。所以想到一个点写一个点,尽量总结一些有用的知识点。线程是个大话题,这个系列可能会有好几遍关于线程的,先从基础的开始,热热身。

一些基础概念

线程(thread)是操作系统能够进行运算调度的最小单位。它是进程中的实际运作单位,一个进程中可以启动多个线程,每个线程可以并行执行不同的任务。严格意义上来说,同一时间可以并行运行的线程数取决于 cpu 的核数。

根据线程运行模式,可以把线程分为前台线程、后台线程和守护(daemon)线程:

  • 前台线程:主程序必须等待线程执行完毕后才可退出程序。c# 中的 thread 默认为前台线程,也可以设置为后台线程。
  • 后台线程:主程序执行完毕立即跟随退出,不管线程是否执行完毕。c# 的 threadpool 管理的线程默认为后台线程。
  • 守护线程:守护线程拥有自动结束自己生命周期的特点,它通常被用来执行一些后台任务。

每次开启一个新的线程都要消耗一定的内存,即使线程什么也不做,也会至少消耗 1m 左右的内存。

多线程并行(parallelism)和并发(concurrency)的区别:

  • 并行:同一时刻有多条指令在多个处理器上同时执行,无论从宏观还是微观上都是同时发生的。
  • 并发:是指在同一时间段内,宏观上看多个指令看起来是同时执行,微观上看是多个指令进程在快速的切换执行,同一时刻可能只有一条指令被执行。

ps:以上概念来源 google 的多个搜索结果,稍加整理。

thread、threadpool 和 task

对 c# 开发者来说,不可不理解清楚 thread、threadpool 和 task 这三个概念。这也是面试频率很高的话题,在 stackoverflow 可以找到有很多不错的回答,我总结整理了一下。

thread

thread 是一个实际的操作系统级别的线程(os 线程),有自己的栈和内核资源。thread 允许最高程度的控制,你可以 abort、suspend 或 resume 一个线程,你还可以监听它的状态,设置它的堆栈大小和 culture 等属性。thread 的开销成本很高,你的每一个线程都会为它的堆栈消耗相对较多的内存,并且在线程之间的处理器上下文切换时会增加额外的 cpu 开销。

threadpool

threadpool(线程池)是一堆线程的包装器,由 clr 维护。你对线程池中的线程没有任何控制权,你甚至无法知道线程池什么时候开始执行你提交的任务,你只能控制线程池的大小。简单来说,线程池调用线程的机制是,它首先调用已创建的空闲线程来执行你的任务,如果当前没有空闲线程,可能会创建新线程,也可能会等待。

使用 threadpool 可以避免创建太多线程的开销。但是,如果你向 threadpool 提交了太多长时间运行的任务,它可能会被填满,这时你提交的后面的任务可能最终会等待前面的长时间运行的任务执行完成。此外,线程池没有提供任何方法来检测一个工作任务何时完成(不像 thread.join()),也没有方法来获取结果。因此,threadpool 最好用于调用者不需要结果的短时操作。

task

task 是 tpl(task parallel library)提供一个类,它在 thread 和 theadpool 之间提供了两全其美的解决方案。和 threadpool 一样,task 并不创建自己的os 线程。相反,task 是由 taskscheduler 调度器执行的,默认的调度器只是在 threadpool 上运行。

与 threadpool 不同的是,task 还允许你知道它完成的时间,并获取返回一个结果。你可以在现有的 task 上调用 continuewith(),使它在任务完成后运行更多的代码(如果它已经完成,就会立即运行回调)。

你也可以通过调用 wait() 来同步等待一个任务的完成(或者,通过获取它的 result 属性)。与 thread.join() 一样,这将阻塞调用线程,直到任务完成。通常不建议同步等待任务执行完成,它使调用线程无法进行任何其他工作。如果当前线程要等待其它线程任务执行完成,建议使用 async/await 异步等待,这样当前线程可以空闲出来去处理其它任务,比如在 await task.delay() 时,并不占用线程资源。

由于任务仍然在 threadpool 上运行,因此不应该将其用于长时任务的执行,因为它们会填满线程池并阻塞新的工作任务。相反,task 提供了一个 longrunning 选项,它将告诉 taskscheduler 启用一个新的线程,而不是在 threadpool 上运行。

所有较新的上层多线程 api,包括 parallel.foreach()、plinq、async/await 等,都是建立在 task 上的。

thread 和 task 简单示例

下面通过一个简单示例演示 thread 和 task 的使用,注意他们是如何创建、传参、执行和等待执行完成的。

运行效果:

注意到,相比之下 task 比 thread 好用得多,加上前文 task 和 thread 的对比,对我们编码的指导意义是:大多数情况我们应该使用 task,而不要直接使用 thread,除非你明确知道你需要一个独立的线程来执行一个长耗时的任务。

小结

本篇内容很基础,整理了 c# 线程编程有关的重要概念,简单演示了 thread 和 task 的使用。thread 和 task 是高频面试话题,尤其是 thread 和 task 的区别,thread 更底层,task 更抽象,回答好这类面试题的关键点在 threadpool。

作者:精致码农

出处:

以上就是c# 面试必备线程基础知识点的详细内容,更多关于c# 线程基础的资料请关注移动技术网其它相关文章!

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

  • c# 面试必备线程基础知识点

    c# 面试必备线程基础知识点

    线程的知识太多,知识点有深有浅,往深的研究会涉及操作系统、cpu、内存,往浅了说就是一些语法。没有一定的知识积累,很难把线程的知识写得全面,当然我也没有这个能力... [阅读全文]
  • C#使用System.Net邮件发送功能踩过的坑

    C#使用System.Net邮件发送功能踩过的坑

    1.eazyemail邮件发送类库net 类库自带了邮件发送功能。笔者对该类库,从使用的角度进行了二次封装,nuget上可搜索eazyemail,注入容器时通过... [阅读全文]
  • C#基于Modbus三种CRC16校验方法的性能对比

    C#基于Modbus三种CRC16校验方法的性能对比

    1.背景介绍主要应用场景在物联网中,底端设备注册报文的上报,需要对报文的有效载荷(data)进行crc16的复验,验证与设备端的crc校验是否相等,如果相等,报... [阅读全文]
  • 深入谈谈C#9新特性的实际运用

    前言你一定会好奇:“老周,你去哪开飞机了?这么久没写博客了。”老周:“我买不起飞机,开了个铁矿,挖了一年半的石头。谁知铁矿垮了,压死了几条蜈蚣,什么也没挖着。”... [阅读全文]
  • C# 泛型集合的自定义类型排序的实现

    C# 泛型集合的自定义类型排序的实现

    一、泛型集合list<t>排序经sort方法之后,采用了升序的方式进行排列的。二、对自定义类型进行排序定义一个普通类:接下来,将定义的person实... [阅读全文]
  • C#开发中常用的加密解密方法汇总

    相信很多人在开发过程中经常会遇到需要对一些重要的信息进行加密处理,今天给大家分享我个人总结的一些加密算法:常见的加密方式分为可逆和不可逆两种方式可逆:rsa,a... [阅读全文]
  • C# 如何添加错误日志信息

    系统日志系统日志包含了由windows系统组件记录的事件。例如,在启动期间装入驱动程序或其他系统组件失败被记录到系统日志。要查看系统日志: 打开命令提示符。 ... [阅读全文]
  • 关于C#委托三种调用的分享使用

    关于C#委托三种调用的分享使用

    一、同步调用1、同步调用会按照代码顺序来执行2、同步调用会阻塞线程,如果是要调用一项繁重的工作(如大量io操作),可能会让程序停顿很长时间,造成糟糕的用户体验,... [阅读全文]
  • 用c# 自动更新程序

    作者:冰封一夏出处:hzhcontrols官网:首先看获取和更新的接口更新程序program.cs更新程序界面定义服务端接口,你可以用任意接口都行,我这里用we... [阅读全文]
  • c# 生成二维码的示例

    二维码是越来越流行了,很多地方都有可能是使用到。如果是静态的二维码还是比较好处理的,通过在线工具就可以直接生成一张二维码图片,比如:草料二维码。但有的时候是需要... [阅读全文]
验证码:
移动技术网