Unity中的多线程编程

简介: Unity本身并不建议使用线程,推荐用协程来代替,但是很多情况下,协程并不能实现想要的功能,因此,Unity的多线程开发还是需要学习的。

协程与线程的区别

协程

本质上是单线程编程,将一个函数放到多个帧中执行,多个协程无法并发,同一时间,只有一个协程运行。

  • 优点:

    1. 不需要考虑数据同步的问题
    2. 可以直接访问游戏对象
    3. 将异步逻辑,以一种类似同步的方法编写
    4. 性能上没有较大的开销
    5. 分散计算压力,允许将耗时操作分为多步运行
  • 缺点:

    1. 容易产生GC
    2. 无法并发,多线程下载等需求效率无法提升
    3. 部分协程操作可能会阻塞主线程,导致游戏卡顿

线程

创建子线程,允许与主线程同时处理逻辑,多个线程支持并发。

  • 优点:

    1. 支持并发,可以提高计算效率
    2. 子线程逻辑独立运行,不会阻塞游戏主线程
  • 缺点:

    1. 无法访问游戏物体
    2. 需要通过加锁等操作,手动保证数据同步
    3. 线程操作较消耗性能

线程使用场景

  • 操作会造成游戏卡顿的逻辑
  • 数据处理相关,数据大又不涉及到游戏物体的功能,如多线程下载、寻路数据计算等

Unity多线程编程的坑

Unity多线程编程有许多坑,这也是官方建议使用协程的原因,这里列举了部分坑及其解决方案

编译器环境下停止游戏后分线程仍在运行

描述

编译器环境下停止游戏是不会销毁主线程,这也意味着游戏过程中开启的子线程,也不会游戏的停止而销毁,虽然这个问题仅仅会在开发阶段出现,但是也很容易出现许多不可预知的BUG,浪费时间去修复。

解决方案

注意在OnApplicationQuit、OnDestroy等生命周期内,加入子线程的销毁,保证停止游戏后,会手动销毁线程。

HTTP多线程开发时,出现“连接被异常关闭”的异常

描述

C#中Http请求的并发连接数默认最大为2,这也意味着,多线程中,超过两个线程并发发送HTTP请求,就会出现错误

解决方案

可以通过System.Net.ServicePointManager.DefaultConnectionLimit来设置最高并发数,建议不要超过1024

System.Net.ServicePointManager.DefaultConnectionLimit = 512;

子线程访问游戏物体,出现异常

描述

多线程编程时,子线程回调需要访问游戏物体,但是Unity只有主线程允许访问游戏物体

解决方案

SynchronizationContext.Current代表主线程

子线程可通过SynchronizationContext.Current.Post(SendOrPostCallback d, object state)向主线程通信,让主线程执行具体的逻辑,下面封装了几个快速通信至主线程回调的函数,可以直接使用

/// <summary>
/// 主线程
/// </summary>
private SynchronizationContext _mainThreadSynContext;

...
_mainThreadSynContext = SynchronizationContext.Current;     //需要在主线程内赋值
...

