C# 线程同步(一)同步概念介绍

目录

1.阻塞(Blocking)

2.阻塞 VS 轮询

3.线程状态


        到目前为止,我们已经阐述了如何在线程上启动任务、配置线程以及实现双向数据传递。同时,我们也说明了局部变量是线程私有的,而引用可以通过共享字段在线程间传递以实现通信。

下一步的关键是同步机制:通过协调线程行为来获得可预测的结果。当多个线程访问同一数据时,同步显得尤为重要——这个领域看似简单却暗藏风险。

同步构造可分为四大类别:

  1. 简单阻塞方法
    这类方法通过等待其他线程结束或计时完成来实现同步。例如:SleepJoin 和 Task.Wait 都属于简单阻塞方法。

  2. 锁定构造
    用于限制同时执行某段代码或操作的线程数量。最常见的独占锁(如 lock/Monitor.Enter/Monitor.ExitMutex 和 SpinLock)仅允许一个线程进入,确保竞争线程访问共享数据时互不干扰。非独占锁包括信号量(Semaphore/SemaphoreSlim)和读写锁。

  3. 信号构造
    允许线程暂停运行直至接收到其他线程的通知,从而避免低效的轮询。常用信号机制有两种:事件等待句柄(event wait handles)和 Monitor 的 Wait/Pulse 方法。.NET Framework 4.0 新增了 CountdownEvent 和 Barrier 类。

  4. 非阻塞同步构造
    通过调用处理器原语来保护共享字段的访问。CLR 和 C# 提供的非阻塞构造包括:Thread.MemoryBarrierThread.VolatileReadThread.VolatileWritevolatile 关键字以及 Interlocked 类。

除最后一类外,阻塞机制在其他类别中均占核心地位。下面我们将简要探讨这一概念。

1.阻塞(Blocking)

        当线程因某些原因暂停执行时(例如通过 Sleep 进入休眠,或通过 Join/EndInvoke 等待其他线程结束),该线程即被视为阻塞状态。阻塞线程会立即释放其处理器时间片,此后在阻塞条件满足前不再消耗任何处理器资源。可通过线程的 ThreadState 属性检测阻塞状态:

bool blocked = (someThread.ThreadState & ThreadState.WaitSleepJoin) != 0;
(由于线程状态可能在检测与后续操作之间发生变化,此代码仅适用于诊断场景。)

当线程阻塞或解除阻塞时,操作系统会执行上下文切换。这将产生几微秒的开销。

线程会通过以下四种方式解除阻塞(当然,按电脑电源键不算!):

  1. 阻塞条件得到满足

  2. 操作超时(如果指定了超时时间)

  3. 通过Thread.Interrupt被中断

  4. 通过Thread.Abort被中止

需要注意的是,如果线程是通过(已弃用的)Suspend方法暂停执行的,则不会被判定为处于阻塞状态。

2.阻塞 VS 轮询

当线程需要暂停直到特定条件满足时,通常有两种实现方式:

  1. 阻塞等待(高效方式)
    通过信号和锁机制实现,线程会进入阻塞状态直到条件满足,此时操作系统会调度其他线程执行。

  2. 轮询等待(简单但低效)
    线程通过循环检测条件来实现等待,例如:

while (!proceed);  // 忙等待

while (DateTime.Now < nextStartTime); // 时间等待

性能考量:

  • 纯轮询会完全占用CPU资源,因为CLR和操作系统会认为线程正在进行重要计算

  • 这种方式的CPU利用率是100%,极其浪费系统资源

改进方案:
可以采用混合阻塞的方式:

while (!proceed) Thread.Sleep(10);  // 每次检查后休眠10ms

虽然不够优雅,但相比纯轮询能显著降低CPU占用率。不过需要注意共享变量(如proceed标志)的并发访问问题,正确的锁和信号量使用可以避免这些问题。

适用场景:
当预期条件能在极短时间内(如几微秒)满足时,短暂轮询可能更高效,因为它避免了上下文切换的开销和延迟。.NET框架为此提供了专门的工具类和方法,这些内容将在并行编程章节详细介绍。

这里小节一下:

  • 阻塞等待:适合大多数情况,资源利用率高

  • 纯轮询:简单但资源浪费严重

  • 混合模式:折中方案,需要处理并发问题

  • 微秒级等待:特殊场景下短暂轮询可能更优

3.线程状态

        您可以通过ThreadState属性查询线程的执行状态。该属性返回一个ThreadState类型的标志枚举,它以位运算方式组合了三个"层次"的数据。不过,大多数枚举值都是冗余的、未使用的或已废弃的。下图展示了其中一个"层次":

