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

UniTask 异步任务

文章目录

  • 前言
  • 一、UniTask是什么?
  • 二、使用步骤
  • 三、常用的UniTask API和示例
    • 1.编写异步方法
    • 2.处理异常
    • 3.延迟执行
    • 4.等待多个UniTask或者一个UniTas完成
    • 5.异步加载资源示例
    • 6.手动控制UniTask的完成状态
    • 7.UniTask.Lazy延迟任务的创建
    • 8.后台线程切换Unity主线程
    • 9.不要返回值
    • 10.缓存UniTask的结果
    • 11.指定异步操作阶段执行
    • 12.创建完成的UniTask
    • 13.UniTask.ToCoroutine转换协程
    • 14.异步迭代集合
    • 15. 异步判断
    • 16. uniTask的取消
    • 17.网络请求加载图片
    • 18.异步判断按钮是否被双击
    • 19.每次点击修改按钮不同的文本
  • 四、 总结

前言

随着Unity游戏开发的流行,异步编程变得越来越重要。UniTask是一个轻量级的异步编程库,它为Unity开发者提供了更高效的编程方式。本文将介绍UniTask的基础内容和一些常用的API,以及它们的使用示例。


一、UniTask是什么?

UniTask是一个专为Unity设计的异步编程库,它提供了类似于C#的Taskasync/await的功能,但是更加适合Unity的执行模型和生命周期。

二、使用步骤

UniTaskGit地址
UniTask中文文档地址

首先,需要在Unity项目中安装UniTask。可以通过Unity Package Manager(UPM)安装,或者直接从GitHub下载源码

本文介绍从GitHub下载
在这里插入图片描述

在这里插入图片描述


三、常用的UniTask API和示例

1.编写异步方法

使用UniTask时,可以像编写普通的异步方法一样使用async关键字和UniTask返回类型

// 示例:一个简单的异步加载场景的方法
async UniTask LoadSceneAsync(string sceneName)
{// 使用UniTask的场景加载方法await SceneManager.LoadSceneAsync(sceneName).ToUniTask();// 场景加载完成后的操作Debug.Log("Scene loaded");
}

2.处理异常

UniTask也支持异常处理,可以使用try/catch块来捕获异步方法中的异常

