Unity面试题整理(二)

目录

Unity基于事件机制的消息系统

Unity委托

Unity工具类系列之对象池

Unity消息机制

Unity游戏客户端通用框架设计

Unity的UI层级管理框架

Unity缓存池

Unity脚本加载和编译

Unity设计模式

C#基础

Unity基础

Unity委托

Unity小型游戏寻路算法

Unity框架设计

unity游戏音效

unity的UI渲染机制

Unity自定义消息通知组件

协程原理分析

协程工作流程

外观模式

委托delegate

委托和事件

委托的使用

射线Ray的原理

游戏UI框架设计思路

物理组件Collider

虚方法,抽象类和接口的区别

资源加载


Unity基于事件机制的消息系统

引入事件机制,其实本质就是观察者模式, 事件本质又是方法的委托 (Delegate),把回调方法委托到事件管理器,
当条件达到,通过事件key来告诉时间管理器可以执行的委托方法。 

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
public interface IMessage
{
    /// 
    /// 事件类型,Key
    /// 
    int Type { get; set; }
 
    /// 
    /// 发送者
    /// 
    System.Object Sender { get; set; }
 
    /// 
    /// 参数
    /// 
    System.Object[] Params { get; set; }
 
    /// 
    /// 转字符串
    /// 
    /// 
    string ToString();
}


为什么实验int类型而非string类型作为消息的key??

接口IMessage的实现Message:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
public class Message : IMessage {
    public int Type { get; set; }
 
    public System.Object[] Params { get; set; }
 
    public System.Object Sender { get; set; }
 
    public override string ToString()
    {
        string arg = null;
        if (Params != null)
        {
            for (int i = 0; i < Params.Length; i++)
            {
                if ((Params.Length > 1 && Params.Length - 1 == i) || Params.Length == 1)
                {
                    arg += Params[i];
                }
                else
                {
                    arg += Params[i] + " , ";
                }
            }
        }
 
        return Type + " [ " + ((Sender == null) ? "null" : Sender.ToString()) + " ] " + " [ " + ((arg == null) ? "null" : arg.ToString()) + " ] ";
    }
 
    public Message Clone()
    {
        return new Message(Type, Params, Sender);
    }
 
    public Message(int type)
    {
        Type = type;
    }
 
    public Message(int type, params System.Object[] param)
    {
        Type = type;
        Params = param;
    }
 
    public Message(int type, System.Object sender, params System.Object[] param)
    {
        Type = type;
        Params = param;
        Sender = sender;
    }
}


定义消息类型:

/// 
/// 消息的类型  
/// 
public enum MessageType
{
    /// 
    /// 启动
    /// 
    START_UP = 1000,
    /// 
    /// 解压
    /// 
    UNPACK,
    /// 
    /// 更新
    /// 
    UPDATE,
    /// 
    /// 更新完成
    /// 
    UPDATE_COMPLETE,
}
 
 
/// 
/// 战斗的类型
/// 
public enum BattleEvent
{
    /// 
    /// 攻击
    /// 
    Attack = 10000,
}
 
/// 
/// 协议的类型
/// 
public enum ProtocolEvent
{
 
}


消息派发接口:IDispatcher:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
public interface IDispatcher
{
    void AddListener(int type, EventListenerDelegate listener);
 
    void RemoveListener(int type, EventListenerDelegate listener);
 
    void SendMessage(Message evt);
 
    void SendMessage(int type, params System.Object[] param);
 
    void Clear();
}

//接口的IDispatcher的实现Dispatcher:

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
 
public delegate void EventListenerDelegate(Message evt);
 
public class Dispatcher : Singleton, IDispatcher
{
    
    private Dictionary events = new Dictionary();
 
    public void AddListener(int type, EventListenerDelegate listener)
    {
        if (listener == null)
        {
            XMDebug.LogError("AddListener: listener不能为空");
            return;
        }
 
        EventListenerDelegate myListener = null;
        events.TryGetValue(type, out myListener);
        events[type] = (EventListenerDelegate)Delegate.Combine(myListener, listener);
    }
 
 
    public void RemoveListener(int type, EventListenerDelegate listener)
    {
        if (listener == null)
        {
            XMDebug.LogError("RemoveListener: listener不能为空");
            return;
        }
 
        events[type] = (EventListenerDelegate)Delegate.Remove(events[type], listener);
    }
 
    public void Clear()
    {
        events.Clear();
    }
 
    public void SendMessage(Message evt)
    {
        EventListenerDelegate listenerDelegate;
        if (events.TryGetValue(evt.Type, out listenerDelegate))
        {
            try
            {
                if (listenerDelegate != null)
                {
                    listenerDelegate(evt);
                }
            }
            catch (System.Exception e)
            {
                XMDebug.LogError("SendMessage:", evt.Type.ToString(), e.Message, e.StackTrace, e);
            }
        }
    }
 
    public void SendMessage(int type, params System.Object[] param)
    {
        EventListenerDelegate listenerDelegate;
        if (events.TryGetValue(type, out listenerDelegate))
        {
            Message evt = new Message(type, param);
            try
            {
                if (listenerDelegate != null)
                {
                    listenerDelegate(evt);
                }
            }
            catch (System.Exception e)
            {
                XMDebug.LogError("SendMessage:", evt.Type.ToString(), e.Message, e.StackTrace, e);
            }
        }
    }
 
 
    public void AddListener(MessageType type, EventListenerDelegate listener)
    {
        AddListener((int)type, listener);
    }
 
    public void AddListener(BattleEvent type, EventListenerDelegate listener)
    {
        AddListener((int)type, listener);
    }
 
    public void AddListener(ProtocolEvent type, EventListenerDelegate listener)
    {
        AddListener((int)type, listener);
    }
 
    public void RemoveListener(MessageType type, EventListenerDelegate listener)
    {
        RemoveListener((int)type, listener);
    }
 
    public void RemoveListener(BattleEvent type, EventListenerDelegate listener)
    {
        RemoveListener((int)type, listener);
    }
 
    public void RemoveListener(ProtocolEvent type, EventListenerDelegate listener)
    {
        RemoveListener((int)type, listener);
    }
 
    public void SendMessage(MessageType type, params System.Object[] param)
    {
        SendMessage((int)type, param);
    }
 
    public void SendMessage(BattleEvent type, params System.Object[] param)
    {
        SendMessage((int)type, param);
    }
 
    public void SendMessage(ProtocolEvent type, params System.Object[] param)
    {
        SendMessage((int)type, param);
    }
}

Unity委托

设计模式当中的“观察者模式”和 “委托”有着千丝万绿的关系。

