13、c#线程

news2026/5/8 5:51:38
1 简介1.1 概念进程正在运行的程序 线程正在运行的程序中 正在执行的代码块 ​ 比喻进程是正在开工的工厂 线程是正在运行的流水线 一个进程中只要有一个线程单线程 一个进程中可以有多个线程同时执行多线程1.2 线程和进程的区别在 C# 及操作系统层面进程Process和线程Thread是实现程序并发执行的核心概念二者既有联系又有本质区别具体如下进程Process进程是操作系统资源分配和管理的基本单位本质上是一个 “正在运行的程序实例”。当你双击一个.exe文件如notepad.exe操作系统会为其创建一个进程分配独立的内存空间、CPU 时间片、文件句柄、网络端口等系统资源。进程拥有自己的 “地址空间”不同进程的内存相互隔离默认情况下无法直接访问对方内存确保程序运行的安全性。线程Thread线程是进程内的执行单元是CPU 调度的基本单位。一个进程至少包含一个线程称为 “主线程”也可以创建多个线程“多线程”所有线程共享所属进程的内存空间和资源如变量、文件句柄、数据库连接等。线程的核心作用是 “并发执行任务”例如一个音乐播放器进程中可能有一个线程负责播放音乐另一个线程负责接收用户输入避免操作互相阻塞。本质区别对比维度进程Process线程Thread资源分配拥有独立的内存空间、文件句柄等资源资源消耗大共享所属进程的资源仅需少量独立栈空间默认约 1MB资源消耗小独立性高度独立一个进程崩溃通常不影响其他进程依赖于进程同一进程内的线程共享资源一个线程崩溃可能导致整个进程崩溃通信方式需通过操作系统提供的 IPC 机制如管道、Socket、共享内存等实现复杂可通过共享变量直接通信但需同步机制如lock、Monitor避免数据冲突创建 / 销毁开销开销大需分配独立资源开销小仅需初始化少量私有数据调度单位操作系统分配资源的单位CPU 调度执行的单位CPU 直接切换线程执行数量限制系统中进程数量较少受内存等资源限制一个进程可包含大量线程但过多会导致调度效率下降总结简单来说进程是 “独立的程序容器”线程是 “容器内的执行流”。进程提供资源隔离线程提供执行效率。2 创建线程的方式12.1 Thread通过创建Thread的实例对应一个线程一个线程对应一个代码块 线程任务此代码块2.2 多线程初次体验static Random Random new Random(); /* 创建线程方式1Thread实例 ​ ​ 注意Mian方法是程序的入口 c#默认为为Main方法自动创建线程 -- main线程 来加载执行main方法中代码 多线程的原理在时间轮片内(时间单位) cup只能执行一个线程时间轮片到期 在所有线程之间进行随机切换 ​ ​ Thread方法 1 构造方法 Thread(ThreadStart); 2 启动线程 cup才会为此线程分配资源 void Start() 3 线程休眠 void Sleep(long)::参数是毫秒值 ​ 属性 1 获取当前线程对象 static Thread CurrentThread; 2 获取和设置线程名字string Name ​ */ static void Main(string[] args) { ​ //2 创建Thread对象 并管理线程任务 Thread t1 new Thread(ThreadStart01); t1.Name 线程1号; //3 开启线程 t1.Start();//此时有两个线程Mian线程 t1线程 ​ ​ //获取当前线程 Thread tt Thread.CurrentThread; tt.Name main线程:::; ​ //随机a-z字母 for (int i 0; i 20; i) { int n (char)Random.Next(a, z1); Console.WriteLine(Thread.CurrentThread.Name:::i :::::::::::: n); Thread.Sleep(100); } } //1:定义方法实现ThreadStart委托封装线程任务 public static void ThreadStart01() { //随机0-9数字 for (int i 0; i 20; i) { int n Random.Next(0,10); ​ Console.WriteLine(Thread.CurrentThread.Name:::i::::n); Thread.Sleep(100); ​ } }2.3 Thread类的成员Thread方法 1 构造方法 Thread(ThreadStart); 2 启动线程 cup才会为此线程分配资源 void Start() 3 线程休眠 void Sleep(long)::参数是毫秒值 ​ 属性 1 获取当前线程对象 static Thread CurrentThread; 2 获取和设置线程名字string Name2.4 案例2/* 定义三个线程 打印30字符一个打印大写字母 一个打印小写字母 一个打印数字 */ /* 线程扩展 01 线程等待 void Join(); 在a线程任务中调用b.Join()::a线程会等待 b线程执行完毕 a线程才能继续执行 */ static Random random new Random(); static void Main(string[] args) { //2 创建thread对象 并管理线程任务 Thread t1 new Thread(renWu1); t1.Name 打印数字; Thread ta new Thread(renWua); ta.Name 打印小写字母; Thread tA new Thread(renWuA); tA.Name 打印大写字母; //3 启动线程 t1.Start(); ta.Start(); tA.Start(); ​ t1.Join();//等待t1执行完毕 ta.Join();//等待ta执行完毕 tA.Join();//等待tA执行完毕 Console.WriteLine(所有线程执行完毕); ​ } //1 创建ThreadStart委托的实现方法 static void renWu1() { for (int i 0; i 30; i) { char c (char)(random.Next(0, 9 1)); Console.WriteLine(Thread.CurrentThread.Name:::i:::::c); Thread.Sleep(100); } } static void renWua() { for (int i 0; i 30; i) { char c (char)(random.Next(a, z 1)); Console.WriteLine(Thread.CurrentThread.Name ::::::::: i ::::: c); Thread.Sleep(100); } } static void renWuA() { for (int i 0; i 30; i) { char c (char)(random.Next(A, Z 1)); Console.WriteLine(Thread.CurrentThread.Name ::::::::::::::: i ::::: c); Thread.Sleep(100); } }2.5 升级线程任务传递参数/* void Start();无参数的Start方法用于启动 关联ThreadStart委托的线程对象 public delegate void ThreadStart(); void Start(object);有参数的Start方法用于启动 关联ParameterizedThreadStart委托的线程对象参数就是实现方法运行的实参 public delegate void ParameterizedThreadStart(object obj); */ static Random random new Random(); static void Main(string[] args) { //2 创建thread对象 并管理线程任务 Thread t1 new Thread(renWu); t1.Name 打印数字; Thread ta new Thread(renWu);//通过实现方法实现委托 ta.Name 打印小写字母; Thread tA new Thread((obj) {//通过lambda实现委托 char[] arr (char[])obj; for (int i 0; i 30; i) { char c (char)(random.Next(arr[0], arr[1] 1)); Console.WriteLine(Thread.CurrentThread.Name ::: i ::::: c); Thread.Sleep(100); } }); // Thread tA new Thread(renWu); tA.Name 打印大写字母; ​ ​ //3 启动线程 t1.Start(new char[] { 0,9});//Start方法的参数是线程任务委托实现方法的实参 ta.Start(new char[] { a,z}); tA.Start(new char[] { A,Z}); ​ t1.Join();//等待t1执行完毕 ta.Join();//等待ta执行完毕 tA.Join();//等待tA执行完毕 Console.WriteLine(所有线程执行完毕); ​ } //1 创建ThreadStart委托的实现方法 static void renWu(object param) { char[] arr(char[])param; for (int i 0; i 30; i) { char c (char)(random.Next(arr[0], arr[1] 1)); Console.WriteLine(Thread.CurrentThread.Name ::: i ::::: c); Thread.Sleep(100); } }2.6 模拟Thread类/* Thread类中定义两种委托ThreadStart和ParameterizedThreadStart 定义了两种构造方法分别关联两种委托的实例 定义了两种start方法分别对应不同的委托实例 定义了Name属性 用于设置和获取线程名字 定义了CurrentThread用于获取当前线程对象 */ class MyThread { private ParameterizedThreadStart pts; private ThreadStart ts; public string Name { get; set; } public static MyThread CurrentThread { get; set; } ​ public delegate void ParameterizedThreadStart(object obj); public delegate void ThreadStart(); ​ public MyThread(ParameterizedThreadStart pts) { this.pts pts; CurrentThread this; } public MyThread(ThreadStart ts) { this.ts ts; CurrentThread this; } ​ public void Start() { this.ts(); } public void Start(object obj) { this.pts(obj); } ​ }3 线程同步3.1 案例四个学生同时交作业​ internal class Demo14_04练习 { /* 四个学生给一个老师交五本作业 为了保证老得到的作业本书是20至始至终只有一个老师 为了保证学生交作业互补干扰定义四个线程 对应四个学生 ​ ​ //出现线程安全问题 线程安全问题多个线程在使用共享数据时 出现结果无法预期的现象 无法预期每次交作业的打印数字 无法预期最终的作业本书 ​ 1打印数字重复 a同学交作业 num5, 时间轮片到期 b同学交作业 num6 时间轮片也到期 随到a 打印输出6 原因a同学在num和打印num之间 时间轮片到期 被别的同学更改了num的值 2最终的作业本书不是20num;不是一个不能再细分的语句int knum1; numk; 两个语句之间被其他线程打断 */ static void Main(string[] args) { TeacherDemo14 t new TeacherDemo14();//定义老师 作为共享数据 Thread t11 new Thread(StudentRenWu); Thread t12 new Thread(StudentRenWu); Thread t13 new Thread(StudentRenWu); Thread t14 new Thread(StudentRenWu); ​ t11.Name 张3; t12.Name 李四44444; t13.Name 王五55555555; t14.Name 赵六6666666666666; ​ ​ t11.Start(t); t12.Start(t); t13.Start(t); t14.Start(t); ​ t11.Join(); t12.Join(); t13.Join(); t14.Join(); Console.WriteLine(老师的作业本书t.num); ​ } ​ //定义学生的线程任务 static void StudentRenWu(object obj) { TeacherDemo14 t(TeacherDemo14)obj; //TeacherDemo14 t new TeacherDemo14();//4个老师StudentRenWu方法被调用4次 for (int i 1; i 5; i) { //TeacherDemo14 tnew TeacherDemo14();//20个老师 每次循环对应的一个新老师 t.num; Thread.Sleep(100); Console.WriteLine(Thread.CurrentThread.Name交了其第i本作业老师的总本书是t.num); ​ } } ​ } //定义类描述老师 class TeacherDemo14 { public int num0; } ​3.2 线程安全问题//出现线程安全问题 线程安全问题多个线程在使用共享数据时 出现结果无法预期的现象 无法预期每次交作业的打印数字 无法预期最终的作业本书 1打印数字重复 a同学交作业 num5, 时间轮片到期 b同学交作业 num6 时间轮片也到期 随到a 打印输出6 原因a同学在num和打印num之间 时间轮片到期 被别的同学更改了num的值 2最终的作业本书不是20num;不是一个不能再细分的语句int knum1; numk; 两个语句之间被其他线程打断3.3 同步代码块internal class Demo14_05同步代码块 { /* 线程安全问题多个线程在操作共享数据时 出现结果无法预期(前后数据不一致)的现象 前提条件1 多个线程 2 必须有共享数据 3 一个线程有多个语句操作共享数据 解决线程安全问题同步代码块 同步代码块保证线程a使用共享数据时 线程b可以执行其他代码 如果也要使用共享数据 就需要等待 做到一个线程使用共享数据 其他线程不能对共享数据更改 同步代码块格式 lock(锁对象){ 所有操作共享数据的代码 } 注意1 锁对象可以是任意对象必须是同一个对象 2 同步代码块必须包含所有操作共享数据的代码 */ static object objLock new object(); static void Main(string[] args) { TeacherDemo142 t new TeacherDemo142();//定义老师 作为共享数据 //定义锁对象 Thread t11 new Thread(StudentRenWu); Thread t12 new Thread(StudentRenWu); Thread t13 new Thread(StudentRenWu); Thread t14 new Thread(StudentRenWu); t11.Name 张3; t12.Name 李四44444; t13.Name 王五55555555; t14.Name 赵六6666666666666; t11.Start(t); t12.Start(t); t13.Start(t); t14.Start(t); t11.Join(); t12.Join(); t13.Join(); t14.Join(); Console.WriteLine(老师的作业本书 t.num); } //定义学生的线程任务 static void StudentRenWu(object obj) { TeacherDemo142 t (TeacherDemo142)obj; for (int i 1; i 5; i) { Thread.Sleep(100); //lock (new object()) {//20个锁对象 lock (t) { //通常情况 把共享数据的对象 作为锁对象 t.num; Thread.Sleep(10); Console.WriteLine(Thread.CurrentThread.Name 交了其第 i 本作业老师的总本书是 t.num); } //锁换手的机会 } } } //定义类描述老师 class TeacherDemo142 { public int num 0; }3.4 案例四个窗口卖票internal class Demo14_06练习 { static void Main(string[] args) { Ticket ticket new Ticket(); Thread t1 new Thread(chuangKou); t1.Name 窗口1; Thread t2 new Thread(chuangKou); t2.Name 窗口1222; Thread t3 new Thread(chuangKou); t3.Name 窗口13333333; Thread t4 new Thread(chuangKou); t4.Name 窗口14444444444444; t1.Start(ticket); t2.Start(ticket); t3.Start(ticket); t4.Start(ticket); } //定义线程任务 static void chuangKou(object obj) { //把共享数据乡下转型 Ticket ticket (Ticket)obj; while (true) { lock (obj) { if (ticket.num 0)//num1 { Thread.Sleep(50); ticket.num--; Thread.Sleep(10); Console.WriteLine(Thread.CurrentThread.Name 卖出1张票票号 (ticket.num 1)); } else { Console.WriteLine(Thread.CurrentThread.Name 票已经售罄); break; } } } } } class Ticket { public int num 100; }4 线程池static Random random new Random(); static void Main(string[] args) { ThreadPool.QueueUserWorkItem((state) { for (int i 0; i 30; i) { char c (char)(random.Next(0, 9 1)); Console.WriteLine(Thread.CurrentThread.Name ::: i ::::: c); Thread.Sleep(100); } }); ThreadPool.QueueUserWorkItem((state) { for (int i 0; i 30; i) { char c (char)(random.Next(a, z 1)); Console.WriteLine(Thread.CurrentThread.Name ::::::::: i ::::: c); Thread.Sleep(100); } }); // 等待线程池任务执行实际开发中需合理同步此处仅为示例 Console.ReadLine(); }

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…