【.NET Core】 多线程之(Thread)详解

【.NET Core】 多线程之(Thread)详解

文章目录

  • 【.NET Core】 多线程之(Thread)详解
    • 一、概述
    • 二、线程的创建和使用
      • 2.1 ThreadStart用于无返回值,无参数的方法
      • 2.2 ParameterizedThreadStart:用于带参数的方法
    • 三、线程的启动执行
    • 四、线程的挂起和恢复
      • 4.1 Suspend方法
      • 4.2 Resume方法
      • 4.3 该方法已废弃及替代方法
    • 五、中止线程(About)
    • 六、线程休眠(Sleep)
    • 七、线程等待(Join())
    • 八、线程让步(Yield())
    • 九、后台线程与前台线程
    • 十、`Thread`线程属性

在这里插入图片描述

一、概述

线程被定义为程序的执行路径。每个线程都定义了一个独特的控制流。如果您的应用程序涉及复杂且耗时的操作,那么设置不同的线程执行路径往往事半功倍,让每个线程执行特定的工作任务。

线程是一个轻量级进程。一个使用线程的常见实例是操作系统中并行编程的实现。使用线程节省了CPU周期的浪费,同时提高了应用程序的效率。本文将讲详细讲解Thread线程的用法。

Thread类位于System.Threading命名空间下,System.Threading命名空间提供一些可以进行多线程编程的类和接口。除同步线程活动和访问数据的类(Mutex,Monitor,InterlockedAutoResetEvent等)外,该命名空间还包含一个ThreadPool类(它允许用户使用系统提供的线程池)和一个Timer类(它在线程池的线程上执行回调方法)。

二、线程的创建和使用

Thread创建一个线程非常简单,只需要将其声明并为其提供线程起始点处的委托即可。创建新的线程时,需要使用Thread类,该类具有接受一个ThreadStart委托或ParameterizedThreadStart委托的构造函数。

2.1 ThreadStart用于无返回值,无参数的方法

public static void Main(string[] args)
{
    ThreadStart threadStart = new ThreadStart(MyThread);
    Console.WriteLine("创建一个线程");
    Thread thread= new Thread(threadStart);
    thread.Start();
}

private static void MyThread() 
{
     Console.WriteLine("This is Thread Method.");
     Thread.Sleep(2000);
     Console.WriteLine("Thread is over.....");
}

运行结果:

创建一个线程
This is Thread Method.
Thread is over.....

2.2 ParameterizedThreadStart:用于带参数的方法

public static void Main(string[] args)
{
   ParameterizedThreadStart threadStart = new ParameterizedThreadStart(MyThread);
   Console.WriteLine("创建一个线程");
   Thread thread= new Thread(threadStart);
   thread.Start(100);
}

private static void MyThread(object n) 
{
   Console.WriteLine("This is Thread Method.");
   Thread.Sleep(2000);
   Console.WriteLine("Thread is over.....");
   for (int i = 0; i <= (int)n; i += 1)
   {
      Console.WriteLine(i);
   }
}

三、线程的启动执行

线程创建后并不会理解执行,这是因为绝大多数系统不是一个实时的操作系统。在操作系统内部会实现特殊的算法进行线程之间的调度。在某个时刻它会决定当前运行哪个线程。看下面的示例:

static int id = 0;
public static void Main(string[] args)
{
   for (int i = 0; i < 10; i++)
   {
       id++;
       Thread thread = new Thread(() =>
       {
           Console.WriteLine(string.Format("{0}:{1}", Thread.CurrentThread.Name, id));
       });
       thread.Name = string.Format("Thread{0}", i);
       thread.IsBackground = true;
       thread.Start();
   }
}

执行结果:

Thread1:3
Thread0:3
Thread2:4
Thread3:5
Thread4:6
Thread5:7
Thread6:8
Thread7:9
Thread8:10
Thread9:10

使用同步参数创建线程

static int id = 0;
static void Main(string[] args)
{
   for (int i = 0; i < 5; i++, id++)
   {
       NewMethod(i, id);
   }
   Console.ReadLine();
}
private static void NewMethod(int i, int realTimeId)
{
    Thread thread = new Thread(() =>
    {
        Console.WriteLine(string.Format("{0}:{1}", Thread.CurrentThread.Name, realTimeId));
    });
    thread.Name = string.Format("Thread{0}", i);
    thread.IsBackground = true;
    thread.Start();
}

