unity C#什么是线程锁,以及使用案例

news2025/1/16 15:39:53

文章目录

    • 原理
      • 1. **互斥**(Mutual Exclusion):
      • 2. **缓存一致性与内存屏障**:
      • 3. **操作系统的支持**:
      • 4. **编程语言级别的实现**:
      • 5. **避免死锁**:
      • 图示
    • 实例1
    • 实例2

原理

线程锁的原理主要是为了在多线程环境下提供对共享资源访问的同步机制,防止多个线程同时读写同一数据导致的数据不一致、竞态条件和死锁等问题。以下是线程锁基本工作原理的概述:

1. 互斥(Mutual Exclusion):

线程锁的核心原理是互斥,即在同一时刻,只有一个线程能够获得锁并执行临界区(Critical Section)内的代码。当一个线程进入临界区时,它会先获取锁,此时其他试图获取该锁的线程将被阻塞或挂起,直到持有锁的线程执行完临界区内的代码,并释放锁。

2. 缓存一致性与内存屏障

在现代CPU架构中,为了提高性能,每个处理器都有自己的高速缓存,这就可能导致不同线程看到的数据不一致。线程锁在硬件层面通常配合内存屏障(Memory Barrier)使用,确保缓存中的数据在锁的获取和释放时与主内存保持一致,即修改后的数据会被及时刷新到主内存,而其他线程在获取锁后能从主内存重新加载最新的数据。

3. 操作系统的支持

在操作系统层面,比如Linux系统中,通过pthread_mutex_t这样的互斥量来实现线程锁。操作系统内核提供了原子操作来管理这些互斥量的状态,使得加锁、解锁的过程对于所有参与竞争的线程都是可见且有序的。

4. 编程语言级别的实现

在C#等高级编程语言中,可以通过关键字如lock(C#)或者synchronized(Java)来简化线程锁的使用。例如,在C#中,lock (obj)语句会隐式地调用Monitor类的Enter和Exit方法,从而实现对指定对象的加锁和解锁操作。

5. 避免死锁

为了避免死锁的发生,线程锁的使用需要遵循一定的规则,如避免循环等待锁的情况,以及在持有锁的情况下尽量不要尝试获取其他锁(除非满足特定的锁顺序)。

总结起来,线程锁通过强制单一线程在某一时刻独占资源,以及协调各个线程之间的执行顺序,来达到多线程环境下的资源共享安全和数据一致性。

图示

在这里插入图片描述

在Unity中,线程锁(Thread Lock)是一种同步机制,用于保护多线程环境下的共享资源不被多个线程同时访问和修改。由于Unity引擎本身主要运行在一个单线程(即主线程)上,但现代Unity项目中经常使用多线程处理耗时操作,如网络通信、文件I/O、计算密集型任务等,这时线程锁就显得尤为重要。

Unity中的线程锁通常采用C#的标准库System.Threading命名空间下的Mutex、Monitor或Semaphore类来实现,最常用的是Monitor类以及其提供的Enter和Exit方法,或者更简洁的lock关键字。

实例1

例如,假设我们有一个在多线程环境中需要安全访问和修改的共享数据结构:

public class ThreadSafeExample : MonoBehaviour
{
    // 共享资源
    private Queue<string> messageQueue = new Queue<string>();
    private object queueLock = new object(); // 创建一个对象作为锁

    // 线程A:添加消息到队列
    private void EnqueueMessage(string msg)
    {
        lock (queueLock) // 使用lock语句进行加锁
        {
            messageQueue.Enqueue(msg);
            Debug.Log("Enqueued message: " + msg);
        }
    }

    // 线程B:从队列中取出并处理消息
    private void ProcessNextMessage()
    {
        string msg;
        lock (queueLock) // 同样在此区域加锁
        {
            if (messageQueue.Count > 0)
            {
                msg = messageQueue.Dequeue();
            }
            else
            {
                msg = null;
            }
        }

        if (msg != null)
        {
            Debug.Log("Processing message: " + msg);
            // 这里处理消息...
        }
    }

    // 启动线程
    private void StartThreads()
    {
        Thread sendThread = new Thread(EnqueueMessages);
        Thread receiveThread = new Thread(ProcessMessages);

        sendThread.Start();
        receiveThread.Start();
    }
}

在这个例子中,queueLock对象作为线程锁来确保当一个线程正在向队列中添加消息时,另一个线程不会同时尝试从队列中移除消息。通过使用lock关键字,我们可以创建一段互斥代码块,同一时刻只有一个线程能够执行这段代码,从而避免了竞态条件和其他并发问题。

实例2

当然,线程锁的使用场景非常广泛,下面是一个Unity中使用线程进行异步加载资源,并利用线程锁确保资源安全加载和访问的例子:

using UnityEngine;
using System.Collections;
using System.Threading;
using System.Collections.Generic;

