美文网首页
【Unity/ECS】从零开始的ECS学习生活

【Unity/ECS】从零开始的ECS学习生活

作者: 江枫枫Maple | 来源:发表于2020-05-14 22:03 被阅读0次

      从OW技术分享ECS到Unity提供ECS插件,我看了两三篇关于ECS的文章,但是始终不明白这到底是个啥,最近有幸换了2019版本的unity,准备借机学习一下这个ECS。这个文章也将是我一边学一边照着其他文章(视频)敲出来的,为了方便以后自己如果真的用到了可以拿出来快速回顾一下。
      首先,我使用的unity版本为2019.3.4f1,我查了网上很多版本的关于ECS的文章,大部分都是基于2018版本写的,那个时候使用ECS插件还需要一份官方提供的XML文件,但是现在网站已经挂掉了,下载不到了。
      还好我在瞎逛github的时候,找到了一个好多星星的插件:Entitas-CSharp,然后发现这个插件已经做成了压缩包,而且有一些教学视频,我就准备通过这个插件入手,从零开始浅浅的接触一下ECS,这里我下载的是1.13.0版本的插件。

插件的全部内容

      插件的全部内容就在上面了,没有实际的代码,没有示例场景,啥都没有,只有纯插件,还好git上有相关的讲解视频。
      首先,我们需要先初始化一下插件,在工具栏中会出现一个tools页签,选择preferences可以弹出一个具体的设置界面。

工具入口 设置界面

     这里正常来说点击auto import插件就会自动帮你设置成默认的设置,但是如果你的C#Project不在默认位置的话可能会出现问题,所以要确认一下csproj的位置,然后就可以生成对应的代码了。这个代码在每次修改ECS代码或者新增代码的时候都需要重新生成。工具中有提供生成代码的快捷键,而且视频中有介绍如何使用指令生成代码或是在你的IDE中集成生成代码的功能(视频中使用的rider,我也不确定其他IDE可不可以,我只是来学学ECS,快捷生成代码还是以后再学习吧)
     然后!!我们就可以来编写一些简单的ECS代码了

1.编写第一段Entity,Component代码

     接下来,我们就要开始编写第一段ECS代码了。首先新建一个C#脚本(我是跟着视频做的,新建了一个HealthComponent脚本),然后填充第一个属性,就像下面这样:

HealthComponent

     第一个脚本就已经写好了,接下来回到Unity,运行一下刚才tools工具栏下的generator生成相关代码:

generated代码
public partial class GameEntity {

    public HealthComponent health { get { return (HealthComponent)GetComponent(GameComponentsLookup.Health); } }
    public bool hasHealth { get { return HasComponent(GameComponentsLookup.Health); } }

    public void AddHealth(float newValue) {
        var index = GameComponentsLookup.Health;
        var component = (HealthComponent)CreateComponent(index, typeof(HealthComponent));
        component.value = newValue;
        AddComponent(index, component);
    }

    public void ReplaceHealth(float newValue) {
        var index = GameComponentsLookup.Health;
        var component = (HealthComponent)CreateComponent(index, typeof(HealthComponent));
        component.value = newValue;
        ReplaceComponent(index, component);
    }

    public void RemoveHealth() {
        RemoveComponent(GameComponentsLookup.Health);
    }
}

     可以看到生成的HealthComponent脚本中包含了三个函数:AddHealth,ReplaceHealth,RemoveHealth。后面关于实体的操作应该也是通过这三个函数进行的。

     下面我们就可以使用代码创建实体了,首先我们需要先创建一个Contexts对象,然后通过这个对象进行创建Entity,创建出来的Entity可以拥有我们之前编写的组件(component):

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

public class GameController : MonoBehaviour
{
    // Start is called before the first frame update
    void Start()
    {
        var context = new Contexts();
        var e = context.game.CreateEntity();
        //给实体赋予组件功能
        e.AddHealth(100);
    }

}

     进行到这里,我们的第一段代码就已经写完了,在一个空场景中挂上这个GameController组件后运行游戏就可以在场景中看到这样的结构:

运行结果

     这就表示我们拥有一个实体,这个实体有一个组件列表,其中有一个组件名字叫Health,其中Health的值是100。这和我们刚才的代码完全吻合!
     这里我们可以发现实体被分为两大类,分别是:game和input,这个是在上面的jenny设置面板中设置的:

contexts

     在contexts中我们可以查看这个模块中有多少实体,包括正在使用的和被回收放在池子中的实体(Reusable entities),当实体被回收时会自动放入回收池中供下次使用:

contexts

2.编写第一段System代码

     上面我们已经写了ECS中的EC模块,接下来再写一段S模块的代码,今天就可以打完收工了!和Component类似,我们需要再新建一个脚本来编写System:

using Entitas;

public sealed class LogHealthSystem : IExecuteSystem
{
    readonly IGroup<GameEntity> _entities;
    public LogHealthSystem(Contexts contexts) {
        _entities = contexts.game.GetGroup(GameMatcher.Health);
    }
    public void Execute()
    {
        foreach (var e in _entities)
        {
            UnityEngine.Debug.Log(e.health.value);
        }
    }
}

     这个System的作用是把health的值打印出来,首先这个类继承自IExecuteSystem,那么需要实现Execute接口,也就是这个System的具体执行操作。然后我们可以通过函数的构造函数拿到Context(感觉这个Context是一个管理类,其中包括了game和input两个大类一样的)。然后可以通过GetGroup函数可以拿到有对应组件的实体列表。然后我们就可以在excute函数执行我们想要做的事情了。
     此外,之前在GameController中获取context是通过实例化一个新的context,除此之外还可以通过Contexts.sharedInstance拿到全局通用的Contexts。接下来我们在这里创建一个新的System,并把刚才的context传入构造函数并执行。

    void Start()
    {
        var context = Contexts.sharedInstance;
        var e = context.game.CreateEntity();
        e.AddHealth(100);
        var system = new LogHealthSystem(context);
        system.Execute();
    }

     这样我们就可以在自己想要执行System的时候调用对应System的excute就可以执行对应的系统。

3.更多的System系统

1.反应式System

     如果你不想通过执行某个函数来调用一个系统,而是希望当实体的组件内的值变化时调用一个系统,我们可以使用一种反应式的System实现:

using Entitas;
using System.Collections.Generic;

public sealed class LogHealthSystem : ReactiveSystem<GameEntity>
{
    public LogHealthSystem(Contexts contexts) : base(contexts.game) { }
    protected override void Execute(List<GameEntity> entities)
    {
        foreach (var e in entities)
        {
            UnityEngine.Debug.Log(e.health.value);
        }
    }

    protected override bool Filter(GameEntity entity)
    {
        return true;
    }

    protected override ICollector<GameEntity> GetTrigger(IContext<GameEntity> context)
    {
        return context.CreateCollector(GameMatcher.Health);
    }
}

     和刚才的System差不多,不过这里是通过GetTrigger获取你想要关心的组件。遗憾的是,这种反应式的组件还是没法做到自行反应,我们还是需要在update函数中执行调用函数才能达到我们想要的效果:

public class GameController : MonoBehaviour
{
    LogHealthSystem system;
    // Start is called before the first frame update
    void Start()
    {
        var context = Contexts.sharedInstance;
        var e = context.game.CreateEntity();
        e.AddHealth(100);
        system = new LogHealthSystem(context);
        
    }
    private void Update()
    {
        system.Execute();
    }

}
2.初始化式System

     这个就不用说太多废话了,就是可以用来初始化或者初始化某些实体的System:

using Entitas;

public sealed class CreatePlayerSystem : IInitializeSystem
{
    readonly Contexts _contexts;
    public CreatePlayerSystem(Contexts contexts) {
        _contexts = contexts;
    }
    public void Initialize()
    {
        var e = _contexts.game.CreateEntity();
        e.AddHealth(100);

    }
}

     调用起来和其他的System一样,在GameController里实例化一个系统,然后调用初始化函数:

public class GameController : MonoBehaviour
{
    LogHealthSystem system;
    CreatePlayerSystem createPlayerSystem;
    // Start is called before the first frame update
    void Start()
    {
        var context = Contexts.sharedInstance;
        system = new LogHealthSystem(context);
        createPlayerSystem = new CreatePlayerSystem(context);
        createPlayerSystem.Initialize();

    }
    private void Update()
    {
        system.Execute();
    }

}

3.System的根节点

     说白了就是一个所有System的集合:

public sealed class SystemRoot : Feature
{
    public SystemRoot(Contexts contexts) {
        Add(new CreatePlayerSystem(contexts));
        Add(new LogHealthSystem(contexts));
    }
}

     但是直接调用这个集合的初始化或者执行函数,其中的所有系统都可以执行对应函数,所以还是很方便的:

public class GameController : MonoBehaviour
{
    SystemRoot _systems;
    void Start()
    {
        _systems = new SystemRoot(Contexts.sharedInstance);
        _systems.Initialize();

    }
    private void Update()
    {
        _systems.Execute();
    }
}

相关文章

网友评论

      本文标题:【Unity/ECS】从零开始的ECS学习生活

      本文链接:https://www.haomeiwen.com/subject/yqwpnhtx.html