以下代码可将ThreadState精简为四个最常用的状态值:未启动(Unstarted)、运行中(Running)、等待休眠或加入(WaitSleepJoin)和已停止(Stopped):

public static ThreadState SimpleThreadState(ThreadState ts)
{return ts & (ThreadState.Unstarted |ThreadState.Running |ThreadState.WaitSleepJoin |ThreadState.Stopped);
}

ThreadState属性适用于诊断目的,但不适合用于同步控制,因为在检测线程状态和基于该状态执行操作之间,线程状态可能会发生变化。


本节完,下一节将开始介绍同步工具

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

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

相关文章

解决leetcode第3588题.找到最大三角形面积

3588.找到最大三角形面积难度&#xff1a;中等问题描述&#xff1a;给你一个二维数组coords&#xff0c;大小为nx2&#xff0c;表示一个无限笛卡尔平面上n个点的坐标。找出一个最大三角形的两倍面积&#xff0c;其中三角形的三个顶点来自coords中的任意三个点&#xff0c;并且该…

WIFI 安全测试记录

之前为实训课特意买的无线网卡没用上&#xff0c;但是我怎么可能让他荒废。所以用了几个下午&#xff0c;浅学了WiFi&#xff0c;当然没找到什么好教材&#xff0c;自己摸索着学的很基础&#xff0c;主要是当练习了&#xff0c;特此把我此前学习…WiFi密码实践过程写上来。 省流…

android14设置--网络--Internet副标题修改

收银机订制项目 插SIM卡&#xff0c;设备使用数据流量时&#xff0c;设置–网络–Internet副标题显示对应SIM卡运营商名称&#xff0c;客户要求修改这时的名称(注意图标也要同步修改) packages\apps\Settings\src\com\android\settings\network\InternetPreferenceController.j…

Web3区块链有哪些岗位?

Web3区块链领域的岗位丰富多样&#xff0c;涵盖技术开发、产品管理、运营、商务等多个方面&#xff0c;以下是具体介绍&#xff1a; - 技术开发类&#xff1a; - 智能合约开发工程师&#xff1a;负责编写、审计和优化智能合约&#xff0c;常见于DeFi开发&#xff0c;包括抵押…

解决 Spring Boot 对 Elasticsearch 字段没有小驼峰映射的问题

场景重现在使用 MyBatis/Mybatis-Plus 框架对 MySQL 操作时习惯了字段名小驼峰映射&#xff0c;然而在操作 Elasticsearch 时发现字段名没有小驼峰映射。解决方法1. 使用 ObjectMapper 手动转换&#xff1a; 这是最直接也最常用的方法。 在 Spring Boot 应用中使用 ObjectMappe…

Error:Cannot find module ‘chokidar‘

错误复现 在vue开发中&#xff0c;出现报错&#xff1a;Error&#xff1a;Cannot find module ‘chokidar’ 原因 缺包导致 解决方案 直接安装依赖包 npm install chokidar依旧无效&#xff0c;删除node_modules重新安装 rm -rf node_modules npm i

Spring AI 向量数据库详解与 RAG 简单实战项目

一、什么是向量数据库&#xff1f; 向量数据库用于存储、检索稠密语义向量&#xff08;Embedding&#xff09;&#xff0c;是构建 RAG&#xff08;检索增强生成&#xff09;系统的核心组件。它支持近似最近邻搜索&#xff08;ANN&#xff09;&#xff0c;可根据语义相似度找出…

【RK3568+PG2L50H开发板实验例程】Linux部分/FPGA FSPI 通信案例

本原创文章由深圳市小眼睛科技有限公司创作&#xff0c;版权归本公司所有&#xff0c;如需转载&#xff0c;需授权并注明出处&#xff08;www.meyesemi.com) 1. 简介 本案例旨在 ARM端运行 Linux系统&#xff0c;基通过 FSPI测试。 2. ARM端和 FPGA端通信流程 (1)ARM端实现SP…

github如何创建一个自己的仓库保姆级教程

文章目录 准备阶段(github官网)添加ssh公钥添加token创建仓库 本地设置本地代理创建仓库添加文件到仓库进行提交 准备阶段(github官网) 添加ssh公钥 创建SSH KEY。先看一下你C盘用户目录下有没有.ssh目录&#xff0c;有的话看下里面有没有id_rsa和id_rsa.pub这两个文件&#…

LabVIEW 网络流通信功能