public class ThreadSafeResourceLoader : MonoBehaviour
{
    private readonly object resourceLock = new object();
    private Dictionary<string, Texture2D> loadedTextures = new Dictionary<string, Texture2D>();

    public void LoadTextureAsync(string texturePath)
    {
        var thread = new Thread(() =>
        {
            Texture2D texture = null;
            
            // 异步加载资源
            WWW www = new WWW("file://" + texturePath);
            while (!www.isDone) { /* 等待加载完成 */ }

            if (string.IsNullOrEmpty(www.error))
            {
                texture = www.texture;
            }
            else
            {
                Debug.LogError("Failed to load texture: " + www.error);
            }

            // 使用线程锁安全地将加载好的资源添加到字典
            lock (resourceLock)
            {
                loadedTextures[texturePath] = texture;
            }
        });

        thread.Start();
    }

    public Texture2D GetLoadedTexture(string texturePath)
    {
        Texture2D texture = null;

        // 使用线程锁安全地从字典中获取已加载的资源
        lock (resourceLock)
        {
            loadedTextures.TryGetValue(texturePath, out texture);
        }

        return texture;
    }
}

在这个例子中,LoadTextureAsync方法在新线程中异步加载纹理资源,加载完成后通过线程锁resourceLock安全地将加载好的纹理添加到loadedTextures字典中。同时,在主线程或其他线程调用GetLoadedTexture方法时,同样需要通过线程锁确保在读取已加载纹理时不会发生竞态条件。

python推荐学习汇总连接:
50个开发必备的Python经典脚本(1-10)

50个开发必备的Python经典脚本(11-20)

50个开发必备的Python经典脚本(21-30)

50个开发必备的Python经典脚本(31-40)

50个开发必备的Python经典脚本(41-50)
————————————————

​最后我们放松一下眼睛
在这里插入图片描述

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

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

相关文章

美国智库发布《用人工智能展望网络未来》的解析

文章目录 前言一、人工智能未来可能改善网络安全的方式二、人工智能可能损害网络安全的方式三、人工智能使用的七条建议四、人工智能的应用和有效使用AI五、安全有效地使用人工智能制定具体建议六、展望网络未来的人工智能&#xff08;一&#xff09;提高防御者的效率&#xff…

如何优雅的实现主机与虚拟机文件共享?

我们在嵌入式开发中使用虚拟机时&#xff0c;经常需要在主机和虚拟机操作系统之间传输文件。以常用的虚拟机软件 Vmware 为例&#xff0c;如果安装了虚拟机工具&#xff0c;我们可以通过直接拖放文件的方式快速实现文件的交互&#xff0c;但这种方式做不到文件的同步、修改&…

华为路由设备DHCPV6配置

组网需求 如果大量的企业用户IPv6地址都是手动配置&#xff0c;那么网络管理员工作量大&#xff0c;而且可管理性很差。管理员希望实现公司用户IPv6地址和网络配置参数的自动获取&#xff0c;便于统一管理&#xff0c;实现IPv6的层次布局。 图1 DHCPv6服务器组网图 配置思路 …

Android 系统启动过程纪要(基于Android 10)

前言 看过源码的都知道&#xff0c;Launcher系统启动都会经过这三个进程 init ->zygote -> system_server。今天我们就来讲解一下这三个进程以及Launcher系统启动。 init进程 准备Android虚拟机环境&#xff1a;创建和挂载系统文件目录&#xff1b;初始化属性服务&…

Halcon基于灰度值的模板匹配

Halcon基于灰度值的模板匹配 基于灰度值的模板匹配是最经典的模板匹配算法&#xff0c;也是最早提出来的模板匹配算法。这种算法的根本思想是&#xff0c;计算模板图像与检测图像之间的像素灰度差值的绝对值总和&#xff08;SAD方法&#xff09;或者平方差总和&#xff08;SSD…

Java NIO (一)简介

1 NIO简介 在1.4版本之前&#xff0c;Java NIO类库是阻塞IO&#xff0c;从1.4版本开始&#xff0c;引进了新的异步IO库&#xff0c;被称为Java New IO类库&#xff0c;简称为Java NIO。New IO类库的目的 就是要让Java支持非阻塞IO。 Java NIO类库包含三个核心组件&#xff1a; …

Vue3中provide,inject使用

一&#xff0c;provide,inject使用&#xff1a; 应用场景&#xff1a;向孙组件传数据 应用Vue3碎片&#xff1a; ref&#xff0c;reactive&#xff0c;isRef&#xff0c;provide, inject 1.provide,inject使用 a.爷组件引入 import {ref,provide} from vue const drinkListre…

LeetCode、2542. 最大子序列的分数【中等,排序+小顶堆】

