We’re no longer leaking huge amounts of data updating the NPCs, and we’re tidying up the corpses, but we’re still wasting both CPU time and memory.
Old defunct NPC objects are kept around, while a new NPC will always have to be instantiated. First we’ll convert to using SetActive
instead of my homemade solution for deactivating NPCs, and add some stats to track our instances.
voidOnGUI(){GUI.Box(newRect(0,0,100,50),"# of cells\n"+mapManager.num_cells);GUI.Box(newRect(100,0,100,50),"NPC Instances\n"+npcManager.NumNpcScripts);GUI.Box(newRect(200,0,100,50),"Active NPC Insts\n"+npcManager.NumActiveNpcScripts);GUI.Box(newRect(300,0,100,50),"NPCs in DB\n"+npcManager.NumNpcsInDb);
voidUpdateAllNpcs(){List<NPCLite>npcs=DragonsSpine.DAL.DBNPC.GetAllNpcs();print("I found "+npcs.Count+" NPCs");numNpcsInDb=npcs.Count;foreach(NPCLitenpcinnpcs){if(npc.lastActiveRound>maxGameRound)maxGameRound=npc.lastActiveRound;}foreach(NPCLitenpcinnpcs){// skip inactive NPCs, and make sure they know they are inactiveif(npc.lastActiveRound<maxGameRound-10){if(npcScripts.ContainsKey(npc.worldNpcID)){npcScripts[npc.worldNpcID].gameObject.SetActive(false);}
Never gonna let you go
58 NPCs have been cleaned up, but we still have an instance for every one that ever existed. Tsk, tsk.
if(!npcScripts.ContainsKey(npc.worldNpcID)){// Try to find an inactive one to re-useNpcScripttempNpc=null;inttempNpcId=0;foreach(KeyValuePair<int,NpcScript>kvpinnpcScripts){if(!kvp.Value.gameObject.activeSelf){tempNpc=kvp.Value;tempNpcId=kvp.Key;break;}}if(tempNpc!=null){print("Found inactive NPC "+tempNpcId+" and am reusing it!");npcScripts.Remove(tempNpcId);tempNpc.Reset();}else{print("Could not find an inactive NPC for "+npc.worldNpcID+" so am instantiating it!");tempNpc=(NpcScript)Instantiate(npcScript);}tempNpc.npcId=npc.worldNpcID;tempNpc.name=npc.Name;SetMaterials(tempNpc);tempNpc.toBeSeen=(position.z<=mapManager.zTop+1);tempNpc.gameObject.SetActive(true);npcScripts[npc.worldNpcID]=tempNpc;}