观察者模式:定义对象之间的一对多关系,。这样一来,当一个对象改变状态时,它的所有依赖者会受到通知并自动更新。

c#通过委托来实现回调函数机制,回调作为有用的变成机制,广泛的运用在观察者模式中。

Unity3D的SendMessage和BroadcastMessage:

SendMessage和BroadcastMessage构建的消息系统,


Unity工具类系列之对象池

##游戏开发的过程当中首当其冲最重要的是: UI。##

工具类之对象池:

游戏UI中产生大量相同类型的物体时。如果每次创建完成都删除掉,会造成频繁的资源回收[GC]。

此时,对象池就是一种良好的设计模式,利于优化。


对象池的核心思想是:预先初始化一组可重用的实体,而不是按需销毁然后重建


我们利用对象池的核心思想就是“当我们第一次实例化好多个Item物体后,下一次打开如果界面更新就没必要再
创建第二次重复的资源物体了”,因此如果这个界面关闭的时候应该把资源临时存放在一个“池”里面。

pool:
--item
--item
--item

外表看来,就是资源保存在池中,外部表现就是item物体修改了父物体,然后关闭激活。

打开界面的时候,因为池中有资源,跳过读取资源实例化的过程。【资源加载和实例化是消耗极大性能的】

/*这里有必要强调的是,我们没必要单独去做第一次对象池的实例化,我们只需要知道,这个物体是重复的,所以这个物体我们都在
对象池中去取。而对象池也只关注,当别人调用了我的创建实例函数,我必须返回给它一个创建好的物体,而池子里面有
没有资源让对象池自己判断,如果没有则创建,如果有则直接拿出来给它,这种设计方法叫做空池触发*/


对象池存储对象【一种或者多种】:


多种:

思路:
将Queue 转换为Dictionary> 处理作为存储对象的功能。同时,
我们需要对象池可以识别多种对象,所以加入 Dictionary类型的变量存储物体Tag.

资料:

查阅:https://www.bilibili.com/read/cv54304


Unity消息机制

开发Unity阶段,大多数采用消息机制。

通过消息触发的事件:即为完成某个任务之后触发其他事件的发生,即为消息事件。

消息事件的实现过程:增加消息监听,分发消息处理。

现在封装消息系统吧:

1)首先,封装消息事件的基类: BaseEvent:

public enum EventType
{
    GAME_OVER,
    GAME_WIN,
    PAUSE,
    GAME_DATA,
}

public class BaseEvent
{
  protected HashTable params;
  protected EventType type;
  protected object sender;
  protected string eventName;
  }


封装对外使用的消息接口:

public class EventDispatcher
{
  static EventDispatcher instance;
  public static EventDispatcher GetInstance()
  {
    if(instance == null)      
    {
      instance = new EventDispatcher();
      
    }
    return instance;
  }

private Hashtable listeners = new Hashtable();


public void AddEventListener(EventType eventType,EventListenerDelegate listener)
{
  EventListenerDelegate eventListenerDelegate = this.listeners[eventType] as EventListenerDelegate;
  eventListenerDelegate = (EventListenerDelegate)Delegate.CVombine(eventListenerDelegate,listener);
  this.listeners[eventType] = eventListenerDelegate;
  
}

public void RemoveEventListener(EventType ventType,EventListenerDelegate listener)
{
  EventListenerDelegate eventListenerDelegate = this.listeners[eventType] as EventListenerDelegate;
  if(eventListenerDelegate != null)
  {
    //combine等于委托  + 
    eventListenerDelegate = (EventListenerDelegate)Delegate.Combine(eventListenerDelegate,listener);
    this.listeners[eventType] = eventListenerDelegate;
  }
  
  
  //使用枚举传递
  public void DispatchEvent(BaseEvent event)
  {
    EventListener eventListenerDelegate = this.listeners[event.Type] as EventListenerDelegate;
      if(eventListenerDelegate ! = null)
      {
        try
        {
          eventListenerDelegate(event);
        }
        catch(Exception e)
        {
          throw new Exception(string.Concat(new string[]
          {
            "Error dispatching event",
            event.Type.ToString(),
            ":",
            e.Message,
            "",
            e.StackTrace
            }),e);
          }
          
        }
      
      }
  }
  
  
  public void DispatchStringEvent(BaseEvent event)
  {
     EventListenerDelegate eventListenerDelegate = this.listeners[event.EventName] as EventListenerDelegate;
     if(eventListenerDelegate != null)
     {
        try
        {
          eventListenerDelegate(event);
        }
        catch(Exception e)
        {
          throw new Exception(string.Concat)
        }
     }
  }  
}
}

Unity游戏客户端通用框架设计


1)UI框架(UGUI + MVC)
2)消息框架
3)网络层框架(Socket + Protobuf)
4)表格数据(Protobuf)
5)资源管理(Unity5.x的AssertBundle方案)
6)热更新框架(toLua)


编写UI框架的意义:
1)打开,关闭,层级,页面跳转等管理问题集中化,将外部切换等逻辑交给UIManager
2)功能逻辑分散化,每个页面维护自身逻辑,一坨框架便于多人协同开发,不用关心跳转和显示关闭等细节
3)通用性框架能够做到简单代码复用和“项目经验”积累

资料:

泰斗破坏神2商业实战班 第1期:
海洋老师:战斗相关开发 
肖红老师:UI框架开发及网络
卡卡大神:带高性能java服务器加盟泰斗2


Unity的UI层级管理框架


UI框架的设计是任何游戏都需要做的事情,其中事件管理器(EventManager)是比较常用的UI和逻辑分离的方法。

通过注册,绑定,分发事件来控制UI界面或者游戏场景的逻辑处理。


UI和逻辑分离:

最基本的需求,不希望日后游戏更换UI需要到处修改逻辑,想要的是UI界面的负责展示。

一个UIManager来定义UI变化的逻辑,UI变化由事件驱动,所以UI界面 + UIManager + 事件分发器 就可以了。
以后换UI只需要修改UI界面就可以了。


Unity缓存池

Unity缓存池:

缓存池,顾名思义就是为了缓存而存在的。
优点:使得游戏的运行更加流畅
缺点:占用部分内存


常使用PoolManager插件,但是会导致代码较为庞大

预加载和对象缓冲池技术:


Unity脚本加载和编译

一.C#作为以Assembly【汇编集】为基本单元,dll就是一个assembly,dll之间有加载以来顺序

