首先要仿照飞船的创建流程创建一个主角
首先直接导入 Starter Assets - Third Person Character Controller 太爽了
我看看啊 他这个是除了 ThirdPersonController 之外还有一个处理刚体碰撞的,一个玩家跟随摄像机,一个 InputSystem 相关的 我在想能不能把他们搬到框架里面
我在看战机的时候,看这两行
IDataTable<DRAircraft> dtAircraft = GameEntry.DataTable.GetDataTable<DRAircraft>();
DRAircraft drAircraft = dtAircraft.GetDataRow(TypeId);
我就一直在想,为什么 GetDataRow 输入的是一个 TypeId 我乍一看上去还以为是,输入行号 TypeId,获得数据表的一行 所以我就奇怪,为什么 typeid 会对应行 因为战机的数据表里面有很多行嘛,那按照我这么想的话,就感觉是,一个表的很多行对应很多 typeid,但是一个战机怎么会在内部有很多 typeid 呢
之后才发现,原来他这个 GetDataRow 原来说的就是获取一整个 DR+数据表名 的类的实例啊hhh
哦我在整理流程的时候最后还是溯源到了一个接口的实例 之前我一直都被这个实例难住,因为我只看得到这个实例,用转到定义也看不到接口中的函数的实现 然后我就不知道怎么看了 之后去网上搜了一下
他一说构造函数,我才想起来要看这个接口的实例的初始化hhhh
比如我现在一路转到定义,转到了
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs
public void ShowEntity(int entityId, Type entityLogicType, string entityAssetName, string entityGroupName, int priority, object userData)
{
if (entityLogicType == null)
{
Log.Error("Entity type is invalid.");
return;
}
m_EntityManager.ShowEntity(entityId, entityAssetName, entityGroupName, priority, ShowEntityInfo.Create(entityLogicType, userData));
}
然后我就想看 m_EntityManager.ShowEntity ,但是如果直接对 ShowEntity() 转到定义,会看到接口类中的函数定义,却看不到实现
C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\832e6a5ddbb7461ba47bec9b9ce43ea6\IEntityManager.cs
public interface IEntityManager
{
void ShowEntity(int entityId, string entityAssetName, string entityGroupName);
void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority);
void ShowEntity(int entityId, string entityAssetName, string entityGroupName, object userData);
void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority, object userData);
这个时候应该看 m_EntityManager 的初始化
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs
protected override void Awake()
{
base.Awake();
m_EntityManager = GameFrameworkEntry.GetModule<IEntityManager>();
往下找
C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs
public static T GetModule<T>() where T : class
{
Type typeFromHandle = typeof(T);
if (!typeFromHandle.IsInterface)
{
throw new GameFrameworkException(Utility.Text.Format("You must get module by interface, but '{0}' is not.", typeFromHandle.FullName));
}
if (!typeFromHandle.FullName.StartsWith("GameFramework.", StringComparison.Ordinal))
{
throw new GameFrameworkException(Utility.Text.Format("You must get a Game Framework module, but '{0}' is not.", typeFromHandle.FullName));
}
string text = Utility.Text.Format("{0}.{1}", typeFromHandle.Namespace, typeFromHandle.Name.Substring(1));
return GetModule(Type.GetType(text) ?? throw new GameFrameworkException(Utility.Text.Format("Can not find Game Framework module type '{0}'.", text))) as T;
}
其实之前也找到过这里,但是脑子里没有一个思路,就没有深究 现在的思路就是,他是用来初始化接口实例的 然后这里把它叫做 module,嗯……很神奇
这个函数就是检查输入的类型是否是接口,是否是框架内的模块,都是的话才进行下一步 下一步是
C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs
private static GameFrameworkModule GetModule(Type moduleType)
{
foreach (GameFrameworkModule s_GameFrameworkModule in s_GameFrameworkModules)
{
if ((object)s_GameFrameworkModule.GetType() == moduleType)
{
return s_GameFrameworkModule;
}
}
return CreateModule(moduleType);
}
遍历已知的模块的列表,没有就创建 这个 s_GameFrameworkModules 是没有别的引用的,那就说明模块唯一的来源就是 CreateModule(moduleType) 那么我要知道接口实例的接口函数是怎么实现的,我还是要看这个模块是怎么创建的
C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs
private static GameFrameworkModule CreateModule(Type moduleType)
{
GameFrameworkModule gameFrameworkModule = (GameFrameworkModule)Activator.CreateInstance(moduleType);
if (gameFrameworkModule == null)
{
throw new GameFrameworkException(Utility.Text.Format("Can not create module '{0}'.", moduleType.FullName));
}
LinkedListNode<GameFrameworkModule> linkedListNode = s_GameFrameworkModules.First;
while (linkedListNode != null && gameFrameworkModule.Priority <= linkedListNode.Value.Priority)
{
linkedListNode = linkedListNode.Next;
}
if (linkedListNode != null)
{
s_GameFrameworkModules.AddBefore(linkedListNode, gameFrameworkModule);
}
else
{
s_GameFrameworkModules.AddLast(gameFrameworkModule);
}
return gameFrameworkModule;
}
后面是插入优先级链表,创建的部分是在前面 GameFrameworkModule gameFrameworkModule = (GameFrameworkModule)Activator.CreateInstance(moduleType); 哦好吧我记得我之前确实是看到了这个Activator.CreateInstance ,然后我之后不知道这个 C# 的 Activator.CreateInstance 是怎么创建出实现了接口函数的
好吧,现在的问题就是 它有一个 接口类的实例,是用 Activator.CreateInstance 创建的 那么我怎么知道这个实例创建出来之后,他对接口的实现是什么
我在网上看到的 Activator.CreateInstance 的作用是 使用与指定参数匹配程度最高的构造函数来创建指定类型的实例 那我应该是找构造函数才对,说不定我就找到了真正创建的那个类
但是那个创建类的代码是只读的,难顶 就是说我还不能加一个打印的代码 这样的话我就看不到他这个到底是输入了个什么函数 我还不想加断点 先试试慢慢分析把
要想知道向 Activator.CreateInstance 输入了什么,源文件只读,但是仍然可以仿照 GetModule 里面的函数,来得到输入的字符串。比如说在流程里面复现一遍
System.Type typeFromHandle = typeof(GameFramework.Entity.IEntityManager);
string text = GameFramework.Utility.Text.Format("{0}.{1}", typeFromHandle.Namespace, typeFromHandle.Name.Substring(1));
GameFramework.GameFrameworkLog.Debug("!!!!!!!!!!!!!" + text);
这里打印的是 GameFramework.Entity.EntityManager
……哦我悟了,原来这就是源码的部分,这就需要看别人 dll 里面的内容了! 太顶了,我去问别人,别人说看源码,我在工程里面找半天不知道啥是源码
就是这个! https://github.com/EllanJiang/GameFramework/blob/master/GameFramework/Entity/EntityManager.cs
GameFramework/Entity/EntityManager.cs
public void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority, object userData)
{
EntityInstanceObject entityInstanceObject = entityGroup.SpawnEntityInstanceObject(entityAssetName);
if (entityInstanceObject == null)
{
int serialId = ++m_Serial;
m_EntitiesBeingLoaded.Add(entityId, serialId);
m_ResourceManager.LoadAsset(entityAssetName, priority, m_LoadAssetCallbacks, ShowEntityInfo.Create(serialId, entityId, entityGroup, userData));
return;
}
InternalShowEntity(entityId, entityAssetName, entityGroup, entityInstanceObject.Target, false, 0f, userData);
GameFramework/Entity/EntityManager.cs
private void InternalShowEntity(int entityId, string entityAssetName, EntityGroup entityGroup, object entityInstance, bool isNewInstance, float duration, object userData)
{
try
{
IEntity entity = m_EntityHelper.CreateEntity(entityInstance, entityGroup, userData);
if (entity == null)
{
throw new GameFrameworkException("Can not create entity in entity helper.");
}
EntityInfo entityInfo = EntityInfo.Create(entity);
m_EntityInfos.Add(entityId, entityInfo);
entityInfo.Status = EntityStatus.WillInit;
entity.OnInit(entityId, entityAssetName, entityGroup, isNewInstance, userData);
entityInfo.Status = EntityStatus.Inited;
entityGroup.AddEntity(entity);
entityInfo.Status = EntityStatus.WillShow;
entity.OnShow(userData);
entityInfo.Status = EntityStatus.Showed;
if (m_ShowEntitySuccessEventHandler != null)
{
ShowEntitySuccessEventArgs showEntitySuccessEventArgs = ShowEntitySuccessEventArgs.Create(entity, duration, userData);
m_ShowEntitySuccessEventHandler(this, showEntitySuccessEventArgs);
ReferencePool.Release(showEntitySuccessEventArgs);
}
}
catch (Exception exception)
{
if (m_ShowEntityFailureEventHandler != null)
{
ShowEntityFailureEventArgs showEntityFailureEventArgs = ShowEntityFailureEventArgs.Create(entityId, entityAssetName, entityGroup.Name, exception.ToString(), userData);
m_ShowEntityFailureEventHandler(this, showEntityFailureEventArgs);
ReferencePool.Release(showEntityFailureEventArgs);
return;
}
throw;
}
}
原来在这里控制了 OnInit() OnShow() ,还有 ShowEntity 的事件
要看最开头的 m_EntityHelper.CreateEntity 又感觉有点难办,这也是一个接口
GameFramework/Entity/EntityManager.cs
public void SetEntityHelper(IEntityHelper entityHelper)
{
if (entityHelper == null)
{
throw new GameFrameworkException("Entity helper is invalid.");
}
m_EntityHelper = entityHelper;
}
在框架源码里面没找到 SetEntityHelper 的引用 居然是在游戏工程里面的
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs
public sealed partial class EntityComponent : GameFrameworkComponent
{
[SerializeField]
private string m_EntityHelperTypeName = "UnityGameFramework.Runtime.DefaultEntityHelper";
[SerializeField]
private EntityHelperBase m_CustomEntityHelper = null;
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs
private void Start()
{
EntityHelperBase entityHelper = Helper.CreateHelper(m_EntityHelperTypeName, m_CustomEntityHelper);
m_EntityManager.SetEntityHelper(entityHelper);
要知道 m_EntityHelper.CreateEntity 就要知道 m_EntityHelper 巴拉巴拉 哎呀,我突然想到或许直接写出关系更简单
m_EntityHelper.CreateEntity <- m_EntityHelper <- Helper.CreateHelper
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Utility\Helper.cs
public static T CreateHelper<T>(string helperTypeName, T customHelper, int index) where T : MonoBehaviour
{
T helper = null;
if (!string.IsNullOrEmpty(helperTypeName))
{
System.Type helperType = Utility.Assembly.GetType(helperTypeName);
if (helperType == null)
{
Log.Warning("Can not find helper type '{0}'.", helperTypeName);
return null;
}
if (!typeof(T).IsAssignableFrom(helperType))
{
Log.Warning("Type '{0}' is not assignable from '{1}'.", typeof(T).FullName, helperType.FullName);
return null;
}
helper = (T)new GameObject().AddComponent(helperType);
}
else if (customHelper == null)
{
Log.Warning("You must set custom helper with '{0}' type first.", typeof(T).FullName);
return null;
}
else if (customHelper.gameObject.InScene())
{
helper = index > 0 ? Object.Instantiate(customHelper) : customHelper;
}
else
{
helper = Object.Instantiate(customHelper);
}
return helper;
}
又是一个动态创建的 由于 EntityHelper 是定义好的,所以他应该会返回一个添加了 UnityGameFramework.Runtime.DefaultEntityHelper 组件的 GameObject 嗯……那我还是要看源码中的 UnityGameFramework.Runtime.DefaultEntityHelper 哦不用,这个就是工程中的
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\DefaultEntityHelper.cs
public override IEntity CreateEntity(object entityInstance, IEntityGroup entityGroup, object userData)
{
GameObject gameObject = entityInstance as GameObject;
if (gameObject == null)
{
Log.Error("Entity instance is invalid.");
return null;
}
Transform transform = gameObject.transform;
transform.SetParent(((MonoBehaviour)entityGroup.Helper).transform);
return gameObject.GetOrAddComponent<Entity>();
}
啊……好像出乎意料地简单 不过在 helper 之前就已经创建了 GameObject 的嘛 我得看看是哪里创建了
GameFramework/Entity/EntityManager.cs
EntityInstanceObject entityInstanceObject = entityGroup.SpawnEntityInstanceObject(entityAssetName);
if (entityInstanceObject == null)
{
int serialId = ++m_Serial;
m_EntitiesBeingLoaded.Add(entityId, serialId);
m_ResourceManager.LoadAsset(entityAssetName, priority, m_LoadAssetCallbacks, ShowEntityInfo.Create(serialId, entityId, entityGroup, userData));
return;
}
一开始的 entityGroup.SpawnEntityInstanceObject(entityAssetName) 会返回 m_InstancePool.Spawn(name) 这看上去像是从对象池中生成,如果池中没有对象……呃好像不对 entityInstanceObject == null 之后就直接返回了,那么如果到了 InternalShowEntity 的话那必然是要有 entityInstanceObject 那还是要继续看 entityInstanceObject 是怎么生成的
GameFramework\Entity\EntityManager.EntityGroup.cs
public EntityInstanceObject SpawnEntityInstanceObject(string name)
{
return m_InstancePool.Spawn(name);
}
要看 m_InstancePool 的初始化
GameFramework\Entity\EntityManager.EntityGroup.cs
public EntityGroup(string name, float instanceAutoReleaseInterval, int instanceCapacity, float instanceExpireTime, int instancePriority, IEntityGroupHelper entityGroupHelper, IObjectPoolManager objectPoolManager)
{
m_InstancePool = objectPoolManager.CreateSingleSpawnObjectPool<EntityInstanceObject>(Utility.Text.Format("Entity Instance Pool ({0})", name), instanceCapacity, instanceExpireTime, instancePriority);
objectPoolManager 是输入的参数,又要看参数是哪来的,所以要看 哪里调用了 EntityGroup()
GameFramework\Entity\EntityManager.cs
public bool AddEntityGroup(string entityGroupName, float instanceAutoReleaseInterval, int instanceCapacity, float instanceExpireTime, int instancePriority, IEntityGroupHelper entityGroupHelper)
{
m_EntityGroups.Add(entityGroupName, new EntityGroup(entityGroupName, instanceAutoReleaseInterval, instanceCapacity, instanceExpireTime, instancePriority, entityGroupHelper, m_ObjectPoolManager));
输入到 EntityGroup() 的参数是 m_ObjectPoolManager ,又要看 m_ObjectPoolManager 的初始化
GameFramework\Entity\EntityManager.cs
public void SetObjectPoolManager(IObjectPoolManager objectPoolManager)
{
if (objectPoolManager == null)
{
throw new GameFrameworkException("Object pool manager is invalid.");
}
m_ObjectPoolManager = objectPoolManager;
}
要看 m_ObjectPoolManager 的初始化,又回到了游戏工程
Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs
private void Start()
{
m_EntityManager.SetObjectPoolManager(GameFrameworkEntry.GetModule<IObjectPoolManager>());
那么还是要看回游戏框架中的 ObjectPoolManager 回顾一下,我要找的是 CreateSingleSpawnObjectPool 方法
在这里,他们都会跳转到 InternalCreateObjectPool
GameFramework\ObjectPool\ObjectPoolManager.cs
private ObjectPoolBase InternalCreateObjectPool(Type objectType, string name, bool allowMultiSpawn, float autoReleaseInterval, int capacity, float expireTime, int priority)
{
Type objectPoolType = typeof(ObjectPool<>).MakeGenericType(objectType);
ObjectPoolBase objectPool = (ObjectPoolBase)Activator.CreateInstance(objectPoolType, name, allowMultiSpawn, autoReleaseInterval, capacity, expireTime, priority);
m_ObjectPools.Add(typeNamePair, objectPool);
return objectPool;
}
哦……只是根据对象的类型创造了相应类型的 ObjectPoolBase
那再往回看,我还要看对象池的 Spawn 方法 但是只在 GameFramework\ObjectPool\ObjectPoolBase.cs 中找到 AllowMultiSpawn 方法 怪了,我要跳转回……
哦我知道了,在框架里面使用转到实现就能看到了 我靠我刚刚在干嘛,我记得是在游戏工程里转到定义没作用,但是刚刚在源码里可以 习惯性地没有尝试过转到定义 难受
GameFramework\ObjectPool\ObjectPoolManager.ObjectPool.cs
public T Spawn(string name)
{
if (name == null)
{
throw new GameFrameworkException("Name is invalid.");
}
GameFrameworkLinkedListRange<Object<T>> objectRange = default(GameFrameworkLinkedListRange<Object<T>>);
if (m_Objects.TryGetValue(name, out objectRange))
{
foreach (Object<T> internalObject in objectRange)
{
if (m_AllowMultiSpawn || !internalObject.IsInUse)
{
return internalObject.Spawn();
}
}
}
return null;
}
GameFramework\ObjectPool\ObjectPoolManager.Object.cs
public T Spawn()
{
m_SpawnCount++;
m_Object.LastUseTime = DateTime.UtcNow;
m_Object.OnSpawn();
return m_Object;
}
哦……生成数量加1,最近一次使用时间设置为当前时间,然后调用生成回调,然后返回 m_Object
然后又要看 m_Object 的初始化……啊……累了 或许应该设断点来看
the breakpoint will not currently be hit. Unable to find a corresponding location
哦我记起来我为什么一直没有用调试 我之前尝试过调试,但是我设的断点不会停的……令人抓狂
看了一下别人的问题,感觉是 visual studio 的问题 https://forum.unity.com/threads/vs2015-the-breakpoint-will-not-currently-be-hit-unable-to-find-corresponding-location.454924/
好吧,我去下了一个 jetbrain rider 运行之后,代码原封不动,确实能够打到断点了,nb
但是他还是只是在调用接口的地方,嗯,就直接步进下去了,没有进到 m_EntityManager.ShowEntity 的里面 麻了,时间成本好高,已经不想再看了……
|