运行结果:

Thread0:0
Thread1:1
Thread2:2
Thread3:3
Thread4:4

四、线程的挂起和恢复

创建完一个线程并启动后,还可以挂起、恢复、休眠或终止线程。线程的挂起与恢复可以通过调用Thread类的Suspend方法和Resume方法实现。

4.1 Suspend方法

该方法用来挂起线程,如果线程已挂起,则不起作用。调用Suspend方法挂起线程时,.NET允许挂起的线程再执行几个指令,目的是为了到达.NET认为线程可以安全挂起的状态。

4.2 Resume方法

该方法用来继续已挂起的线程。通过Resume方法来恢复被暂停的线程时,无论调用多少次Suspend方法,调用Resume方法均会使另一个线程脱落挂起状态。

通过Resume()方法来恢复被暂停的线程时,无论调用了多少次Suspend()方法,调用Resume()方法均会使另一个线程脱离挂起状态,并导致该线程继续执行。

4.3 该方法已废弃及替代方法

Thread.Suspend()Thread.Resume()这两个方法已过时,需要使用AutoResetEventEventWaitHandle代替。

五、中止线程(About)

线程被中止,就停止运行,是无法恢复的,因为操作系统会删除被中止线程所有数据。

static void Main(string[] args)
{
     Thread myThread;                       
     myThread = new Thread(new ThreadStart(createThread));
     myThread.Start();
     myThread.Abort();
}
private static void createThread()
{
    for (long i = 0; i < 1000000000; i++)
    { 
        Console.WriteLine($"workThread-->i={i}");
    }
}

线程中止和线程挂起是一样结果,中止工作线程后,线程将不会在被执行,如果强制运行程序,将出现运行时错误。

六、线程休眠(Sleep)

使线程休眠使用Sleep()方法,此方法可以让线程休眠一定的时间,示例代码如下:

Thread workThread = new Thread(new ThreadStart(createThread));
workThread.Start();
Thread.Sleep(10000);//修改

Sleep()会让线程暂停一段时间后接着运行,休眠线程不需要手动恢复,到指定的时间后会自动执行。

七、线程等待(Join())

如果后续的处理依赖于另一个已终止的线程,可调用Join()方法,等待线程中止。

Thread workThread = new Thread(new ThreadStart(createThread));
workThread.Start();
workThread.Join();

工作线程调用了Jion()方法,需待工作线程中止后,主线程才会被执行。Jion()的其他重载方法可以指定等待的时间期限,超过了时间期限,程序会继续执行。

八、线程让步(Yield())

Yield 的中文翻译为 “屈服,让步”,这里意思是主动放弃当前线程的时间片,并让操作系统调度其它就绪态的线程使用一个时间片。但是如果调用 Yield,只是把当前线程放入到就绪队列中,而不是阻塞队列。如果没有找到其它就绪态的线程,则当前线程继续运行。Yield可以让低于当前优先级的线程得以运行,调用者可以通过返回值判断是否成功调度了其它线程。注意,Yield只能调度运行在当前CPU上的线程。

九、后台线程与前台线程

线程是后台线程或前台线程。后台线程与前台线程相同,只不过后台线程不会阻止进程终止。一旦属于进程的所有前台线程都终止,公共语言进行时将结束该进程。所有剩余的后台线程将停止,并且无法完成。

默认情况下,以下线程是前台线程:

  • 主线程(或主应用程序线程)。
  • 通过调用类构造函数创建Thread的所有线程都是前台线程。

默认情况下,以下线程在后台执行:

  • 线程池线程,由运行时维护的工作线程池。可以使用类配置线程池并计划线程池线程ThreadPool上工作。
  • 从非托管代码进入托管执行环境的所有线程。

