当前位置: 首页 > news >正文

C# 线程(1)

目录

        • 1 线程与进程
        • 2 创建线程
        • 3 线程等待
        • 4 线程优先级
        • 5 前台线程与后台线程
        • 6 Lock与线程安全
        • 7 Monitor
        • 8 死锁
        • 9 线程中异常处理

1 线程与进程

进程是计算机概念,一个程序运用时占用的的所有计算机资源(CPU、内存、硬盘、网络)统称为进程。
线程是操作系统中能够独立运行的最小单位,是进程(包含多个线程)中的一部分,线程也有自己的计算资源,多个线程间可以共享进程的资源
C#中的Thread其实是对计算机中线程概念的封装(API的封装),它的执行归根结底是向底层操作系统申请了线程资源。

在C#中的线程实现包括 Thread/ThreadPool/Task/Await Async
多线程的本质是资源换性能(CPU、内存、硬盘、网络)。好处是提高利用率(快) 坏处是耗费资源。但是在使用N个线程时,性能并不是成N倍的递增。
因为操作系统对CPU的调度时必须耗费些许时间,包括CPU的分片,线程的调度以及上下文的切换。同时CPU不够时降低线程的性能,这也意味着不是线程越多越好。

以下代码示例 默认引用了System.Threading命名空间

2 创建线程

