【Unity3D开发】简单工厂模式

本文同时发布至我的个人博客,点击进入我的个人博客阅读。本博客供技术交流与经验分享,可自由转载。转载请在评论区或私信简单通知,感谢!

游戏需求

  • 实现点击效果。
  • Plane 或其他物体做地面, tag 为Finish
  • 点击地面后,出现一个圆形攻击标记,两秒后自动消失。注意:该攻击标记不能挡住点击。(Primitive Objects / Cylinder)
  • 请使用一个简单工厂创建、管理这些的标记,并自动收回这些标记(注意,这些对象创建后,放在列表内,不必释放)。

效果展示:

实现过程

(一)CameraPlane 的基本配置

​ 设置好Camera的位置与角度,并设置视图为正交,如下图:

​ 为方便显示,创建Plane后再添加一个Material,使其变为蓝色,如下图:

(二)定义并实现EsayFactory

​ 首先创建一个继承MonoBehaviourBasecode类,自定义一个namespace并引用,在该namespace定义并实现EsayFactory类。

EsayFactory类为工厂类,它负责管理Cylinder的创建、停用和重复使用。重复使用的实现主要通过讲使用完毕的Cylinder加入一个List中,在需要时重新拿出使用的方法来实现。这样做的好处是,避免了重复地创建和销毁游戏对象,能减少游戏的内存消耗。

​ 需要注意的是:严格控制单实例,避免用户创建多个EsayFactory对象。

public class EsayFactory : System.Object
{
    public List<GameObject> markList = new List<GameObject>();
    private static EsayFactory _factory;
    private Camera _camera;
    
    //控制单实例
    public static EsayFactory GetInstance()
    {
        if (_factory == null)
        {
            _factory = new EsayFactory();
            _factory._camera = Camera.main;
        }
        return _factory;
    }

    public void placeAttackMark(Vector3 target)
    {
        Ray _ray = _camera.ScreenPointToRay(target);
        RaycastHit _hit;
        if (Physics.Raycast(_ray, out _hit))
        {
            if (_hit.collider.gameObject.tag.Contains("Finish"))
            {
                GameObject attackMark;
                //List中若无可用对象则创建新对象
                if (markList.Count == 0)
                {
                    attackMark = GameObject.CreatePrimitive(PrimitiveType.Cylinder);
                    //避免Cylinder挡住点击
                    attackMark.GetComponent<Collider>().enabled = false;
                }
                //若有可用对象则拿出重用
                else
                {
                    attackMark = markList[0];
                    markList.RemoveAt(0);
                }
            attackMark.transform.SetParent(_hit.collider.gameObject.transform);
            attackMark.transform.position = _hit.point;
              //添加一个自定义的实现了延时2s删除功能的组件
            DelayToDelete _delay = attackMark.AddComponent<DelayToDelete>();
            _delay.delete(markList, attackMark, 2.0f);
        }
    }
}

(三)实现延时删除

DelayToDelete继承MonoBehaviour类,实现了两个方法:私有方法delayTo()实现了具体延时功能的实现,公有方法delete()实现了具体的2秒后删除的功能需求。这里采用了Action类来代替Invoke()方法,其好处是拓展性更强,可读性也更高。StartCoroutineMonoBehaviour中的一个调用协同程序的方法,通过使用 yield return new WaitForSeconds(delaySeconds);可以实现延时一定实现后执行。

public class DelayToDelete : MonoBehaviour
{
    public void delete(List<GameObject> markList, GameObject attackMark, float  delaySeconds)
    {
        StartCoroutine(delayTo(() =>
        {
          //移出游戏范围并加入List中等待重用
            attackMark.transform.position = new Vector3(0f, -99f, 0f);
            markList.Add(attackMark);
        }
        , 2.0f));
    }

    private IEnumerator delayTo(Action action, float delaySeconds)
    {
      //延时执行action();
        yield return new WaitForSeconds(delaySeconds);
        action();
    }
}

(四)完成Basecode

Basecode类主要负责每帧检查用户是否做了点击事件,若点击则调用placeAttackMark方法:

public class Basecode : MonoBehaviour {
    // Update is called once per frame
    void Update () {
        if (Input.GetMouseButtonDown(0))
        {
            Vector3 point = Input.mousePosition;
            EsayFactory.GetInstance().placeAttackMark(point);
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,292评论 19 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 174,935评论 25 709
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 47,014评论 6 342
  • 重读《向死而生》,心有所感 向死而生,大概只有在鬼门关走过一遭的人才会真正明白生的可贵和如何过好这一生吧,才会体会...
    你就是我最美的未来阅读 2,405评论 0 0
  • 你说过,但你没做。 你没说,并也没做。 你说了,并且做了。 你没说,但你做了
    风与光阅读 1,215评论 0 1