LabVIEW 的网络流技术实现主机 VI&#xff08;Host VI&#xff09;与客户端 VI&#xff08;ClientVI&#xff09;间的双向数据交互&#xff0c;包含命令发送与波形数据传输&#xff0c;支持跨设备、跨进程的实时通信&#xff0c;满足分布式系统中数据交互与控制需求。 主机 VI逻…

Prompt 精通之路(一)- AI 时代的新语言:到底什么是 Prompt?为什么它如此重要?

AI 时代的新语言&#xff1a;到底什么是 Prompt&#xff1f;为什么它如此重要&#xff1f; 标签&#xff1a; #Prompt新手指南 #提示词入门 #AI指令 #人工智能 #ChatGPT &#x1f680; Prompt 精通之路&#xff1a;系列文章导航 第一篇&#xff1a;AI 时代的新语言&#xff1a…

uniapp 滚动tab

uniapp woui unibest <route lang"json5">{style: {navigationBarTitleText: 知识产权,navigationBarBackgroundColor: #C80F06,navigationBarTextStyle: white,backgroundColorTop: #C80F06,},} </route> <template><view class"bgc-b …

日事清驾驶舱模式上线:实时数据更新+项目管理+数据可视化,提升决策效率​

大家好&#xff01;我们在日事清最新更新中推出了一个令人激动的新功能——驾驶舱模式。这一全新界面将为企业管理者和团队提供一个全面、实时的数据展示平台。下面&#xff0c;让我们详细了解这个功能如何帮助您更好地把握企业动态和提升决策效率。 快速入口&#xff1a;一键激…

【Maven】Maven深度避坑指南:依赖冲突全维度解决方案与工业级实战(超万字解析)

注&#xff1a;本文基于50大型企业级项目经验&#xff0c;结合Maven底层源码机制&#xff0c;系统化解决依赖冲突问题。包含20个实战场景、10类特殊案例及5大防御体系构建方案。 Maven深度避坑指南&#xff1a;依赖冲突全维度解决方案与工业级实战&#xff08;超万字解析&#…

Rust Web 全栈开发(二):构建 HTTP Server

Rust Web 全栈开发&#xff08;二&#xff09;&#xff1a;构建 HTTP Server Rust Web 全栈开发&#xff08;二&#xff09;&#xff1a;构建 HTTP Server创建成员包/库&#xff1a;httpserver、http解析 HTTP 请求HTTP 请求的构成构建 HttpRequest 构建 HTTP 响应HTTP 响应的构…

小架构step系列01:小架构初衷

1 概述 小公司做业务服务&#xff0c;需要聚焦到实际的业务上&#xff0c;尽快通过业务服务客户&#xff0c;给客户创建价值&#xff0c;公司才能生存下去。在技术上采用的Web应用架构具备以下特点&#xff1a; 主要由开源组件组装而成。这样既可以节省成本&#xff0c;也可以把…

苹果AR/VR头显路线图曝光,微美全息推进AI/AR智能眼镜新品开启视觉体验篇章

日前&#xff0c;郭明錤发表了一篇关于苹果&#xff08;AAPL.US&#xff09;2025-2028头戴式产品路线图的文章&#xff0c;里面提到苹果正在开发涵盖MR头显、AI眼镜、AR眼镜、Birdbath眼镜等共计7款设备。 苹果的头显设备中&#xff0c;大量出货的产品是类似于Ray-Ban Meta的智…

python pyecharts 数据分析及可视化(2)

一、任务要求 任务二&#xff1a;感冒高发期分析 【任务说明】 感冒是一种常见的急性上呼吸道病毒性感染性疾病&#xff0c;多由鼻病 毒、副流感病毒、呼吸道合胞病毒、埃可病毒、柯萨奇病毒、冠状病 毒、腺病毒等引起。临床表现为鼻塞、喷嚏、流涕、发热、咳嗽、头 痛等&#…

React自学 基础一

React基础 React 是一个由 Facebook&#xff08;现 Meta&#xff09;开发并维护的、开源的 JavaScript 库&#xff0c;主要用于 构建用户界面&#xff08;UI&#xff09;&#xff0c;尤其是单页面应用程序中的动态、交互式界面。 简单示例&#xff1a; import React, { useSt…

PHP语法基础篇(八):超全局变量

超全局变量是在 PHP 4.1.0 中引入的&#xff0c;并且是内置变量&#xff0c;可以在所有作用域中始终可用。 PHP 中的许多预定义变量都是"超全局的"&#xff0c;这意味着它们在一个脚本的全部作用域中都可用。在函数或方法中无需执行 global $variable; 就可以访问它们…