线程创建 可以不通过显示的调用ThreadStart和ParameterizedThreadStart,主要是通过lamda表达式去创建线程

 static void Main(string[] args){Console.OutputEncoding = Encoding.UTF8;//public Thread(ThreadStart start) ThreadStart 一个无参数的委托Thread t1 = new Thread(() =>{Console.WriteLine("一个无参数的的线程创建");});t1.Start();//Thread(ParameterizedThreadStart start) ParameterizedThreadStart 一个有参数的委托//lambda 表达式是向线程传递数据的最强大的方法。然而必须小心,不要在启动线程之后误修改被捕获变量(captured variables).解决方法就是使用临时变量Thread t2 = new Thread((x) =>{Console.WriteLine("一个有参数的的线程创建{0}", x);});//传入参数t2.Start(100);}
3 线程等待

可以使用 Join()方法来等待其他线程完成任务

  public   class ThreadWork{public void Method1() {Console.OutputEncoding = Encoding.UTF8;Console.WriteLine("当前线程名称" + Thread.CurrentThread.Name);for (int i = 0; i < 10; i++) {Thread.Sleep(1000);//线程暂停Console.WriteLine(i);}}}     public class Program{static void Main(string[] args){ThreadWork work = new ThreadWork();Thread t = new Thread(work.Method1);t.Start();t.Join();//主线程等待子线程完成任务Console.WriteLine("Finish");}
}
4 线程优先级

线程的Priority属性决定了相对于操作系统中的其它活动线程,它可以获得多少执行时间。线程优先级的取值如下:
enum ThreadPriority { Lowest, BelowNormal, Normal, AboveNormal, Highest } 。只有当多个线程同时活动时,线程优先级才有意义。默认创建的线程优先级别为Normal

  public   class ThreadWork{public void Method1() {Console.OutputEncoding = Encoding.UTF8;Console.WriteLine("当前线程名称" + Thread.CurrentThread.Name);for (int i = 0; i < 10; i++) {Thread.Sleep(1000);//线程暂停Console.WriteLine(i);}}}
      static void Main(string[] args){ThreadWork work = new ThreadWork();Thread t1 = new Thread(work.Method1) { Priority = ThreadPriority.Highest, Name = "t1" };Thread t2 = new Thread(work.Method1) { Priority = ThreadPriority.Normal, Name = "t2" };t2.Start(); //t1线程优先级别高于t2线程,将优先获得CPU的时间片,先执行t1.Start();Console.WriteLine("Finish");}
5 前台线程与后台线程

可以通过设置IsBackground来设定线程是否为后台线程 。默认显示创建的线程都是前台线程。前台线程与后台线程最大的区别就是 进程会等待所有的前台线程,当前台线程全部结束后,尽管后台线程的任务没有执行完,进程会自动结束所有存在的后台线程。所以当程式定义了一个永远不会完成的前台线程时,进程将永远不会结束。线程的前台/后台状态与它的优先级和执行时间的分配无关

  public   class ThreadWork{public void Method2(int num) {Console.OutputEncoding = Encoding.UTF8;for (int i = 0; i < num; i++){Thread.Sleep(1000);//线程暂停Console.WriteLine("当前线程名称{0} 打印数据{1}",  Thread.CurrentThread.Name,i);}}}
       static void Main(string[] args){ThreadWork work = new ThreadWork();Thread t1 = new Thread(() => work.Method2(5)) { Name = "t1" };Thread t2 = new Thread(() => work.Method2(10)) { Name = "t2", IsBackground = true };t1.Start();t2.Start(); // t1线程为前台线程(执行时间>10s) t2线程为后台线程(执行时间>5s) 前台线程结束后,会中断t2线程Console.WriteLine("Finish");}
6 Lock与线程安全

竞态: 字面意思是竞争,并发的执行单元对共享资源(硬件资源和软件上的全局变量,静态变量等)的访问容易发生竞态,竞态是出现线程不安全的重要因素。
为了解决多线程并发抢夺共享资源的问题,可采用lock 关键字,lock会锁定最先抢到资源的线程,并且会阻塞(不占用CPU资源)其他试图抢夺该资源的其他线程,直到lock锁定的线程释放这个锁资源
这样就确保了在同一时刻只有一个线程能进入临界区(critical section,不允许并发执行的代码)像这种用来避免在多线程下的不确定性的方式被称为线程安全(thread-safe)
尽管lock关键字的存在会解决线程安全的问题,但是运用不恰当,存在严重的性能问题,因为上下切换上下文的时间可能会比单个线程执行还要慢。

   public  class CounterBase{//多线程共享这个变量,当存在读写操作时,会出现线程不安全的问题public int num { get; private set; }public CounterBase(int num){this.num = num;}public void DecrementWithLock(){Console.WriteLine(Thread.CurrentThread.Name);//使用lock关键字,标识这段方法只允许一个线程持有lock (this) {num--;}}public void IncrementWithLock(){Console.WriteLine(Thread.CurrentThread.Name);lock (this){num++;}}public void Count (){for (int i = 0; i < 10; i++){IncrementWithLock();DecrementWithLock();}}}
 static void Main(string[] args){CounterBase counter = new CounterBase(100);Thread t1 = new Thread(() => counter.Count()) { Name = "t1" }; // t1、t2 线程共享counter实例,并同时执行CountWhihLock()方法。意味着此时出现了竞态现象,容易出现线程不安全的问题 Thread t2 = new Thread(() => counter.Count()) { Name = "t2" };//多线程并行t1.Start();t2.Start();t1.Join();t2.Join();Console.WriteLine("Finish");Console.WriteLine(counter.num); //100}
7 Monitor

在Lock与线程安全这一section中,用到lock关键字来确保代码屏障区(用lock关键字包裹的代码区,同一时间,只有一个线程能进入到代码中)的安全执行 。lock关键字Monitor类的的一个语法糖。可以通过TryEnter()进入锁 Monitor.Exit()进出锁。上述示例可以用Monitor类来重构功能

   public  class CounterBase{//多线程共享这个变量,当存在读写操作时,会出现线程不安全的问题public int num { get; private set; }public CounterBase(int num){this.num = num;}public void DecrementWithLock(){Console.WriteLine(Thread.CurrentThread.Name);//使用Monitor替换lock关键字Monitor.TryEnter();num++;Monitor.Exit();}public void IncrementWithLock(){Console.WriteLine(Thread.CurrentThread.Name);//使用Monitor替换lock关键字Monitor.TryEnter();num--;Monitor.Exit();}public void Count (){for (int i = 0; i < 10; i++){IncrementWithLock();DecrementWithLock();}}}
8 死锁

死锁 是多个线程等待永远拿不到的锁资源时 出现的程式异常问题。线程处于阻塞状态中。

    public class DeadLockWork{private readonly object _lock1 = new object();private readonly object _lock2 = new object();public void Method1() {lock (_lock1) {Console.WriteLine(Thread.CurrentThread.Name+" enterd Method1");Thread.Sleep(10);lock (_lock2) {}}}public void Method2(){lock (_lock2){Console.WriteLine(Thread.CurrentThread.Name + " enterd Method2");Thread.Sleep(10);lock (_lock1){}}}}
        static void Main(string[] args){DeadLockWork deadLock = new DeadLockWork();// t1、t2 线程共享deadLock实例Thread t1 = new Thread(() => deadLock.Method1()) { Name = "t1" };Thread t2 = new Thread(() => deadLock.Method2()) { Name = "t2" };t1.Start();//t1线程进入方法后,持有_lock1锁,暂停10ms后尝试获取_lock2锁t2.Start();//t2线程进入方法后,持有_lock2锁,暂停10ms后尝试获取_lock1锁//主线程等待t1、t2线程完成任务t1.Join();t2.Join();//由于lock关键字会阻塞其他正在尝试获取锁的线程,t1持有lock1并尝试获取_lock2锁,t2持有lock2并尝试获取_lock1锁。//t1永远拿不到_lock2锁,t2永远拿不到_lock1锁。导致线程死锁。//进程等待两个前台线程执行完成,死锁导致程式永远不会结束Console.WriteLine("Finish");}
9 线程中异常处理

在实际项目中,每一个线程中应该有异常处理的逻辑,
因为每个线程都独享栈空间,主线程无法捕获其他工作线程中抛出的异常。未被捕获的异常可能会导致程式的异常中断。为了加强程式的健壮性,应该在工作任务中设计异常处理的逻辑

 static void Main(string[] args){Console.OutputEncoding = Encoding.UTF8;Thread t1 = new Thread(() =>{try{Console.WriteLine("I am Thread " + Thread.CurrentThread.Name);throw new Exception("模拟工作的中的异常:端口被占用");}catch (Exception e){//捕获线程中的异常Console.WriteLine(e.Message);}}){ Name = "t1" };t1.Start();}

相关文章:

C# 线程(1)

目录 1 线程与进程2 创建线程3 线程等待4 线程优先级5 前台线程与后台线程6 Lock与线程安全7 Monitor8 死锁9 线程中异常处理 1 线程与进程 进程是计算机概念&#xff0c;一个程序运用时占用的的所有计算机资源&#xff08;CPU、内存、硬盘、网络&#xff09;统称为进程。 线程…...

冒泡排序以及改进方案

冒泡排序以及改进方案 介绍&#xff1a; 冒泡排序属于一种典型的交换排序&#xff08;两两比较&#xff09;。冒泡排序就像是把一杯子里的气泡一个个往上冒一样。它不断比较相邻的元素&#xff0c;如果顺序不对就像水泡一样交换它们的位置&#xff0c;直到整个序列像水泡一样…...

QTextEdit 是 Qt 框架中的一个类,用于显示和编辑多行文本内容的可编辑部件

QTextEdit 是 Qt 框架中的一个类&#xff0c;用于显示和编辑多行文本内容的可编辑部件。 QTextEdit 提供了一个用于显示和编辑富文本&#xff08;包括格式化文本、图像和链接等&#xff09;和纯文本的文本编辑器。它支持基本的文本操作&#xff08;如复制、粘贴、撤销、重做等…...

vue+jsonp编写可导出html的模版,可通过外部改json动态更新页面内容

效果 导出后文件结果如图所示&#xff0c;点击Index.html即可查看页面&#xff0c;页面所有数据由report.json控制&#xff0c;修改report.json内容即可改变index.html展示内容 具体实现 1. 编写数据存储的json文件 在index.html所在的public页面新建report.json文件&#xff…...

查看各ip下的连接数

netstat -n | awk /^tcp/ {print $5} | awk -F: {print $1} | sort | uniq -c| sort -rn netstat -n&#xff1a;显示所有的网络连接&#xff0c;不包括任何服务名的解释。awk /^tcp/ {print $5}&#xff1a;使用awk命令过滤出tcp协议的连接&#xff0c;并打印出每个连接的第五…...

Linux—进程状态

目录 一.前言 1.1.通过系统调用获取进程标示符 1.2.通过系统调用创建进程 二.进程状态 三.Z(zombie)-僵尸进程 四.僵尸进程危害 一.前言 学习进程的状态&#xff0c;我们首先了解一下进程的基本数据 1.1.通过系统调用获取进程标示符 由getpid&#xff08;&#xff09…...

万宾科技可燃气体监测仪科技作用全览

燃气管网在运行过程中经常会遇到燃气管道泄漏的问题&#xff0c;燃气泄漏甚至会引起爆炸&#xff0c;从而威胁人民的生命和财产安全&#xff0c;因此对燃气管网进行定期巡检是十分必要的工作。但是传统的人工巡检已不能满足城市的需要&#xff0c;除了选择增加巡检人员之外&…...

Python与GPU编程快速入门(三)

3、使用Numba加速Python代码 Numba 是一个 Python 库,它使用行业标准 LLVM 编译器库在运行时将 Python 函数转换为优化的机器代码。 您可能想尝试用它来加速 CPU 上的代码。 然而,Numba还可以将Python 语言的子集转换为CUDA,这就是我们将在这里使用的。 所以我们的想法是,…...

praseInt 和 逻辑或连用

这是做项目时遇到json文件转换 的一个小坑 将json 对象中的值 由字符串(数字字符串) 转换为 数值类型&#xff0c;如果是 转换失败 &#xff0c;就返回 -1 这里的 parseInt 看起来非常简洁&#xff0c;但是存在一个小坑 transformedData[fieldsToCheck[i]] parseInt(origina…...

对属于国家秘密的地理信息的获取、持有、提供、利用情况进行登记并长期保存,实行可追溯管理

对属于国家秘密的地理信息的获取、持有、提供、利用情况进行登记并长期保存&#xff0c;实行可追溯管理 数据记录&#xff08;包括获取、持有、提供、利用、销毁等全闭环&#xff09;...

XAER_RMERR: Fatal error occurred in the transaction branch异常解决

XAER_RMERR: Fatal error occurred in the transaction branch异常解决 数据库权限问题&#xff01;&#xff01;&#xff01;不是mysql驱动问题&#xff0c;执行下面命令解决 GRANT XA_RECOVER_ADMIN ON *.* TO root% ;...

Redis面试常见问题

Redis中的Lua脚本到底能不能保证原子性&#xff1f; Redis中Lua脚本的执行&#xff0c;可以保证并发编程中不可再拆分的这个原子性&#xff0c;但是没有保证数据库ACID中要么都执行要么都回滚的这个原子性。Lua脚本执行过程中命令产生错误&#xff0c;是不会回滚的&#xff0c…...

浏览器触发下载Excel文件-Java实现

目录 1:引入maven 2:代码实现 3.导出通讯录信息到Excel文件 4.生成并下载Excel文件部分解释 1:引入maven 添加依赖:首先,在你的项目中添加EasyExcel库的依赖。你可以在项目的构建文件(如Maven的pom.xml)中添加以下依赖项:<dependency><groupId>com.alib…...

每日汇评:黄金在上涨趋势恢复之前面临修正性回调的风险

周三早些时候&#xff0c;金价触及六个月来的最高水平&#xff0c;接近 2050 美元&#xff1b; 在美联储转向鸽派立场后&#xff0c;美元和美债双双下跌&#xff1b; 日线图上的超买RSI指标警示多头&#xff0c;但即将到来的金十字图形使上行保持不变&#xff1b; 金价在周三亚…...

【开源】基于Vue.js的大学计算机课程管理平台的设计和实现

项目编号&#xff1a; S 028 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S028&#xff0c;文末获取源码。} 项目编号&#xff1a;S028&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 实验课程档案模块2.2 实验资源模块2…...

c++环形队列

c环形队列 c环形队列 c环形队列 #pragma once#include <iostream> #include <vector>/// <summary> /// - 环形队列 /// - 不是线程安全 /// </summary> /// <typeparam name"T"></typeparam> template <typename T> cla…...

智能客服核心技术——预测会话与答案生成

1.信息检索 2. 句型模板匹配标准问题生成答案 3.根据知识图谱推理得到答案...

C语言——计算Fibonacci数列

方式一 for循环 (20位) #define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h> int main() {int n;int a[20]{1,1};for ( n 1; n <20; n){a[n]a[n-2]a[n-1];}for ( n 0; n < 20; n){if(n%50)printf("\n");printf("%12d ",a[n]);}return 0; …...

【ASP.NET CORE】.NET 6.0 NET CORE MVC连接SQLSERVER数据库

项目装NuGet包&#xff0c;具体版本如下 在appsettings.json中&#xff0c;添加连接字符串 代码如下&#xff1a; "ConnectionStrings": {"MVCSqlContext": "Serverlocalhost;DatabaseAddress;User IDsa;Passwordsa;TrustServerCertificatetrue&q…...

filebeat日志收集工具

1、优缺点 filebeat收集的数据可以发往多个主机&#xff0c;即远程收集 filebeat无法实现数据过滤&#xff0c;一般结合logstash的数据过滤一块使用 2、远程收集多主机的日志实验 实验目的&#xff1a;一体化查看日志 实验条件&#xff1a; 主机名 服务器 IP地址 组件 …...

PLC立体车库智能仿真系统:博途V15 3×2车库模型,西门子PLC控制,触摸屏操作,自动出入...

PLC立体车库智能仿真 博途V15 32立体车库 西门子1200PLC 触摸屏仿真 不需要实物 自带人机界面 小车上下行有电梯效果 每一个程序段都有注释 FC块标准化编写 自带变频器输出也可以仿真 现在拥有自动出入仓库的功能 IO表已列出最近在搞的32立体车库仿真项目挺有意思&#xff0c;用…...

如何高效协作开发Fisher插件:团队合作的最佳实践指南

如何高效协作开发Fisher插件&#xff1a;团队合作的最佳实践指南 【免费下载链接】fisher A plugin manager for Fish 项目地址: https://gitcode.com/gh_mirrors/fi/fisher Fisher是Fish Shell的终极插件管理器&#xff0c;让团队协作开发插件变得简单高效。无论你是新…...

性价比高的南昌实体店线上获客哪个靠谱

在南昌&#xff0c;实体店想要在竞争激烈的市场中脱颖而出&#xff0c;线上获客是关键。然而&#xff0c;面对众多的线上获客途径&#xff0c;哪个才靠谱且性价比高呢&#xff1f;今天&#xff0c;我们就来详细探讨一下&#xff0c;同时为大家推荐南昌琨瑜象限本地生活运营服务…...

全志科技Linux驱动开发面试经验与Cache一致性解析

1. 全志科技Linux驱动开发工程师面试全解析作为一名在嵌入式Linux领域摸爬滚打多年的老司机&#xff0c;最近刚经历了全志科技的社招面试。这家国产芯片大厂的面试风格相当有特色&#xff0c;特别是对Cache一致性和驱动开发细节的考察&#xff0c;堪称"灵魂拷问"级别…...

基于springboot+vue高校物资信息采购系统hx0807

文章目录详细视频演示技术介绍功能介绍核心代码系统效果图源码获取详细视频演示 文章底部名片&#xff0c;获取项目的完整演示视频&#xff0c;免费解答技术疑问 技术介绍 开发语言&#xff1a;Java 框架&#xff1a;ssm JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomca…...

避坑指南:SAP冲销原因配置常见错误及解决方案(附SPRO操作截图)

SAP FI模块冲销原因配置实战避坑指南 刚接触SAP FI模块的财务顾问们&#xff0c;在配置冲销原因时往往会遇到各种"坑"。这些看似简单的后台配置&#xff0c;一旦出错可能导致整个月结流程卡壳。本文将结合真实项目案例&#xff0c;带你避开那些教科书上不会写的配置陷…...

你还在用传统线程池扛高并发?Java 25虚拟线程真实压测对比:错误率下降92.7%,但90%团队正踩这6个配置雷区

第一章&#xff1a;Java 25虚拟线程高并发架构演进全景图Java 25正式将虚拟线程&#xff08;Virtual Threads&#xff09;从预览特性转为稳定特性&#xff0c;并深度整合至JDK核心运行时与工具链&#xff0c;标志着JVM并发模型进入“轻量级线程即原语”新纪元。相比传统平台线程…...

归并排序力扣题(leetcode)萍

1.概述在人工智能快速发展的今天&#xff0c;AI不再仅仅是回答问题的聊天机器人&#xff0c;而是正在演变为能够主动完成复杂任务的智能代理。OpenAI的Codex CLI就是这一趋势的典型代表——一个跨平台的本地软件代理&#xff0c;能够在用户的机器上安全高效地生成高质量的软件变…...

毕业论文开挂指南:好写作AI助你实现学术写作“降维打击”

写论文这件事&#xff0c;你需要的不是更拼命的自己&#xff0c;而是一套颠覆认知的思维加速器 深夜的自习室&#xff0c;你面前的Word文档还停留在那行刺眼的光标&#xff0c;而这已经是你刷的第三个整晚了。论文进度&#xff1a;0字。 你开始怀疑人生&#xff1a;明明看了那…...

1篇1章4节:生成对抗网络GAN和图像生成领域的StyleGAN

近年来&#xff0c;人工智能&#xff08;AI&#xff09;技术迅猛发展&#xff0c;特别是在计算机视觉领域&#xff0c;生成对抗网络&#xff08;Generative Adversarial Network&#xff0c;GAN&#xff09;推动了图像生成技术的巨大进步。从早期的基本GAN架构&#xff0c;到渐…...