访问官方
网址:https://github.com/neuecc/UniRx/
UniRx(Unity反应式扩展)是.NET 反应式扩展的重新实现。官方 Rx 实现很棒,但在 Unity 上不起作用,并且存在 iOS IL2CPP 兼容性问题。此库修复了这些问题,并为 Unity 添加了一些特定的实用程序。支持的平台是 PC/Mac/Android/iOS/WebGL/WindowsStore/etc 和库。
通常,Unity 中的网络操作需要使用 WWW 和 Coroutine。也就是说,出于以下(和其他)原因,使用 Coroutine 并不是异步操作的良好做法:
协程不能返回任何值,因为其返回类型必须是 IEnumerator。
协程无法处理异常,因为 yield return 语句不能用 try-catch 构造括起来。
这种缺乏可组合性会导致操作紧密耦合,这通常会导致巨大的整体式 IEnumerator。
Rx 治愈了那种 "异步忧郁症"。Rx 是一个库,用于使用可观察集合和 LINQ 样式的查询运算符编写异步程序和基于事件的程序。
游戏循环(每次 Update、OnCollisionEnter 等)、传感器数据(Kinect、Leap Motion、VR Input 等)都是事件类型。Rx 将事件表示为反应序列,这些序列既易于组合,又通过使用 LINQ 查询运算符支持基于时间的操作。
Unity 通常是单线程的,但 UniRx 有助于对联接、取消、访问游戏对象等进行多线程处理。
UniRx 使用 uGUI 帮助 UI 编程。所有 UI 事件(clicked、valuechanged 等)都可以转换为 UniRx 事件流。
Unity 从 2017 年开始通过 C# 升级支持 async/await,UniRx 系列项目提供更轻量级、更强大的 async/await 与 Unity 的集成。请参阅 Cysharp/UniTask。
学习的代码
csharp
using UniRx;
using UnityEngine;
public class NewBehaviourScript : MonoBehaviour
{
// 定义一个int类型的信号
public static ReactiveProperty<int> id = new ReactiveProperty<int>(9);
// 定义一个int float 类型的信号
public static ReactiveProperty<(int id, float value)> hello = new ReactiveProperty<(
int id,
float value
)>((0, 1.0f));
// 定义一个状态变量,判断状态是否会跟随变化
public string m_str_state = "hello world";
// 定义通知类型
public static ScheduledNotifier<float> s_notifier = new ScheduledNotifier<float>();
// Start is called before the first frame update
void Start()
{
// 监听信号 重点在后面的addto,这样可以绑定到具体的对象,这样当对象注销的时候这个也会注销掉.
_ = id.Subscribe(
(value) =>
{
Debug.Log(value);
m_str_state = $"fuck_{id.Value}";
}
)
.AddTo(this);
// 监听并且有状态的传递 , 这个状态值是不可变的.
_ = hello
.SubscribeWithState(
m_str_state,
(value, value1) =>
{
Debug.Log(
$"id = {value.id} | value = {value.value} | state = {value1} | m_str_state = {m_str_state}"
);
}
)
.AddTo(this);
// 监听
_ = hello
.Subscribe(
(value) =>
{
Debug.Log($" 值发生改变的时候{value.id} {value.value}");
},
(e) =>
{
Debug.Log($" 发生错误{e.Message}");
},
() =>
{
Debug.Log($" 值赋值完毕");
}
)
.AddTo(this);
Debug.Log(id.Value);
Debug.Log($"初始化得到的值为{hello.Value.id} {hello.Value.value}");
}
// Update is called once per frame
private void FixedUpdate()
{
// 开始测试
id.Value = id.Value + 1;
hello.Value = (id: hello.Value.id - 1, value: hello.Value.value + 0.1f);
}
}