/// <summary>
/// 通知主线程回调
/// </summary>
private void PostMainThreadAction(Action action)
{
    _mainThreadSynContext.Post(new SendOrPostCallback((o) =>
    {
        Action e = (Action)o.GetType().GetProperty("action").GetValue(o);
        if (e != null) e();
    }), new { action = action });
}
private void PostMainThreadAction<T>(Action<T> action, T arg1)
{
    _mainThreadSynContext.Post(new SendOrPostCallback((o) =>
    {
        Action<T> e = (Action<T>)o.GetType().GetProperty("action").GetValue(o);
        T t1 = (T)o.GetType().GetProperty("arg1").GetValue(o);
        if (e != null) e(t1);
    }), new { action = action, arg1 = arg1 });
}
public void PostMainThreadAction<T1, T2>(Action<T1, T2> action, T1 arg1, T2 arg2)
{
    _mainThreadSynContext.Post(new SendOrPostCallback((o) =>
     {
         Action<T1, T2> e = (Action<T1, T2>)o.GetType().GetProperty("action").GetValue(o);
         T1 t1 = (T1)o.GetType().GetProperty("arg1").GetValue(o);
         T2 t2 = (T2)o.GetType().GetProperty("arg2").GetValue(o);
         if (e != null) e(t1, t2);
     }), new { action = action, arg1 = arg1, arg2 = arg2 });
}
相关文章
|
5天前
|
安全 Java 调度
Java编程时多线程操作单核服务器可以不加锁吗?
Java编程时多线程操作单核服务器可以不加锁吗?
18 2
|
9天前
|
Java 调度 开发者
Java中的多线程编程:从基础到实践
本文旨在深入探讨Java多线程编程的核心概念和实际应用,通过浅显易懂的语言解释多线程的基本原理,并结合实例展示如何在Java中创建、控制和管理线程。我们将从简单的线程创建开始,逐步深入到线程同步、通信以及死锁问题的解决方案,最终通过具体的代码示例来加深理解。无论您是Java初学者还是希望提升多线程编程技能的开发者,本文都将为您提供有价值的见解和实用的技巧。
15 2
|
11天前
|
Java 数据处理
Java中的多线程编程:从基础到实践
本文旨在深入探讨Java中的多线程编程,涵盖其基本概念、创建方法、同步机制及实际应用。通过对多线程基础知识的介绍和具体示例的演示,希望帮助读者更好地理解和应用Java多线程编程,提高程序的效率和性能。
19 1
|
15天前
|
Java
深入理解Java中的多线程编程
本文将探讨Java多线程编程的核心概念和技术,包括线程的创建与管理、同步机制以及并发工具类的应用。我们将通过实例分析,帮助读者更好地理解和应用Java多线程编程,提高程序的性能和响应能力。
20 4
|
23天前
|
Java 调度 开发者
Java并发编程:深入理解线程池
在Java的世界中,线程池是提升应用性能、实现高效并发处理的关键工具。本文将深入浅出地介绍线程池的核心概念、工作原理以及如何在实际应用中有效利用线程池来优化资源管理和任务调度。通过本文的学习,读者能够掌握线程池的基本使用技巧,并理解其背后的设计哲学。
|
15天前
|
安全 Java 调度
Java 并发编程中的线程安全和性能优化
本文将深入探讨Java并发编程中的关键概念,包括线程安全、同步机制以及性能优化。我们将从基础入手,逐步解析高级技术,并通过实例展示如何在实际开发中应用这些知识。阅读完本文后,读者将对如何在多线程环境中编写高效且安全的Java代码有一个全面的了解。
|
25天前
|
API Python
探索Python中的多线程编程
探索Python中的多线程编程
39 5
|
24天前
|
算法 Java 数据处理
Java并发编程:解锁多线程的力量
在Java的世界里,掌握并发编程是提升应用性能和响应能力的关键。本文将深入浅出地探讨如何利用Java的多线程特性来优化程序执行效率,从基础的线程创建到高级的并发工具类使用,带领读者一步步解锁Java并发编程的奥秘。你将学习到如何避免常见的并发陷阱,并实际应用这些知识来解决现实世界的问题。让我们一起开启高效编码的旅程吧!
|
4天前
|
Java
COMATE插件实现使用线程池高级并发模型简化多线程编程
本文介绍了COMATE插件的使用,该插件通过线程池实现高级并发模型,简化了多线程编程的过程,并提供了生成结果和代码参考。
|
26天前
|
Java 开发者
Java中的多线程编程基础与实战
【9月更文挑战第6天】本文将通过深入浅出的方式,带领读者了解并掌握Java中的多线程编程。我们将从基础概念出发,逐步深入到代码实践,最后探讨多线程在实际应用中的优势和注意事项。无论你是初学者还是有一定经验的开发者,这篇文章都能让你对Java多线程有更全面的认识。
22 1