文章目录 前言LeetCode、2542. 最大子序列的分数【中等&#xff0c;排序小顶堆】题目及类型思路及代码实现 资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝2W&#xff0c;csdn博客专家、Java领域优质创作者&#xff0c;博客之星、阿里云平台优质作者、专注于Java后端技术领…

安泰射频功率放大器的主要的类型及主要参数

射频功率放大器是一种将射频信号放大到更高功率级别的电子设备。根据其工作原理和应用要求&#xff0c;射频功率放大器可以分为不同的类型。下面西安安泰将介绍一些常见的射频功率放大器类型和相关的主要参数。 A类功率放大器&#xff1a;A类功率放大器是一种广泛使用的线性放大…

Navicat教程

下载连接&#xff08;无限使用版&#xff09; 链接&#xff1a;https://pan.baidu.com/s/1IprYLRv0bSnW-XKn0trRtw 提取码&#xff1a;j6qx 连接使用 1.1 连接数据库 打开navicat&#xff0c;点击连接&#xff0c;选择数据库 1.2 操作数据库 右键连接&#xff0c;点击新建数…

Excel 动态可视化图表分享

AIGC ChatGPT 职场案例 AI 绘画 与 短视频制作 PowerBI 商业智能 68集 数据库Mysql 8.0 54集 数据库Oracle 21C 142集 Office 2021实战应用 Python 数据分析实战&#xff0c; ETL Informatica 数据仓库案例实战 Excel 2021实操 100集&#xff0c; Excel 2021函数大全 80集 Exc…

【C++】入门C++前想要了解的小知识

个人主页 &#xff1a; zxctsclrjjjcph 文章封面来自&#xff1a;艺术家–贤海林 如有转载请先通知 目录 1. 前言2. 什么是C3. C的发展史4. C的重要性4.1 语言的使用广泛度4.2 在工作领域中4.3 在校招领域中 5. 如何学习C5.1 看看别人怎么学习的5.2 自己怎么学 1. 前言 今天开…

【FastAPI】请求体

在 FastAPI 中&#xff0c;请求体&#xff08;Request Body&#xff09;是通过请求发送的数据&#xff0c;通常用于传递客户端提交的信息。FastAPI 使得处理请求体变得非常容易。 请求体是客户端发送给 API 的数据。响应体是 API 发送给客户端的数据 注&#xff1a;不能使用 …

docker容器和常用命令

1.什么是容器 容器是隔离的环境中运行的一个 进程 , 如果进程结束 , 容器就会停止. 细致: 容器的隔离环境 , 拥有自己的 ip 地址 , 系统文件 , 主机名 , 进程管理 , 相当于一个 mini的系统 2.容器 vs 虚拟机 3.Docker极速上手指南 #1.安装相关依赖. sudo yum install -y …

链表的中间节点

链表的中间节点 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能&#xff0c;轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/middle-of-the-…

python之AI视频去码图片修复CodeFormer,把你的老照片变漂亮

CodeFormer介绍 CodeFormer是由南洋理工大学-商汤科技联合研究中心S-Lab在NeurIPS 2022上提出的一种基于VQGANTransformer的人脸复原模型。该方法基于预训练VQGAN离散码本空间&#xff0c;改变复原任务的固有范式&#xff0c;将人脸复原任务转成Code序列的预测任务&#xff0c…

DOM 的 diff 算法

经典面试题&#xff1a; 1&#xff09;react/vue中的 key 有什么作用&#xff1f;&#xff08;key的内部原理是什么&#xff1f;&#xff09; 2&#xff09;为什么遍历列表时&#xff0c;key 最好不用 index&#xff1f; 1、虚拟dom中key的作用&#xff1a; 1&#xff09; 简…

SpringBoot新手入门完整教程和项目示例

文章目录 SpringBoot新手入门完整教程和项目示例1、SpringBoot简介2、Spring Boot的核心功能&#xff1f;&#xff08;优点&#xff09;3、SpringBoot与SpringMVC 的区别&#xff1f;4、构建SpringBoot项目4.1、在官网自动生成下载spring boot项目4.2、手动使用maven创建Spring…

vue中使用高德地图,根据类型显示不同点,点击出现弹框居中显示,并在可视化区域显示所有点

效果图 一、安装 vue-amap 插件 npm install vue-amap --save二、vue页面 <template> <div><ul styledisplay:flex;><liv-for"(item, index) in checkList":key"index"click"onClick(item)":class"item.checked ?…

直观全面的数据库管理 RazorSQL

RazorSQL是一款功能强大的数据库管理工具&#xff0c;适用于各种数据库系统&#xff0c;如MySQL、Oracle、SQL Server、SQLite等。它提供了一个直观的界面&#xff0c;让用户能够轻松地管理数据库&#xff0c;包括查询、编辑、创建和管理数据库对象等操作。RazorSQL还支持多种S…