Assets/*.dll
Stamdard Assets/*.cs            -> Assembly-Csharp-First-Pass.dll
Assets/*.cs                -> Assembly-Csharp.dll
Standard Assets/Editor/*.cs   -> Assembly-Csharp-First-pass-editor.dll
/Editor/*.cs                   -> Assembly-Csharp-Editor.dll


二、脚本编译顺序
Unity保留了一些项目文件夹名称来标识其内的内容具有一些特殊目的。而其中一些文件夹会影响脚本编译的编译顺序。这些文件夹名称是:

Assets
Editor
Editor default resources
Gizmos
Plugins
Resources
Standard Assets
StreamingAssets

脚本编译有四个不同的阶段。处于哪个阶段编译由其父目录确定。
在脚本必须引用其他脚本中定义的类容的情况下,编译的顺序就非常重要了。基本规则是,
在当前阶段之后编译的任何东西都引用不到,在当前阶段之前或早期的阶段编译的任何内容都可以完全引用。

编译阶段如下:

阶段1:编译Standard Assets、Plugin中的运行时脚本
阶段2:编译Standard Assets、Plugin中的Editor脚本
阶段3:编译其他不在Editor目录下的脚本
阶段4:编译Editor目录下的脚本


Unity设计模式

常用的设计模式主要是:
单利没送货i,观察者模式,迭代器模式,访问者模式等等

1)单例模式 : 很简单,即保证一个类仅有一个实例,并且提供访问的全局访问点。
游戏中国,有两种类:
1)不继承MonoBehaiour
2)继承MonoBehaviour


2)观察者模式:

概念:将对象和对象之间创建依赖关系,当其中一个对象发生变化,将变化通过给其创建
关系的对象中,实现自动化的通知更新。


如:游戏开发中,UI上存在下拉List,选择的每一项都会导致UI变化

3)迭代器模式:

就拿c#迭代器为例。既可以了解迭代器模式概念,也可理解c#迭代器的实现。

迭代器模式作为行为模式的一个,本身是简化对象通讯和模式,非常容易理解和使用,
迭代器模式使得能够获取序列中所有元素而不关系类型是什么数据结构。

4)访问者模式:

希望对一个结构对象添加一个功能时,我们能够在不影响结构的前提下,定义一个新的对其元素的操作。
 


C#基础


1.简述值类型和引用类型的区别:
1)值累心光存储虎仔内存栈中,引用类型数据存储在内存堆中。而内存单元中存储的是堆中存放的地址。
2)值类型存取快,引用类型存取慢
3)值类型表示实际存储的数据,引用类型表示指向存储在内存堆中的数据的指针和引用
4)栈的内存是自动释放,堆内存中是.Net中会由GC来自动释放
5)值类型继承自System.ValueType,引用类型继承自System.Object

2.描述Interface和抽象类之间的不同
语法不同:
1.抽象类中可以有字段,接口没有
2.抽象类中可以有实现成员,接口只能包含抽象成员
3.抽象类中所有成员修饰符都是可以使用,接口中所有的成员都是对外的,所以不需要修饰符修饰。
用法不容:
1.抽象类是概念的抽象,接口关注行为
2.抽象类的子类和父类的关系是泛化关系,实现类和接口之间关系是实现关系
3.一个类只能继承一个类,但是可以实现多个接口


3.c#中List,Hashtable,Dictionary区别?
List是对数组的封装,当List对象的Item元素数量超过了Capacity,List对象重新生成一块
大小是原来Capacity的两倍内存空间,然后将当前所有Item元素以及待添加的元素复制到新的内存空间。
频繁的添加数据消耗大,但数据查询速度快,内存连续。


Hashtable、Dictionary它们的内存是用key值根据hash算法算出来的,它们的内存区域是不连续的,
所以在遍历的时候难免会对内存进行换页操作这样遍历的时间就长;

Dictionary:1、类型安全,对值类型没有装箱拆箱操作;
2、在单条数据的读取上速度快;
3、数据的存放是有序的
Hashtable:线程安全,适合多线程,无序;

简述StringBuilder和String的区别?
string是字符串常量
stringBuffer是字符串常量,线程安全
stringBuilder是字符串变量,线程不安全

string类型是不可变的对象,当每次对string进行改变时都需要生成一个新的string对象,
然后将指针指向一个新的对象,如果在一个循环里面,不断改变一个对象,就要不断的生成新的对象,
所以效率很低。建议不断更改string对象的地方不要使用string类型。
stringBuilder对象是对象在做字符串连接操作时在原来的字符串上进行修改,改善了性能。连接操作
频繁是,使用stringBuilder对象。


索引器与属性的区别:
1)索引器以函数签名方式this来标识,而属性采用名称来标识,名称可以任意
2.)索引器可以重载,而属性不能重载
3)索引器不能以static来声明,而属性可以

c#解释代码编译:

1.通过c#编译器将源代码编译为托管代码
2.将创建的代码组合到程序集合
3.加载公共语言运行时CLR
4.通过CLR执行程序集

类型安全之拆箱和装箱:
引用类型之间的类型转换

将一个对象转化为任何基类类型。
转换时,将等号左边的和等号右边的类型进行转换。
如果左边是基类,则安全,
否则编译异常,必须显示转换
Object a = new Manager()  正确, Object是基类

装箱的过程:
int x = 1023;
object o = x;  //装箱

执行第一句,托管堆没有任何东西。栈上有一个整型变量。第二局就是装箱,因为Object是引用类型,必须指向堆上的某个对象,而x是值类型,没有堆上的对应对象。
所以需要使用装箱,在堆上创建x.

装箱的步骤:
1.分配内存。 这个例子需要一个整型变量,加上托管堆上所有对象都有的两个
额外成员【类型对象指针和同步块索引】。类型对象指针指向int类型对象。
2.值类型的变量赋值到新分匹配的堆内存
3.返回对象的地址


拆箱的过程:
拆箱不需要额外的内存。
int i  = 1;
object o = i;
var j = (byte)o;
拆箱步骤:
1.如果已经装箱实例为Null,抛出NullReference异常
2.对象不是null但是类型不是原先未装箱的值类型,则抛出InvalidCast异常
3.获取已经装箱实例当中值类型字段的地址
4.创建新的值类型变量,其值使用第三步获取到的值


类(class)和结构(struct)的区别?结构对象可能分配在堆上吗?何时考虑使用结构体?
类和结构是c#两个最主要的研究对象。
1.结构是值类型,继承自 System.ValueType,而类是引用类型
2.值类型不能继承,所以结构struct不能继承
3.结构struct作为值类型,但是结构中可以存在引用类型
【但是如果struct中包含引用类型,建议使用类】
结构体如果包含引用类型,则部分分配在堆上

结构:
结构作为值类型,
结构是值类型,new运算符和其他引用类型的工作方式不同。
new运算符并不分配堆中内存,只是调用相应的构造函数。

结构中可以存在构造函数,但是结构内部不能存在无参构造函数。


Unity基础

1)什么是协同程序?

主线程运行时同时开启一段逻辑处理,来协助当前程序的执行。换而言之,开启协程就是开启一个可以与程序并行的逻辑。可以控制运动,序列和对象的行为。

2)Unity中物体发生碰撞的条件:

两个物体都必须带有碰撞器Collider,其中一个物体还必须带有RigidBody刚体。


3)Unity中碰撞器和触发器的区别?

碰撞器是触发器的载体,而触发器只是碰撞其的一个属性。

当Is Trigger=false时,碰撞器根据物理引擎引发碰撞,产生碰撞的效果,可以调用OnCollisionEnter/Stay/Exit函数;
当Is Trigger=true时,碰撞器被物理引擎所忽略,没有碰撞效果,可以调用OnTriggerEnter/Stay/Exit函数。
如果既要检测到物体的接触又不想让碰撞检测影响物体移动或要检测一个物件是否经过空间中的某个区域这时就可以用到触发器

4)OnEnable、Awake、Start运行时的发生顺序?哪些可能在同一个对象周期中反复的发生?

Awake–>OnEnable->Start OnEnable在同一周期中可以反复地发生!

5)Update(),Awake(),Start()的执行顺序是?游戏开始后,分别在何时执行?

脚本自带函数执行顺序如下:将下面脚本挂在任意的物体运行即可看到。

Awake:用于在游戏开始之前初始化变量或者游戏状态。在脚本整个生命周期内它仅被调用一次,Awake()在所有对象呗初始化之后调用,所以可以安全的和其他对象对话或者用如GameObject.FindWithTag()这样的函数搜索他们。每个游戏物体上的Awake()以随机的顺序被调用。因此,使用Awake()来设置脚本间的引用,并且利用start来传递信息Awake总是在Start之前被调用。不能用来执行协同程序。

Start():仅仅在Update()第一次被调用前调用。Start()在behaviour的生命周期中被调用一次。他和Awake()的不同在于 Start()只是在脚本实例被启用是调用/因此,可以按需调整延迟初始化代码。
Awake()总是在Start之前被执行,这允许协调初始化顺序。在所有脚本呢实例之中,Start()总是在Awake()之后被调用。


Update:正常帧更新,用于更新逻辑。每一帧都执行,处理RigidBody时,需要使用FixedUpdate代替Update.例如:给刚体加一个作用力,必须用作用力在FixedUdate里的固定帧,而不是Update中的帧。两者帧长不同,FixedUpdate,每固定帧绘制时执行一次,和Update不同的是FixedUpdate是渲染执行。,如果你的渲染效率低下的时候FixedUpdate调用次数就会跟着下降。FixedUpdate比较适用于物理引擎的计算,因为是跟每帧渲染有关。Update就比较适合做控制。


6)如何优化内存?
有很多方式,比如:
1。压缩自带类库
2。将暂时不用的以后还需要适应的物体隐藏起来而不是直接Destory
3。释放AssertBundle占用的资源
4。降低摩西你给的片面数,降低模型的谷歌数量,降低贴图的大小
5。使用光照贴图,使用多层次细节【LOD】,使用着色器【Shader】,使用预设【Prefab】

7)使用Unity3D实现2d游戏的方式?
1。使用本省的GUI,在Unity4.6后出现的UGUI
2。把摄像机的Projection【投影】值调整为Orthographic【正交投影】,不考虑Z轴
3。使用2d插件,如2DTookit,NGUI

unity面试--对象池

1.对象池就是存放需要被反复调用的资源的一个空间,当一个对象会产生大量生成的时候,如果每次所以通过对象池把暂时不用的对象放到一个空间中(也就是一个集合),当下次要重新生成这个对象的时候,先去池中查找一下是否有可用的对象,如果有的话就直接拿出来使用,不需要再创建。如果没有可用对象的话,才进行创建。利用空间换时间来达到游戏的高速运行效果。在FPS游戏中,常常被大量赋值的对象包括:子弹、敌人、粒子等...都销毁创建浪费时间。


2.使用unity制作2d游戏。

1.使用本身的GUI,在unity4.6以后出现的UGUI
2.把摄像机的Projection(投影)值调为Orthographic(正交投影),不考虑z轴。
3.使用2d插件,如:2DToolKit 和 NGUI。
 


Unity委托

Unity中事件和委托大多数时候和事件相关。

比如:当前需要设计一款计时器(Timer)功能,拥有基础事件:开始计时、暂停计时,停止计时【计时器事件】

using UnityEngine;
using System.Collections;
/// 
///计时器 
/// 
public class Timer : MonoBehaviour {
    public delegate void MyEventHandler(float currentTime);
    #region 计时器的三种基础事件
 
    public static event MyEventHandler onTimerStart;
    public static event MyEventHandler onTimerPause;
    public static event MyEventHandler onTimerStoped;
 
    #endregion
    private bool isStarted=false;
    public bool IsStarted{
        get{ 
            return isStarted;
        }
    }
    private bool isStoped = true;
    public bool IsStoped{
        get{ 
            return isStoped;
        }
    }
    private float totalTime = 0;
    // Update is called once per frame
    void Update () {
 
        //空格键当作“开始/暂停”键
        if (Input.GetKeyDown (KeyCode.Space)) {
            OnChangeState ();
        }
        //回车键当作“停止”键
        if (Input.GetKeyDown (KeyCode.Return)) {
            OnSetStop ();
        }
 
        if (isStarted) {
            isStoped = false;
            totalTime += Time.deltaTime;
        }
    }
    void OnChangeState(){
        var _startState = !isStarted;
        isStarted = _startState;
        if (isStarted) {
            //检查onTimerStart是否为空,防止报空 (废话了。。。下面不做赘述)
            if (onTimerStart != null) {
                onTimerStart (totalTime);
            } else {
                Debug.Log ("onTimerStart is Empty");
            }
        } else {
            if (onTimerPause != null) {
                onTimerPause (totalTime);
            } else {
                Debug.Log ("onTimerPause is Empty");
            }
        }
    }
    void OnSetStop(){
        if (onTimerStoped != null) {
            onTimerStoped (totalTime);
        } else {
            Debug.Log ("onTimerStoped is Empty");
        }
        isStarted = false;
        isStoped = true;
        totalTime = 0;
    }
}

资料:
https://blog.csdn.net/GhostOrange/article/details/53150383


Unity小型游戏寻路算法

1)随机寻路算法:
当NPC不管是遇到障碍物还是遇到了边界(利用碰撞检测),都会随机选取一个前进的方向,继续行走

2)跟踪算法:
当游戏中的主角进入到NPC 的“警戒区域”后,游戏的AI 可轻易获得目标的位置,然后控制NPC 对象移向被跟踪的对象

3)闪避算法:
和跟踪算法完全相反,也就是当游戏中的主角进入到NPC 的“警戒区域”后,主角可以去追着NPC跑
 


Unity框架设计

游戏本身业务仅仅提供抽象层从而提高业务逻辑的独立性和可维护行。
框架部分提供项目的基础设置:
1)资源管理
2)网络通信
3)UI框架
4)消息管理
5)场景管理
6)数据解析和存取

(一)资源管理:

资源管理模块负责按照场景的划分:将所有游戏资源打包到AssertBundle之中,并且在游戏中动态加载
打包前将对应的资源索引文件和二进制文件(AB)放入StreamingAssertsPath下。
游戏初次运行将所有资源拷贝到可读写路径PersistentDataPath


更新阶段从服务器下载更新配置文件并根据本地资源的MD5更新资源文件,场景载入阶段异步加载场景需要的AssetBundle,资源
加载时根据资源索引文件加载资源,离开场景时卸载相关AssetBundle

ResourceManager:

资料:

https://blog.csdn.net/u013108312/article/category/6669595/2?

https://blog.csdn.net/aawoe/article/details/80903614


unity游戏音效


Unity3D游戏引擎一共支持4个音乐格式的文件:

  .AIFF  适用于较短的音乐文件可用作游戏打斗音效

  .WAV  适用于较短的音乐文件可用作游戏打斗音效

  .MP3  适用于较长的音乐文件可用作游戏背景音乐

  .OGG  适用于较长的音乐文件可用作游戏背景音乐

 
1,游戏音乐:适用较长的音乐,如背景音乐。

2,游戏音效:适用较短的音乐,如游戏出大招的音效。

===========================================
Unity优化中对游戏性能优化,声音优化非常关键。


了解Audio Clips、Audio Listeners、AudioSources的基本操作:


AudioClip设置:
取消Preload Audio Data(预加载音频数据)选项。然后把Override for Android选项勾上,并且设置Load Type为Streaming。
这样做可以很好的优化游戏音效的占用


音频可视化:https://blog.csdn.net/noEnoughChief/article/details/82984635


public interface IAudio{
AudioSource GetAudioSource();
void OnPlayVideo(AudioClip clip);
}

public class AudioManager{

}

unity的UI渲染机制

. Canvas[画布]是一个控制一组被渲染UI元素的组件,  所有的UI元素必须是Canvas的子物体,  同一个
   场景中允许存在多个Canvas,  但是使用UI元素必须少有一个可用的 Canvas组件。如果场景中没有
   Canvas,  创建一个UI元素,  新的Canvas组件就会被创建, 该UI元素会附加为 Canvas的子物体.

2. 每一个Canvas都有一些可选的Render Modes (渲染模式),  
    Screen Space - Overlay 是最常见的渲染模式, 也是Canvas默认的渲染模式
    Screen space - Overlay 被选中时,  UI会覆盖整个场景 所有的UI元素会在场景最上层渲染输出
    在这个模式下 UI 会自动填充整个场景.  同时也会随着场景视图大小变化而变化。
    值得注意的是 Canvas 会影响Rect Transform组件,  整个 Rect Transform 不可编辑。
    Canvas会修改Rect Transform 的数值来自动填充整个屏幕。
    在这个模式下有个 Pixel Perfect 选项      Pixel Perfect 选中后, UI元素渲染输出时,  会自动吸附
    到最近的像素点。在某些情况下会增强UI的展示效果。

    Screen Space - Camera 和 Screen Space - Overlay 十分相似,  只不过它是由场景中指定的摄像
    机渲染输出。这会使Camera相关的设定作用于UI元素上。最常使用的场景是使用透明摄像机对UI进行
    渲染输出,  得到景深的效果。 
    在这个模式下 Canvas 会自动填充相机输出视图,  并且根据摄像机输出视图变化而变化,  在这个模式下
    Canvas 组件同样会影响 Rect Transform,  这个模式下有几个选项,  包括Pixel Perfect,作用与
    Screen Space - Overlay 相同。Render Camera 用来设置用于渲染这个 Canvas上UI 元素的摄像机.
    如果被设置为None,  Canvas 就会选择 Screen Space - Overlay 的设置来渲染这个 Canvas, 
    当 Screen Space - Camera 选中且富于摄像机对象时。 UI 会被移至相机的视椎体中,  并改变自身
    大小适应填充。这使得UI能够根据场景视图中游戏对象进行渲染。 为了控制场景中UI元素坐标位置,
    使用Plane Distance 向远或靠近相机,   这时 Canvas 和 UI 会根据与相机的距离自动改变大小,适应
    视椎体。 值得注意的是 Plane Distance 的数值应在相机渲染范围内,  当使用 Screen Space - Camera
    Canvas可以被场景中任意摄像机渲染,  为了使Canvas 及其内容与场景中其他 Canvas 区分隔离开,
    可以通过设置摄像机的 Clear Flag Culling Mask Depth 属性

    World Space 渲染模式在世界坐标系下渲染UI,  UI元素可以是场景中静态的对象 也可以是动态的对象
    例如对话框, 标签等在场景中跟随游戏对象运动,  首先要了解的是在 World Space 渲染模式下 Canvas
    不再影响 Rect Transform , Canvas 可以位于世界坐标系下任意位置,  明白同一场景下允许多个Canvas
    后, 给世界空间坐标系的UI元素创建Canvas的做法就十分常见了。
    Event Camera 是接收事件必须具备的,  确定哪个摄像机来检测例如UI上点击等事件,  大多数情况下, 当
    World Space 选中时,Event Camera 会是当前渲染场景的摄像机.   Receives Events 选项设置这个
    Canvas 中的UI元素是否接收事件  比如点击或悬停。  禁用这个选项则该 Canvas 下的所有子UI元素都将
    忽略事件检测,  Sorting Layer 和 Order Inlayer 用于控制场景中 Canvas 之间渲染的顺序
    Sorting Layer 和 Order InLayer 在 Canvas 中可以进行设置 包含 World Space Screen Space

3. Canvas 中UI元素会以自顶向下的方式进行渲染,  第一个对象会渲染在底层  ,  改变UI元素渲染的顺序,
    只需简单的改变UI元素在Hierarchy 面板的顺序


Unity自定义消息通知组件

客户端消息机制是十分广泛的。

using System;
public class Notification
{
  public Enum Type{set;get;}
  
  public Object[] Params{set;get;}
  
  public Object Target{set;get;}
  
  public Notification(){}
}

资料:
https://www.cnblogs.com/monkeycoder/articles/6860132.html


协程原理分析

工作中协程的作用大多为两点:
1)延时(等待)一段时间执行代码
2)等某个操作完成之后再执行后面的代码。总结起来就是一句话:控制代码在特定的时机执行。

很多初学者,都会下意识地觉得协程是异步执行的,都会觉得协程是C# 线程的替代品,是Unity不使用线程的解决方案。

所以首先,请你牢记:协程不是线程,也不是异步执行的。
协程和 MonoBehaviour 的 Update函数一样也是在MainThread中执行的。
使用协程你不用考虑同步和锁的问题。

Unity协程执行原理:
 UnityGems.com给出了协程的定义:
  A coroutine is a function that is executed partially and, presuming suitable conditions are met, will be resumed at some point in the future until its work is done.
  即协程是一个分部执行,遇到条件(yield return 语句)会挂起,直到条件满足才会被唤醒继续执行后面的代码。
  
Unity在每一帧(Frame)都会去处理对象上的协程。Unity主要是在Update后去处理协程(检查协程的条件是否满足),但也有写特例:

协程和Update其实一样,都是对Unity的每一帧进行处理的函数【如果有的话】

如果MonoBehaviour 是处于激活(active)状态的而且yield的条件满足,就会协程方法的后面代码。还可以发现:如果在一个对象的前期调用协程,协程会立即运行到第一个 yield return 语句处,如果是 yield return null ,就会在同一帧再次被唤醒。如果没有考虑这个细节就会出现一些奇怪的问题『1』。
结论都是从UnityGems.com 上得来的,经过下面的验证发现与实际不符,D.S.Qiu用的是Unity 4.3.4f1 进行测试的。经过测试验证,协程至少是每帧的LateUpdate()后去运行。

MonoBehaviour.enabled = false 协程会照常运行,但 gameObject.SetActive(false) 后协程却全部停止

资料:

参考:https://dsqiu.iteye.com/blog/2029701


协程工作流程

yield的返回值:

yield返回如果是空,【c#是yield return null】,coroutine会被终止。
控制权交回调用的位置,就像coroutine结束了一样——事实上它还处于活动状态。 
下一帧 coroutine会自动在yield的位置执行,直到Coroutine结束为止。

yield返回值非空:

 常见的有:
yield return new WaitForEndOfFrame();//等到该帧结束 
yield return new WaitforSeconds(xxx);//等xxx秒
yield return new WaitForFixedUpdate();//等物理帧结束 
yield return www;//这里www是一个WWW对象
--------------------- 

总提来说,只要返回的值非空,yield 语句的作用就是让coroutine在这里暂停,主线程继续,直到yield的条件得到满足,再继续执行yield后面的代码。

有了这个认识,其实上面的四种情况都是一样的。
前三个分别是等到该帧结束、等xxx秒、等物理帧结束,其实都是满足了 new WaitForXXX的这个条件。而最后一个www,则是等www加载完成,也就是www可以访问了,条件满足了。

另外需要注意的是,每个开启的coroutine会在堆中分配一个对象,并且会一直存在直到结尾。在周期性的函数中(例如Update,FixedUpdate, LateUpdate)开启coroutine一定要警惕,因为你可能会平行运行数以千计的coroutine,占据大量内存空间。帧率会直降,Unity也可能会因为CPU和内存负荷而崩溃掉。通常这种情况下可以用一个布尔变量来避免灾难发生。
--------------------- 


一个协程的执行可以在任何地方用yield语句来暂停,yield return的值决定了什么时候协程恢复执行。
协程在协调在几帧中执行的操作时有极大的用处.协程几乎没有任何性能开销。
StartCoroutine一般都会立即返回,然而你也可以获得返回结果的值。但是这一步会等到协程结束执行才能生效。

public class MainTest : MonoBehaviour{

  void Start(){
    Debug.Log("start");
    StartCroutine(Test());
    Debug.Log("start2");
  }
  
  IEnumerator Test(){
    Debug.Log("test1");
    yield return null;
    Debug.Log("test");
  }

}


运行结果: 
start1   test1   start2   test2


这下就一目了然了,当StartCoroutine刚调用的时候,可以理解为正常的函数调用,然后接着看调用的函数里面。

当被调用函数执行到yield return null;(暂停协程,等待下一帧继续执行)时,根据Unity解释协同程序就会被暂停,
其实我个人认为他这个解释不够精确,先返回开始协程的地方,然后再暂停协程。也就是先通知调用处,“你先走吧,不用管我”,然后再暂停协程。。

public class MainTets : Monobehaviour{

  void Start(){
    Debug.Log("start1");
    StartCoroutine(Test());
    Debug.Log("start2");
  }
  
  IEnumerator Test(){
    Debug.Log("test1");
    yield return new WaitForSeconds(3);
    Debug.Log("test2");
  }
  
}

执行结果: 
start1 test1 start2  test2 [等待3s才打印出来]

验证了“yield return的值决定了什么时候协程恢复执行”这句。
协程->协同程序->其实就是协同两个任务,


外观模式

外观模式:Facade Pattern:隐藏系统的复杂性,向现有系统添加接口,来隐藏系统复杂性。向客户端提供一个可以访问的接口。
这种模式设计一个单一的类,此类向客户端请求的简化方法和对现有系统的方法的委托调用。
外观角色:Facade
客户端通过操作外观角色达到控制子系统的目的。
子系统角色:SubSystem
表示系统的子系统模块


委托delegate

委托的本质:
1)声明委托:
public delegate void MyDelegate(string s);
2)ILSpy反编译:
public classauto ansi sealed MyDelegate : MulticastDelegate
自动生成了类,可见委托是一种数据类型。sealed表明是封闭类,不可继承


创建委托变量的本质:
1)既然=委托是类,并且不是静态类,想要使用,就必须new

public delegate void MyDelegate(string s);

public static void main(string[] args){

MyDelegate delegate = new MyDelegate(MyDelegate_2);  //创建委托变量

MyDelegate_2 delegate2 = MyDelegate_2;

}

static void MyDelegate_2(){
Console.write(""hhhh);
}


调用委托变量和本质:

1)给委托变量加()即可调用

MyDelegate delegate = new MyDelegate(MyDelegate_2);
delegate();


委托和匿名方法:

如果委托指向的方法知识临时使用,就不需要声明了,直接使用匿名方法。

private delegate void MyDelegateHandler(string name);

private void MyDelegate(string name,MyDelegateHandler mydelegateHandler)
{
  mydelegateHandler(name);
}

MyDelegate delegate = new MyDelegate(delegate(string name){
  Console.WriteLine();
});


委托好处:
1)逻辑分离,解除耦合
2)对修改封闭,对扩展开放


委托和事件

委托和事件没有可比性,主要在于;
1)委托作为数据类型,事件是对象【理解为对委托变量的封装】

委托对象【委托方法实现的事件】和【标准的event方式实现】的事件的区别:

事件内部是委托实现

三种实现事件方式的区别:
1)直接使用委托实现
2)用私有委托+共有方法模拟事件
3)直接event事件

对于事件来说,外部只能“注册自己” : +=
注销自己“”: -=
外部不可以注销其他的注册这,外界不可主动触发事件
如果使用Delegate就没有办法进行这样的控制
所以诞生了事件

事件是用来阉割委托实例的。事件只能add、remove自己,不能赋值。事件只能+=、-=,不能=、不能外部触发事件。 

1、委托的作用:

占位,在不知道将来要执行的方法的具体代码时,可以先用一个委托变量来代替方法调用(委托的返回值,参数列表要确定)。在实际调用之前,需要为委托赋值,否则为null。

2、事件的作用:

事件的作用与委托变量一样,只是功能上比委托变量有更多的限制。(比如:1.只能通过+=或-=来绑定方法(事件处理程序)2.只能在类内部调用(触发)事件。)

3、在自定义控件(自己编写控件的时候,会大量用到.编写控件的时候,会写一些事件。但是当这些事件 被触发以后,具体执行的那些事件处理程序是编写控件的人没法确定的。这个时候只能通过事件来占位(调用),具体调用的是哪个方法,由使用控件的人来决定(Click+=new 委托(方法名);))

先有观察者模式,再有委托事件这一技术。它有一个前后顺序,如果你要彻底学会委托事件,那么就要先学会观察者模式,还要了解观察者模式的缺陷。

观察者模式的缺陷:

观察者模式是对主题对象和观察者对象进行解耦,使双方都依赖与抽象,而不是依赖于对方的具体对象,
、使双方的变化都不会影响到对方的具体对象。当多个对象需要根据一个对象的状态发生相应的改变或操作时,
可使用观察者模式。

猫和人抓老鼠:
1)老鼠是主体对象
2)猫和人是观察者

资料:

https://blog.csdn.net/q493201681/article/details/82352616


委托的使用


向函数传参:

void  Start(){
Print("xxx");
}

void Print(string name){
base.print(name);
}

传统的传递参数的方式

c#委托的方便在于只需要告诉某个函数的名称就可以实现调用。

委托其实 实质就是  事件和Observer观察者模式

class X
{
  delegate void Helper();
  
  void A()
  {
    Print("A");
  }
  
  void B()
  {
    Print();
  }

}

射线Ray的原理


射线检测故名就是通过射线去检测是否和碰撞器产生了交集,和碰撞器与碰撞器发生交集一样,会返回一个真。

射线的用法很多:比如检测是否跳跃,通过向地面投射射线控制在地面时候可以跳起。

        射击游戏中可以通过定长射线去判断目标物体是否被击中,等

主要用到的工具类是:

Physics
RaycastHit 光线投射碰撞


Unity中用于检测鼠标点在屏幕上提供了一个控制方案。
具体点在unity三维世界坐标的哪个点。
通过鼠标点击屏幕,由屏幕点向unity三维直接发射一条射线。

当检测到碰撞物体后,会返回被碰撞物体的所有信息,以及交点信息等等…

射线检测即对一个碰撞器进行检测,如果碰撞到碰撞器,则返回true,否则返回false,这个检测是一条射线,
这条射线由我们自己设置 。一般在Update函数配合if()来使用。

RaycastHit hit 此为射线碰撞到的物体碰撞器, 
以out hit传入 当射线碰撞到物体时,返回true,hit则是对应的碰撞物体的碰撞器 
可以通过hit来找到碰撞到的物体对象信息。

        
        //1. 射线开始的位置
        //2. 射线的方向
        //3. 碰撞信息
        //4. 射线的最大距离
        //5. 层级
Physics.Raycast (transform.position, transform.forward, out hit, 2, _layerMask)

应用的场景之一:
1)通过射线使得物体移动:
通过鼠标点击的方法,获取Ray类型对象
—ray = Camera.main.ScreenPointToRay(Input.mousePosition);
2)然后根据Ray类型对象做碰撞检测(Plane),得到RaycastHit _hit
if(Physics.Raycast(_ray,out _hit))
3)_hit.point就是需要移动的地方,此时我们使用transform.LookAt()看向他,然后设置transform.forward()移动到该点。
注意参数是vector3.forward而非transform.forward,两者调用区别很大


游戏UI框架设计思路

UI窗体功能设计分布:
1)窗体自动加载管理
2)窗体生命周期管理
3)模态窗体实现
4)资源管理化
5)缓存UI窗体
6)窗体层级管理
7)消息中心
8)日志调试

UI框架核心设计:

根据UI框架的核心设计理念,定义三个核心功能:
1)UI窗体的自动加载 
2)缓存UI窗体
3)窗体生命周期(状态)管理

UI框架设计的目的: 完成和具体游戏逻辑无关的底层事物。

四个核心类:
1)BaseUIForms: 基础UI窗体脚本
2)UIManager: UI窗体管理器脚本
3)UIType:窗体类型 【引用窗体的重要属性:枚举类型】
4)系统定义类 【框架使用的枚举类型,委托事件,系统常量


物理组件Collider

无论碰撞的两个物体是利用碰撞器,还是利用触发器,都必须带有碰撞器组件【Collider】。其中一个物体必须带有RigidBody刚体。

简单来说:碰撞发生必须有两个必要条件:
1)双方物体必须有一个是刚体
2)两个物体比须有Collider
*静态碰撞 只有一个有刚体的碰撞,如果没有刚体的物体去碰撞带有刚体的物体时,有的时候会检测不到,因为刚体休眠的原因。
动态碰撞 两个都有刚体的碰撞
*运动学碰撞 两个都必须有刚体,而且有一个是运动学刚体,这时两个物体碰撞的时候,运动学刚体不会受到影响。


1)碰撞器是一群组件,它包含了很多种类,比如:Box Collider,Capsule Collider等,这些碰撞器应用的场合不同,
但都必须加到GameObjecet身上。
2)所谓触发器,只需要在检视面板中的碰


撞器组件中勾选IsTrigger属性选择框。

触发信息检测:
1)MonoBehaviour:OnTriggerEnter(Collider other):进入触发器
2)MonoBehaviour:OnTriggerExit(Collider other):退出触发器
3)MonoBehaviour:OnTriggerStat(Collider other):逗留触发器

碰撞器:
1)MonoBehaviour:OnCollisionEnter(Collision collisionInfo):进入碰撞器
2)MonoBehaviour:OnCollisionExit(Collision collisionInfo):退出碰撞器
3)MonoBehaviour:OnCollisionStay(Collision collisionInfo):逗留碰撞器


CharacterController和RigidBody的区别?


虚方法,抽象类和接口的区别

虚方法和普通方法其实差不多。

虚方法(virtual)可以实例化调用,也可以被继承类重写(Override)

它的作用是当实例类型从继承类转换为基类时,类中方法仍然是继承类中的实现。


如果不使用虚方法而使用普通方法隐藏基类方法,那么当实例类型转换时,编程了基类的实现。

抽象类(抽象方法)和接口类似,继承类都需要实现它们声明的函数。 
抽象方法声明使用,是必须被派生类覆写的方法,抽象类就是用来被继承的;可以看成是没有实现体的虚方法;如果类中包含抽象方法,那么类就必须定义为抽象类,不论是否还包含其他一般方法;抽象类不能有实体的。

相同点: 
(1) 都可以被继承 
(2) 都不能被实例化 
(3) 都可以包含方法声明 
(4) 派生类必须实现未实现的方法

不同点: 
(1) 抽象基类可以定义字段、属性、方法实现;接口只能定义属性、索引器、事件、和方法声明,不包含字段以及方法实现。 
(2) 抽象类是一个不完整的类,需要进一步细化,而接口是一个行为规范。 
(3) 接口可以被多重实现,抽象类只能被单一继承 
(4) 抽象类的成员可以具有访问级别,而接口的成员全部public级别 
(5) 抽象的子类可以选择性实现其基类的抽象方法,而接口的子类必须实现 
(6) 接口基本上不具备继承的任何具体特点,它仅仅承诺了能够调用的方法 
(7) 接口可以用于支持回调,而继承并不具备这个特点 
(8) 抽象类实现的具体方法默认为虚的,但实现接口的类中的接口方法却默认为非虚的,当然您也可以声明为虚的 
(9) 如果抽象类实现接口,则可以把接口中方法映射到抽象类中作为抽象方法而不必实现,而在抽象类的子类中实现接口中方法


 

资源加载

通常推荐使用AssetBundle来加载资源,使用AssetBundle可以按更小的包来管理资源、更新资源,同时还可以加快游戏启动速度。
但是,
加载AssetBundle需要我们自己去维护依赖关系,对比起Resources来说更加麻烦。通常在开发的时候使用Resources加载,
而在发布版本使用AssetBundle。这里需要实现自己的加载器来满足两套资源的切换。

资源管理器:
1)统一的Resources和AssetBundles加载
2)类似的加载接口设计,包括同步与异步
3)强引用计数管理,Load与Unload匹配
4)支持按优先级加载资源
5)支持配置系统开销,异步加载开销


类似的加载接口设计,包括同步与异步
强引用计数管理,Load与Unload匹配
支持按优先级加载资源
支持配置系统开销,异步加载开销

外实现为静态接口,正常情况下支持Editor运行时与非运行时,运行时不管在PC还是手机都支持Resources与AssetBundles无缝切换。
所有的加载路径参数统一为Resources目录相对路径且不包含扩展名,这里要求在同一目录不要有同名文件(仅扩展名不一样)。
按类型匹配资源是较烦琐的工作,而且对于Object基类加载,无法匹配到正确的资源。
异步接口定义一个自己Request类返回,除了原有的ResourceRequest数据,这里新增一个打断属性。
当不再持有这个对象的时候设置打断属性来中断加载。同时这里还支持配置回调接口,这样不需要每次更新去查询状态,资源管理器在异步加载完成后执行回调接口。
异步加载接口增加优先级参数,优先加载高优先级的对象。自己维护一个优先级列表,并发起一定数量的异步加载请求,
对于在队列中被打断的资源则可以节省一次资源价值请求。


然后还要关注异步加载的开销,避免异步加载占用太多的主线程时间。

由于实现了自己Request,所以这里也要实现自己的时间片管理器。实例化对象与回调接口的开销都是不可预期的,
我们配置一个每帧最大执行时间做平滑。
最后讨论下资源卸载策略,实时卸载资源导致资源反复加载,引起游戏卡顿。
通常会缓存一定数量的资源来改善体验,由于只有调用了UnloadUnusedAsset才会真正清理资源,所以一般情况下会一直持有资源,
然后根据未使用的资源数量情况触发统一的UnloadUnusedAsset,这时候资源才会被真正释放。由于我们使用了强引用计数管理,
所以在清理的时候通过对引用计数的判断就可以正确的清理资源。特别对于使用AssetBundle加载资源的情况,错误的管理可能会导致资源重复加载,浪费内存。

资源对象池
资源加载器负责加载、卸载资源,同时缓存资源,这里的资源对象池特指GameObject资源池。GameObject资源通常带有自己的数据,在加载的时候需要实例化一份以便使用。实例化GameObject是一个开销较大的操作,同时也会带来较高的GC Alloc(内存分配)。资源对象池就是一个GameObject对象池用于缓存实例化的GameObject对象。
资源对象池在使用上要注意GameObject对象的可复用性,开始的时候加载一个预制体(Prefab)是一个干净的数据。
外部逻辑会修改GameObject上的数据、添加新的组件,之后这个对象会入池。设计上如果一个对象需要使用对象池的复用功能,
逻辑需要保证这个GameObject是可复用的,这并不是一件容易的事情。把状态还原重置本身就有一定的开销,如果实例化一个对象的成本低于重置数据的
开销,那就不需要对象池,每次重新实例化即可。


同时在对象入池的时候还需要做一项工作是让对象不可见,与销毁一个对象(对象入池)在这里保持行为一致。有两个常见的做法,一个是SetActive(false),
还有一个做法是把对象移出摄像机。对于对象数特别多的对象修改坐标的开销较大,对于组件较多的对象修改激活状态的开销可能会更大。这里提供了三种入池行为,InActive、InVisible、Destroy用于处理上面讨论的情况。

你可能感兴趣的:(Unity)