Job System
前言
多半内容都是文章内容加上自身理解,受限于时间以及自身能力水平,谨慎参考
首先需要3个Unity的官方包,请打开Unity的预览包选项,不然很多预览阶段的包你是无法在包管理器中看到的,具体如图
随后在Package Manager中安装Jobs、Burst以及Mathematics
其中前两个可以直接在包管理器中搜到,而数学库Mathematics需要通过gir url安装,具体如图
在出现的搜索栏中输入com.unity.mathematics@1.1并点击Add就可以添加数学库了
顺带一提,Burst似乎是随着Jobs一起安装的,目前好像不需要额外安装了,具体关系我没有研究,反正保证这三个包都装上了就行
关于Job System
Jobs System(任务系统)在我看来就是一种安全的多线程使用方式,不过目前看来使用方法也比较局限,正常情况下咱们PC端都是多核心的CPU了(这都2021年了),但是很多时候做游戏写C#脚本很少会直接调用多线程,一是我没那水平,二是多线程也容易翻车…
Jobs就是Unity官方提供的一种操作多线程的方式…虽然说必须得按照官方给的”模板“来写代码,有点难受,但不论如何,用起来是很方便快捷的,也很安全…这就是我对Jobs的理解…
Starter
打开Introduction to Job System Starter项目,并打开主场景,是一个水面,本节的目的就是通过移动水面的顶点来制造波浪,这个模型的顶点还是不少的,如果通过传统的遍历顶点的方法,一个一个移动,那可想而知FPS肯定是非常之低的…
打开脚本WaveGenerator.cs
其中[Header(“Wave Parameters”)]下的3个参数用于生成随机数,也就是随机波浪,并不关键
而[Header(“References and Prefabs”)]下的参数则是对于波浪的引用,也不是关键,等下直接拖Prefab进去就行了
首先添加两个参数
NativeArray<Vector3> waterVertices; // 顶点
NativeArray<Vector3> waterNormals; // 法线
NativeArray是Job System所提供的的特殊容器类型,为什么多线程不安全?一个原因就是各个线程之间同时访问某个数据时,容易引起竞争,从而导致很多的BUG,而这个NativeArray类型的目的就是为了防止竞争导致的问题,Job System提供了以下几种容器类型:
- NativeList 一个可以调整长度的数组(Job版List)
- NativeHashMap Job版HasMap
- NativeMultiHashMap 一个键可以对应多个值
- NativeQueue Job版Queue
当两个Job同时写入某一个相同的NativeContainer时,系统会自动报错,保证了线程安全
这里还有2个点需要注意
一、不可以向Native容器中传入引用类型的数据,那会导致线程安全问题,所以只能传值类型
二、如果某一个Job线程只需要访问Native容器,而不需要写入数据,那需要将容器标记为[ReadOnly],下文会提到
在Start()中初始化
private void Start()
{
waterMesh = waterMeshFilter.mesh;
waterMesh.MarkDynamic(); // 1
waterVertices =
new NativeArray<Vector3>(waterMesh.vertices, Allocator.Persistent); // 2
waterNormals =
new NativeArray<Vector3>(waterMesh.normals, Allocator.Persistent);
}
-
让顶点能够被动态修改,和Job无关
-
初始化Native容器,第一个参数是数据,第二个参数是分配方式,有3种分配方式,如下:
目前来看我还是喜欢第三种,毕竟生命周期长,随时能用…前两种分配方式的作用没有搞明白…
在OnDestroy()中销毁
private void OnDestroy()
{
waterVertices.Dispose();
waterNormals.Dispose();
}
就当做是必须操作就好…
实现Job System
Job本质是一个一个的Struct,每一个Struct就是一个小的Job,但Job必须实现自以下3个接口之一:
- IJob,标准任务,可以和其他所有定义的Job并行工作
- IJobParallelFor,并行任务,可以对某个Native容器中的所有数据并行的执行某个操作
- IJobParallelForTransform,类似于第二种,但专门为Transform类型的Native容器工作
我们的目的是动态修改所有的顶点数据,那肯定是采用第二种方式了,顶点数据不涉及Transform哦!
定义如下结构体
private struct UpdateMeshJob : IJobParallelFor
{
// 1
public NativeArray<Vector3> vertices;
// 2
[ReadOnly]
public NativeArray<Vector3> normals;
// 3
public float offsetSpeed;
public float scale;
public float height;
// 4
public float time;
// 5
private float Noise(float x, float y)
{
float2 pos = math.float2(x, y);
return noise.snoise(pos);
}
// 6
public void Execute(int i)
{
// 7
if (normals[i].z > 0f)
{
// 8
var vertex = vertices[i];
// 9
float noiseValue =
Noise(vertex.x * scale + offsetSpeed * time, vertex.y * scale +
offsetSpeed * time);
// 10
vertices[i] =
new Vector3(vertex.x , vertex.y, noiseValue * height + 0.3f);
}
}
}
咱们一个个解析
-
定义一个原生容器,定义在结构体内部的容器是用于复制或者修改外部容器的数据的,此处的容器没有定义为[ReadOnly]说明这个容器会修改某个容器的数据
-
同上,但注意这里是[ReadOnly],说明这个容器只是用于访问某个主线程上的数据
P.S.这里我的理解是,可以吧定义在类内的容器理解为主线程上的数据,定义在结构体内的容器理解为子线程上的数据,子线程想要访问主线程的数据,只可以通过这样的方式进行复制,而且如果两个子线程同时在修改主线程上某个容器的数据,那么Unity会报错。
-
这些数据用于生成随机数,但我们没有初始化,因为这些数据也是可以通过外面传入进来的,由于是值类型,所以无所谓,但如果你传一个Transform进来就不行了
P.S.这里也有个疑惑,第一个容器需要定义为NativeArray可以理解,因为他会修改主线程的数据,但是第二个容器定义为了[ReadOnly]说明他只是访问数据而不修改,那我们为什么不定以为普通的List,然后从外面传入呢?事实上按我的理解,这是可行的,但是!加入这个List有1000的长度,我们想从外面拷贝一个List进来,那就需要一个一个复制,这是很浪费空间的,而NativeArray本质是一个共享指针,NativeArray之间的复制是没有消耗的!
- 不能传Time.time进来,所以就用一个参数暂存了,也是用于生成随机数的
- 生成随机数的一个方法,很常见的噪声,具体不赘述了,和Job无关
- 这个方法来自于接口IJobParallelFor,属于必须实现的接口,咱们不是在上面传入了一个NativeArray
vertices嘛?这个原生数组中包含了咱们水模型的所有顶点信息,而我们的目标就是同时修改他的顶点数据,这个Execute(i)方法代表着我们将对每一个顶点所做的操作,其中i代表着这个顶点在数组中的序号,看起来这个函数类似于一个For循环有木有?但For循环是一个一个往后执行的,而这个Execute方法会被Unity自动分配到多个线程中 - vertices是所有顶点的信息,而normals则是所有顶点的法线信息,通过这个i,我们很轻易的就能够访问到某个顶点的法线信息,这个法线可能需要一点点图形学知识,咱们不多说,这里的意思就是说,如果某个顶点是朝上的,那么我们才修改他的信息,再说的通俗一点,咱们只修改水面的上一侧,下侧不管
- 获取噪声,与Job无关,就是个随机数
- 修改顶点的数据
执行任务
上面只是定义好了任务,但还没执行,我们想每一帧都修改数据,这样才能形成连续的波浪!
先定义两个参数
JobHandle meshModificationJobHandle; // 1
UpdateMeshJob meshModificationJob; // 2
- 句柄,比较抽象我也不知道该怎么解释,可以当做是某一个任务的当前信息,拟执行了某个Job他就会返回一个句柄给你,里面可以访问当前这个Job的执行情况,多数情况下我们需要等一个Job执行完成,才执行下一个Job,句柄的作用就是等待
- 就是上面定义的那个Job结构体
在Update()中执行任务
private void Update()
{
// 1
meshModificationJob = new UpdateMeshJob()
{
vertices = waterVertices,
normals = waterNormals,
offsetSpeed = waveOffsetSpeed,
time = Time.time,
scale = waveScale,
height = waveHeight
};
// 2
meshModificationJobHandle =
meshModificationJob.Schedule(waterVertices.Length, 64);
}
- 初始化任务,其中的参数就是咱们定义好的各种参数
- 执行任务,有两个参数,第一个参数的数据长度,第二个参数是并行长度,比如顶点数是128,咱们并行长度是64,那么这个Job就会被划分为2部分,分别在2个子线程中进行,这个解释很模糊,大概是这个意思,着重在理解,表述可能是错的,具体还得看官方给的线程图,很难几句话说清楚
等待任务结束
private void LateUpdate()
{
// 1
meshModificationJobHandle.Complete();
// 2
waterMesh.SetVertices(meshModificationJob.vertices);
// 3
waterMesh.RecalculateNormals();
}
- 句柄的作用就是这个,表示在这里需要等待上面Schedule()的Job结束,如果不结束我们怎么能获取新的顶点数据呢…
- 设置顶点数据
- 重新计算法线
测试一下
设置好脚本数据
点击运行,应该是有用的,此时查看一下Status
这么多的顶点还能有这个数据,也是挺喜人的,查看一下任务管理器,发现所有核心确实是都在跑的
Burst
轻松一步,瞬间超神,具体原理我没有研究,总之咱们现在给刚刚定义的Job加上一个标签
然后再运行
直接超神!
IJobParallelForTransform
上面我们还提到过一种特殊的Job,专门针对于Transform类型,因为Transform类型是引用类型,所以必须通过特定的方法实现Job化
打开脚本FishGenerator.cs,这个脚本会生成大量自由移动的小鱼
首先添加两个属性
// 1
private NativeArray<Vector3> velocities;
// 2
private TransformAccessArray transformAccessArray;
- 移动速度,和上面提到的容器是一样的,只不过储存的是Vector3类型
- 专门针对于Transform的特殊容器,只能输出Transform类型的数据,且是一个数组
在Start()中初始化
private void Start()
{
// 1
velocities = new NativeArray<Vector3>(amountOfFish, Allocator.Persistent);
// 2
transformAccessArray = new TransformAccessArray(amountOfFish);
for (int i = 0; i < amountOfFish; i++)
{
float distanceX =
Random.Range(-spawnBounds.x / 2, spawnBounds.x / 2);
float distanceZ =
Random.Range(-spawnBounds.z / 2, spawnBounds.z / 2);
// 3
Vector3 spawnPoint =
(transform.position + Vector3.up * spawnHeight) + new Vector3(distanceX, 0, distanceZ);
// 4
Transform t =
(Transform)Instantiate(objectPrefab, spawnPoint,
Quaternion.identity);
// 5
transformAccessArray.Add(t);
}
}
- 和上面没啥区别,就是初始化容器,第一个参数是鱼的数量,第二个参数是分配方式
-
也是初始化容器,不过初始化Transform容器和其他容器稍微有一点区别,只要写好大小就行
3.4. 生成随机数罢了,和Job无关,目的是随机生成鱼的位置,随便看看就行
- 把生成的随机位置添加到容器中
同样需要在OnDestroy()时销毁容器
private void OnDestroy()
{
transformAccessArray.Dispose();
velocities.Dispose();
}
然后填写好参数,测试一下
会发现随机生成了很多小鱼,但是是静止的
创建小鱼移动Job
[BurstCompile]
struct PositionUpdateJob : IJobParallelForTransform
{
public NativeArray<Vector3> objectVelocities;
public Vector3 bounds;
public Vector3 center;
public float jobDeltaTime;
public float time;
public float swimSpeed;
public float turnSpeed;
public int swimChangeFrequency;
public float seed;
public void Execute (int i, TransformAccess transform)
{
// 1
Vector3 currentVelocity = objectVelocities[i];
// 2
random randomGen = new random((uint)(i * time + 1 + seed));
// 3
transform.position +=
transform.localToWorldMatrix.MultiplyVector(new Vector3(0, 0, 1)) *
swimSpeed *
jobDeltaTime *
randomGen.NextFloat(0.3f, 1.0f);
// 4
if (currentVelocity != Vector3.zero)
{
transform.rotation =
Quaternion.Lerp(transform.rotation,
Quaternion.LookRotation(currentVelocity), turnSpeed * jobDeltaTime);
}
Vector3 currentPosition = transform.position;
bool randomise = true;
// 5
if (currentPosition.x > center.x + bounds.x / 2 ||
currentPosition.x < center.x - bounds.x/2 ||
currentPosition.z > center.z + bounds.z / 2 ||
currentPosition.z < center.z - bounds.z / 2)
{
Vector3 internalPosition = new Vector3(center.x +
randomGen.NextFloat(-bounds.x / 2, bounds.x / 2)/1.3f,
0,
center.z + randomGen.NextFloat(-bounds.z / 2, bounds.z / 2)/1.3f);
currentVelocity = (internalPosition- currentPosition).normalized;
objectVelocities[i] = currentVelocity;
transform.rotation = Quaternion.Lerp(transform.rotation,
Quaternion.LookRotation(currentVelocity),
turnSpeed * jobDeltaTime * 2);
randomise = false;
}
// 6
if (randomise)
{
if (randomGen.NextInt(0, swimChangeFrequency) <= 2)
{
objectVelocities[i] = new Vector3(randomGen.NextFloat(-1f, 1f),
0, randomGen.NextFloat(-1f, 1f));
}
}
}
}
这个Job和上面那个波浪Job很像,区别是实现自接口IJobParallelForTransform,只有这个接口可以访问Transform容器,同样也有一个Execute方法需要实现,其中i代表容器内的序号,transform则是引用
具体算法其实并不重要,主要目的就是让小鱼的Transform线性变化,如果不使用Job而在Update中写,相信很多人随便就能写了
- 第i条小鱼当前的速度
- 随机种子,需要using random = Unity.Mathematics.Random;
- 随机移动距离,会根据上面几个指定的参数决定
- 旋转
-
- 防止小鱼移出水面的范围
执行任务
同样需要先创建一个句柄以及任务
private PositionUpdateJob positionUpdateJob;
private JobHandle positionUpdateJobHandle;
在Update()中初始化并执行
private void Update()
{
// 1
positionUpdateJob = new PositionUpdateJob()
{
objectVelocities = velocities,
jobDeltaTime = Time.deltaTime,
swimSpeed = this.swimSpeed,
turnSpeed = this.turnSpeed,
time = Time.time,
swimChangeFrequency = this.swimChangeFrequency,
center = waterObject.position,
bounds = spawnBounds,
seed = System.DateTimeOffset.Now.Millisecond
};
// 2
positionUpdateJobHandle = positionUpdateJob.Schedule(transformAccessArray);
}
- 初始化任务,注意我们没并有传入一个Transform数组
- 执行任务,此时可以传入咱们的transformAccessArray
同样在LateUpdate()中等待任务结束
private void LateUpdate()
{
positionUpdateJobHandle.Complete();
}
点击运行,小鱼就动了起来!