十、Thread线程属性

  • CurrentThread:获取当前正在运行的线程。
  • IsAlive :获取指示当前线程的执行状态的值。
  • IsBackgroud:获取或设置一个值,该值指示某个线程是否为后台线程。
  • IsThreadPoolThread:获取指示线是否属于托管线程池的值。
  • Name:获取或设置线程的名称。
  • Priority:获取或设置线程调度优先级的值。
  • ThreadState:获取一个值,该值包含当前线程的状态。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/332478.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

REVIT二次开发生成三维轴网

步骤1 确定轴网 步骤2 生成3D轴网 using System; using System.Collections.Generic; using System.Linq; using System.Text;

C#winform上位机开发学习笔记2-串口助手的中文支持功能添加

分为两步&#xff1a; 1.串口接收支持中文显示 1.1.在软件初始化时写入此代码以支持汉字显示 //串口接收支持中文显示serialPort1.Encoding Encoding.GetEncoding("GB2312"); //串口1的解码支持GB2312汉字 2.串口发送支持中文输出 //支持中文输出Encoding Chine…

文心一言使用分享

ChatGPT 和文心一言哪个更好用&#xff1f; 一个直接可以用&#xff0c;一个还需要借助一些工具&#xff0c;还有可能账号会消失…… 没有可比性。 通用大模型用于特定功能的时候需要一些引导技巧。 import math import time def calculate_coordinate(c, d, e, f, g, h,…

【Origin绘图系列第3棒】箱型图:

Origin绘制箱型图 箱型图&#xff08;Boxplots&#xff09;案例1&#xff1a;基本绘制参考 箱型图&#xff08;Boxplots&#xff09; 案例1&#xff1a;基本绘制 选择箱型图后界面如下&#xff1a; 设置分组&#xff0c;如下设置&#xff0c; 图形如下所示&#xff1a; 根…

防火墙部署安全区域实验

目录 实验拓扑web登防火墙配置对象配置安全策略配置NAT配置安全策略测试抓包测试 实验拓扑 安全区域如下图&#xff1a; web登防火墙 按接口划分各区域&#xff0c;GE1/0/1为trust区域&#xff0c;内网是信任区域。 配置如下&#xff0c;可起别名方便操作&#xff0c;区域为t…

C++后端笔记

C后端笔记 资源整理一、高级语言程序设计1.1 进制1.2 程序结构基本知识1.3 数据类型ASCII码命名规则变量间的赋值浮点型变量的作用字符变量常变量 const运算符 二、高级语言程序设计&#xff08;荣&#xff09; 资源整理 C后端开发学习路线及推荐学习时间 C基础知识大全 C那…

ICCV2023 | PTUnifier+:通过Soft Prompts(软提示)统一医学视觉语言预训练

论文标题&#xff1a;Towards Unifying Medical Vision-and-Language Pre-training via Soft Prompts 代码&#xff1a;https://github.com/zhjohnchan/ptunifier Fusion-encoder type和Dual-encoder type。前者在多模态任务中具有优势&#xff0c;因为模态之间有充分的相互…

Docker部署Flask项目

Docker部署Flask项目 一、准备项目代码二、编写Dockerfile三、服务器部署 一、准备项目代码 这里写了一个简单的Flask的demo&#xff0c;源代码如下&#xff1a; from flask import Flaskapp Flask(__name__)app.route("/") def index():return "<h1 styl…

解决Windows下VSCode控制台乱码问题

我们在Windows使用VSCode编写C/C程序时&#xff0c;如果代码中的中文字符串使用的是UTF8编码&#xff0c;且代码内没有设置控制台的输出编码&#xff0c;或者编译时没有指定运行时编码&#xff08;GCC可以在编译时使用-fexec-charsetGBK来指定运行时的字符串编码&#xff1b;cl…

解决Uniapp插件市场试用原生插件项目 没有MD5签名安卓无法自定基座打包的情况

Uniapp插件市场中&#xff0c;有些插件是原生插件&#xff0c;必须使用自定义基座才能打包。但是传统keytool命令&#xff0c;已经无法看到安卓证书的MD5签名。现采用Android Studio查询signingReport的办法获取证书的MD5签名&#xff0c;并对插件的示例项目进行打包运行。一、…

c++基础3