// 示例:带有异常处理的异步方法
async UniTask DoSomethingAsync()
{try{// 可能会抛出异常的操作await SomeRiskyOperation().ToUniTask();}catch (Exception ex){// 处理异常Debug.LogError(ex.Message);}
}

3.延迟执行

UniTask.Delay: 延迟执行,类似于Task.Delay

// 延迟执行示例
async UniTask DelayedOperation()
{await UniTask.Delay(TimeSpan.FromSeconds(5));Debug.Log("Operation executed after 5 seconds");
}

4.等待多个UniTask或者一个UniTas完成

UniTask.WhenAll: 等待多个UniTask完成,类似于Task.WhenAll

// 并行执行多个任务示例
async UniTask RunMultipleTasks()
{// 创建多个任务var task1 = DoTask1Async();var task2 = DoTask2Async();// 等待所有任务完成await UniTask.WhenAll(task1, task2);Debug.Log("全部任务完成");
}
async void RunSingleTask()
{UniTask task1 = UniTask.WaitUntil(() => isClick1);UniTask task2 = UniTask.WaitUntil(() => isClick2);await UniTask.WhenAny(task1, task2);Debug.Log("一个任务完成");
}

5.异步加载资源示例

// 异步加载资源示例
async UniTask<Texture> LoadTextureAsync(string url)
{using (var request = UnityWebRequestTexture.GetTexture(url)){await request.SendWebRequest().ToUniTask();if (request.isNetworkError || request.isHttpError){Debug.LogError(request.error);return null;}return DownloadHandlerTexture.GetContent(request);}
}

6.手动控制UniTask的完成状态

UniTaskCompletionSource允许你手动控制UniTask的完成状态。这在需要等待非异步方法完成的情况下非常有用。

UniTaskCompletionSource<bool> ucs = new UniTaskCompletionSource<bool>();// 在某个事件发生时完成任务
void OnSomeEvent()
{ucs.TrySetResult(true);
}// 等待事件完成
async UniTask WaitSomeEventAsync()
{await ucs.Task;Debug.Log("Event occurred");
}

7.UniTask.Lazy延迟任务的创建

UniTask.Lazy可以延迟任务的创建,直到真正需要执行任务时才会创建。

// 示例:延迟创建任务
UniTask<string> lazyTask = UniTask.Lazy(async () =>
{await UniTask.Delay(1000);return "Result after delay";
});// 在需要结果时等待任务
string result = await lazyTask;
Debug.Log(result);

8.后台线程切换Unity主线程

在需要从后台线程切换回Unity主线程时,可以使用UniTask.SwitchToMainThread

// 示例:在后台线程上执行操作,然后切换回主线程
async UniTask DoOperationAndSwitchBack()
{// 在后台线程上执行耗时操作await UniTask.Run(() => SomeHeavyOperation());// 切换回主线程await UniTask.SwitchToMainThread();// 在主线程上执行Unity相关操作Debug.Log("Back on main thread");
}

9.不要返回值

当你不需要关心异步方法的返回值时,可以使用UniTaskVoid。这对于触发事件或执行不需要返回结果的后台操作非常有用。

// 示例:一个不返回任何结果的异步方法
async UniTaskVoid PerformBackgroundOperation()
{// 执行一些后台操作await UniTask.Delay(1000);Debug.Log("Background operation completed");
}

10.缓存UniTask的结果

如果你想要缓存一个UniTask的结果以供后续使用,可以使用UniTask<T>.Preserve方法。

// 示例:缓存异步操作的结果
UniTask<int> originalTask = CalculateValueAsync();
UniTask<int> preservedTask = originalTask.Preserve();// 在不同的地方使用缓存的结果
int result1 = await preservedTask;
int result2 = await preservedTask;

11.指定异步操作阶段执行

PlayerLoopTiming枚举允许你指定异步操作应该在Unity的哪个阶段执行。这可以帮助你更精确地控制代码的执行时间。

// 示例:在FixedUpdate阶段执行异步操作
async UniTask PerformOperationInFixedUpdate()
{await UniTask.Yield(PlayerLoopTiming.FixedUpdate);// 在FixedUpdate阶段执行的代码Debug.Log("This is executed in FixedUpdate");
}

12.创建完成的UniTask

当你已经有了结果,并且想要创建一个已经完成的UniTask时,可以使用UniTask.FromResult

// 示例:创建一个已经完成的UniTask
UniTask<int> completedTask = UniTask.FromResult(42);

13.UniTask.ToCoroutine转换协程

如果你需要将UniTask与旧的协程系统兼容,可以使用UniTask.ToCoroutine将其转换为协程。

// 示例:将UniTask转换为协程
IEnumerator MyCoroutine()
{yield return LoadSceneAsync("MyScene").ToCoroutine();
}

14.异步迭代集合

UniTask支持异步枚举器,这允许你使用await foreach来异步迭代集合。

// 示例:使用异步枚举器迭代集合
async UniTask ProcessItemsAsync(IAsyncEnumerable<int> items)
{await foreach (var item in items){Debug.Log(item);}
}

15. 异步判断

UniTask.WaitUntilUniTask.WaitWhile这些方法允许你等待直到某个条件为真或为假。

// 示例:等待直到条件为真
async UniTask WaitUntilConditionMet()
{await UniTask.WaitUntil(() => someCondition);Debug.Log("Condition met");
}// 示例:等待直到条件为假
async UniTask WaitWhileConditionMet()
{await UniTask.WaitWhile(() => someCondition);Debug.Log("Condition no longer met");
}

16. uniTask的取消

通过 CancellationTokenSource完成UniTask的取消


public class Test : MonoBehaviour
{private CancellationTokenSource cts;void Start(){cts = new CancellationTokenSource();StartCount();cts.CancelAfter(TimeSpan.FromSeconds(10));}private async UniTask<int> Count(int count, CancellationToken token){for (int i = 0; i < count; i++){Debug.Log(i);await UniTask.Delay(TimeSpan.FromSeconds(1), cancellationToken: token);}return 0;}private async void StartCount(){try{await Count(100, cts.Token);}catch (OperationCanceledException){Debug.Log("计数取消");}}
}

17.网络请求加载图片

private async UniTaskVoid UnityTaskTest()
{var webRequest = UnityWebRequestTexture.GetTexture(url);var result = await webRequest.SendWebRequest();var texture = ((DownloadHandlerTexture)webRequest.downloadHandler).texture;Sprite sprite = Sprite.Create(texture,new Rect(Vector2.zero, new Vector2(texture.width, texture.height)), new Vector2(0.5f, 0.5f));image2.sprite = sprite;}

18.异步判断按钮是否被双击

using System;
using System.Collections;
using System.Collections.Generic;
using System.Threading;
using Cysharp.Threading.Tasks;
using UnityEngine;
using UnityEngine.UI;public class Test : MonoBehaviour
{public Button btn1;void Start(){CheckClickInternal(this.GetCancellationTokenOnDestroy(), 1).Forget();}/// <summary>/// 异步判断按钮是否被双击/// </summary>/// <param name="token"></param>private async UniTaskVoid CheckClickInternal(CancellationToken token, float time){while (true){var firstClick = btn1.OnClickAsync(token);await firstClick;var secondClick = btn1.OnClickAsync(token);//判断第二次点击和等待time时间谁先执行,如果是secondClick返回0,如果不是返回1int index = await UniTask.WhenAny(secondClick,UniTask.Delay(TimeSpan.FromSeconds(time), cancellationToken: token));if (index == 0){Debug.Log("被双击了");}}}
}

19.每次点击修改按钮不同的文本

using System.Collections;
using System.Collections.Generic;
using System.Threading;
using Cysharp.Threading.Tasks;
using Cysharp.Threading.Tasks.Linq;
using UnityEngine;
using UnityEngine.UI;public class Test : MonoBehaviour
{public Button btn1;// Start is called before the first frame updatevoid Start(){btn1.GetComponentInChildren<Text>().text = "杀";ChangeBtnTextClick(this.GetCancellationTokenOnDestroy()).Forget();}/// <summary>/// 每次点击修改按钮不同的文本;/// </summary>/// <param name="token"></param>async UniTask ChangeBtnTextClick(CancellationToken token){var click = btn1.OnClickAsAsyncEnumerable();await click.Take(3).ForEachAsync((_, index) =>{if (token.IsCancellationRequested) return;btn1.GetComponentInChildren<Text>().text = index switch{0 => "再杀",1 => "再补一刀",_ => ""};}, cancellationToken: token);btn1.GetComponentInChildren<Text>().text = "结束";}
}

四、 总结

Unitask 是一个优秀的异步任务库,它简化了在 Unity 中处理异步操作的方式,提高了开发效率和代码可维护性。通过本文的介绍,可以更加深入地了解 Unitask 的使用方法,并在自己的项目中应用它,从而提升游戏开发的质量和效率。

在使用 Unitask 时,需要注意以下几点:

​ 不要在 Update 方法中使用 async/await,因为 Unity 主线程是单线程的,使用异步方法可能导致性能问题。

​ 尽量避免过多的异步嵌套,以免导致代码复杂度过高。

​ 使用 try/catch 来处理异步操作中的错误,确保程序的稳定性。

相关文章:

UniTask 异步任务

文章目录 前言一、UniTask是什么&#xff1f;二、使用步骤三、常用的UniTask API和示例1.编写异步方法2.处理异常3.延迟执行4.等待多个UniTask或者一个UniTas完成5.异步加载资源示例6.手动控制UniTask的完成状态7.UniTask.Lazy延迟任务的创建8.后台线程切换Unity主线程9.不要返…...

【git分支管理策略】如何高效的管理好代码版本

目录 1.分支管理策略 2.我用的分支管理策略 3.一些常见问题 1.分支管理策略 分支管理策略就是一些经过实践后总结出来的可靠的分支管理的办法&#xff0c;让分支之间能科学合理、高效的进行协作&#xff0c;帮助我们在整个开发流程中合理的管理好代码版本。 目前有两套Git…...

css的transition详解

CSS的transition属性是一个简写属性&#xff0c;用于设置四个过渡效果属性&#xff0c;以在元素的状态改变时创建平滑的动画效果。这四个属性分别是&#xff1a; transition-property&#xff1a; 定义应用过渡效果的CSS属性名称。当指定的CSS属性改变时&#xff0c;过渡效果将…...

agent利用知识来做规划:《KnowAgent: Knowledge-Augmented Planning for LLM-Based Agents》笔记

文章目录 简介KnowAgent思路准备知识Action Knowledge的定义Planning Path Generation with Action KnowledgePlanning Path Refinement via Knowledgeable Self-LearningKnowAgent的实验结果 总结参考资料 简介 《KnowAgent: Knowledge-Augmented Planning for LLM-Based Age…...

01 React新建开发环境

https://create-react-app.dev/docs/getting-started npx create-react-app my-appJSX使用表达式嵌入 function App() {const count 100;function getSelfName() {return "SelfName"}return (<div>Hello World!<div>{This is Javascript message~!}&l…...

nginx--解决响应头带Set-Cookie导致的验证失败

解决响应头带Set-Cookie导致的验证失败 前言给nginx.conf 设置Secure配置完成后会发现cookie就不会发生变化了 前言 在用nginx做代理的时候&#xff0c;会发现nginx在访问不同ip请求的时候会带setCookie 导致后端就是放开cookie验证&#xff0c;在访问玩这个链接他更新了cooki…...

InstructGPT的流程介绍

1. Step1&#xff1a;SFT&#xff0c;Supervised Fine-Tuning&#xff0c;有监督微调。顾名思义&#xff0c;它是在有监督&#xff08;有标注&#xff09;数据上微调训练得到的。这里的监督数据其实就是输入Prompt&#xff0c;输出相应的回复&#xff0c;只不过这里的回复是人工…...

docker容器下部署hbase并在springboot中通过jdbc连接

我在windows的docker中部署了一个hbase服务&#xff0c;然后用springboot连接到此服务并访问数据。 详情可参考项目中的README.md。项目中提供了用于构建镜像的dockerfile&#xff0c;以及测试代码。 项目连接&#xff1a; https://gitee.com/forgot940629/hbase_phoenix_sprin…...

Qt——智能指针实战

目录 前言正文一、理论介绍1、QPointer2、QScopedPoint3、QSharedPoint4、QWeakPoint 二、实战演练1、QPoint2、QScopedPoint3、QSharedPointa、示例一b、示例二 4、QWeakPoint END、总结的知识与问题 参考 前言 智能指针的使用&#xff0c;对很多程序员来说&#xff0c;都算是…...

Unity Mobile Notifications推送问题

1.在部分机型点击通知弹窗进不去游戏 把这里改成自己的Activity 2.推送的时候没有横幅跟icon红点 主要是第一句话 注册的时候选项可以选择 defaultNotificationChannel new AndroidNotificationChannel(“default_channel”, “Default Channel”, “For Generic notifica…...

C++_回文串

目录 回文子串 最长回文子串 分割回文串 IV 分割回文串 II 最长回文子序列 让字符串成为回文串的最少插入次数 回文子串 647. 回文子串 思路&#xff0c;i j表示改范围内是否为回文串&#xff0c; ②倒着遍历是为了取出dp[i 1][j - 1] ③i j 只有一对&#xff0c;不会重复…...

【阅读论文】When Large Language Models Meet Vector Databases: A Survey

摘要 本调查探讨了大型语言模型&#xff08;LLM&#xff09;和向量数据库&#xff08;VecDB&#xff09;之间的协同潜力&#xff0c;这是一个新兴但迅速发展的研究领域。随着LLM的广泛应用&#xff0c;出现了许多挑战&#xff0c;包括产生虚构内容、知识过时、商业应用成本高昂…...

兼职副业大揭秘:六个潜力满满的赚钱途径

亲爱的朋友&#xff0c;你对兼职副业充满好奇与期待&#xff0c;这非常好&#xff01;在此&#xff0c;我将为你分享一些能够助你赚取额外收入的兼职副业建议。以下是六个颇具潜力的兼职副业方向&#xff0c;希望能为你的探索之路提供些许启发。 1&#xff0c;网络调查与市场洞…...

C++ Qt开发:QUdpSocket实现组播通信

Qt 是一个跨平台C图形界面开发库&#xff0c;利用Qt可以快速开发跨平台窗体应用程序&#xff0c;在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置&#xff0c;实现图形化开发极大的方便了开发效率&#xff0c;本章将重点介绍如何运用QUdpSocket组件实现基于UDP的组播通信…...

excel 表中有图片并在筛选特定行时,只显示该行的图片

建议&#xff1a;选中excel 表中某张图片&#xff0c;CtrlA&#xff0c;选中所有图片。再右键&#xff0c;在菜单中选设置对象格式 在属性里按下图设置&#xff0c; 生效之后&#xff0c;筛选某个产品的时候&#xff0c;就不会显示其他的不符合筛选条件的产品的图片了。...

【QA】MySQL多表查询详解

文章目录 前言关系型数据库中数据表之间的关系数据准备数据内容表间关系 基础查询 | 全部查询多表查询分类1 | 连接查询内连接外连接 | 左外连接外连接 | 右外连接自连接 | 自连接自连接 | 联合查询 分类2 | 子查询返回结果分类 | 标量子查询返回结果分类 | 列子查询返回结果分…...

【Entity Framework】 EF三种开发模式

【Entity Framework】 EF三种开发模式 文章目录 【Entity Framework】 EF三种开发模式一、概述二、DataBase First2.1 DataBase First简介2.2 DataBase First应用步骤2.3 DataBase First总结 三、Model First3.1 Model First简介3.2 Model First实现步骤 四、Code First4.1 Cod…...

数据分析---SQL(5)

目录 子查询单行子查询多行子查询视图(View)创建视图使用视图更新视图视图的优缺点存储过程存储过程的创建存储过程的参数存储过程的优缺点可能导致性能问题避免存储过程引入性能问题子查询 子查询是指在一个查询语句中嵌套另一个查询语句,内部的查询语句称为子查询,外部的…...

《剑指 Offer》专项突破版 - 面试题 93 : 最长斐波那契数列(C++ 实现)

题目链接&#xff1a;最长斐波那契数列 题目&#xff1a; 输入一个没有重复数字的单调递增的数组&#xff0c;数组中至少有 3 个数字&#xff0c;请问数组中最长的斐波那契数列的长度是多少&#xff1f;例如&#xff0c;如果输入的数组是 [1, 2, 3, 4, 5, 6, 7, 8]&#xff0…...

代码随想录算法训练营第五十五天|583. 两个字符串的删除操作、72. 编辑距离

583. 两个字符串的删除操作 刷题https://leetcode.cn/problems/delete-operation-for-two-strings/description/文章讲解https://programmercarl.com/0583.%E4%B8%A4%E4%B8%AA%E5%AD%97%E7%AC%A6%E4%B8%B2%E7%9A%84%E5%88%A0%E9%99%A4%E6%93%8D%E4%BD%9C.html视频讲解https://…...

StringRedisTemplate Autowired注入为空解决

如下注入方式报空指针异常&#xff1a; java.lang.NullPointerException: null Autowiredprivate StringRedisTemplate redisTemplate; 解决办法&#xff1a;查看该类上有没有加注解&#xff0c;如Component等&#xff0c;没加的话加上。 还有一种是在工具类中使用&#xff0c;…...

c语言:文件操作

1. 为什么使⽤⽂件&#xff1f; 如果没有⽂件&#xff0c;我们写的程序的数据是存储在电脑的内存中&#xff0c;如果程序退出&#xff0c;内存回收&#xff0c;数据就丢失 了&#xff0c;等再次运⾏程序&#xff0c;是看不到上次程序的数据的&#xff0c;如果要将数据进⾏持久…...

C#事件实例详解

一、什么是事件&#xff1f; 在C#中,事件(event)是一种特殊的类成员,它允许类或对象通知其他类或对象发生了某些事情。 从语法上看,事件的声明类似于字段,但它们在功能和行为上有一些重要的区别。 从技术角度来说,事件实际上是一个封装了事件订阅和取消订阅功能的委托字段。…...

零基础机器学习(3)之机器学习的一般过程

文章目录 一、机器学习一般过程1.数据获取2.特征提取3.数据预处理①去除唯一属性②缺失值处理A. 均值插补法B. 同类均值插补法 ③重复值处理④异常值⑤数据定量化 4.数据标准化①min-max标准化&#xff08;归一化&#xff09;②z-score标准化&#xff08;规范化&#xff09; 5.…...

用java做一个双色球彩票系统

代码如下&#xff1a; import java.util.Random; public class HelloWorld{public static void main(String[] args){//1、生成中奖号码 int[] arrcreateNumber();for (int i 0;i<arr.length;i) {System.out.print(arr[i]" ");}}public static int[] createNu…...

某对象存储元数据集群改造流水账

软件产品&#xff1a;某厂商提供的不便具名的对象存储产品&#xff0c;核心底层技术源自HDFS和Amazon S3&#xff0c;元数据集群采用了基于MongoDB的NOSQL数据库产品和MySQL数据库产品相结合。 该产品的元数据逻辑示意图如下&#xff1a; 业务集群现状&#xff1a;当前第3期建…...

前端理论总结(js)——filter、foearch、for in 、for of 、for的区别以及返回值

Filter&#xff1a; 用途&#xff1a;用于筛选数组中符合条件的元素&#xff0c;返回一个新数组。 返回值&#xff1a;返回一个新数组&#xff0c;包含经过筛选的元素。 Foreach&#xff1a; 用途&#xff1a;遍历数组中的每个元素&#xff0c;执行回调函数。 返回值&#x…...

【JavaEE初阶系列】——多线程案例一——单例模式 (“饿汉模式“和“懒汉模式“以及解决线程安全问题)

目录 &#x1f6a9;单例模式 &#x1f388;饿汉模式 &#x1f388;懒汉模式 ❗线程安全问题 &#x1f4dd;加锁 &#x1f4dd;执行效率提高 &#x1f4dd;指令重排序 &#x1f36d;总结 单例模式&#xff0c;非常经典的设计模式&#xff0c;也是一个重要的学科&#x…...

革新水库大坝监测:传统软件与云平台之比较

在水库大坝的监测管理领域&#xff0c;传统监测软件虽然曾发挥了重要作用&#xff0c;但在多方面显示出了其局限性。传统解决方案通常伴随着高昂的运维成本&#xff0c;需要大量的硬件支持和人员维护&#xff0c;且软件整合和升级困难&#xff0c;限制了其灵活性和扩展性。 点击…...

C++模版(基础)

目录 C泛型编程思想 C模版 模版介绍 模版使用 函数模版 函数模版基础语法 函数模版原理 函数模版实例化 模版参数匹配规则 类模版 类模版基础语法 C泛型编程思想 泛型编程&#xff1a;编写与类型无关的通用代码&#xff0c;是代码复用的一种手段。 模板是泛型编程…...