一 、构造函数的初始化列表 可以指定成员对象的初始化方式 构造函数的初始化列表是在 C 中用于初始化成员变量的一种机制。它在构造函数的参数列表之后&#xff0c;构造函数的函数体之前使用&#xff0c;并使用冒号 : 分隔。初始化列表可以用于给成员变量赋初值&#xff0c;而不…

ycsb压测mongodb

下载解压 https://github.com/brianfrankcooper/YCSB/releases/download/0.17.0/ycsb-mongodb-binding-0.17.0.tar.gz tar -zxvf ycsb-mongodb-binding-0.17.0.tar.gzycsb提前已经在workload文件夹下准备好了几个压测场景分别对应workload[a:f] workloads/workloada 样例 …

Docker-harbor私有仓库部署与管理

目录 实验前准备部署Docker-compose服务部署Harbor服务修改配置文件创建项目 客户端测试修改客户端配置测试 维护管理Harbor创建项目创建用户查看日志修改Harbor.cfg配置文件补充 实验前准备 Harbor服务器&#xff1a;192.168.188.11 docker-ce、docker-compose、harbor-offli…

【计算机网络】OSI七层模型与TCP/IP四层模型的对应与各层介绍

1 OSI七层模型与TCP/IP四层模型对应 2 OSI七层模型介绍 OSI&#xff08;Open Systems Interconnection&#xff09;模型是一个由国际标准化组织&#xff08;ISO&#xff09;定义的七层网络体系结构&#xff0c;用于描述计算机网络中的通信协议。每一层都有特定的功能&#xff…

C#,入门教程(38)——大型工程软件中类(class)修饰词partial的使用方法

上一篇&#xff1a; C#&#xff0c;入门教程(37)——优秀程序员的修炼之道https://blog.csdn.net/beijinghorn/article/details/125011644 一、大型&#xff08;工程应用&#xff09;软件倚重 partial 先说说大型&#xff08;工程应用&#xff09;软件对源代码的文件及函数“…

2023 China DevOpsDays(DOD) DXCon 国际数字化转型与创新管理企业峰会:核心内容与学习收获(附大会核心PPT下载)

随着科技的飞速发展&#xff0c;数字化转型已成为企业持续发展的必经之路。2023年的China DevOpsDays & DXCon国际数字化转型与创新管理企业峰会&#xff0c;汇集了业界顶尖的专家、学者和企业领袖&#xff0c;共同探讨数字化转型的最新趋势和实践。本文将深入剖析大会的核…

多功能隐写融合

最近尝试了一个融合了多功能隐写的项目&#xff0c;参考了一些现有的资料和相关的开源库&#xff0c;并最终集成到了可视化UI当中。这篇文章讲述了实现的几项隐写技术的原理以及最终呈现的效果&#xff0c;后续会在“隐私保护”以及“ui”的专栏中继续更新详细的ui设计思路以及…

Asp .Net Core 系列:集成 Ocelot+Nacos+Swagger+Cors实现网关、服务注册、服务发现

文章目录 简介什么是 Ocelot ?什么是 Nacos ?什么是 Swagger ?什么是 Cors ? Asp .Net Core 集成 Ocelot网关集成 Nacos下游配置 Nacos配置跨域&#xff08;Cors&#xff09;网关和微服务中配置Swagger效果 简介 什么是 Ocelot ? Ocelot是一个开源的ASP.NET Core微服务网…

某C2鸡肋漏洞分析:你的CS安全吗?

CobaltStrike是一个知名的红队命令与控制框架&#xff0c;采用Beacon <-> TeamServer <-> Client架构。TeamServer存在受限路径穿越写文件与反序列化漏洞&#xff0c;可以被认证后客户端恶意利用。Client存在反序列化漏洞&#xff0c;可以被RogueCS攻击。 山寨威胁…

PyTorch各种损失函数解析:深度学习模型优化的关键(2)

目录 详解pytorch中各种Loss functions mse_loss 用途 用法 使用技巧 注意事项 参数 数学理论公式 代码演示 margin_ranking_loss 用途 用法 使用技巧 注意事项 参数 数学理论公式 代码演示 multilabel_margin_loss 用途 用法 使用技巧 注意事项 参数 …