HI!I have script, which works so good without FindClosestEnemies(); function.
Here is part of code:
GameObject FindClosestEnemy()
{
gos = GameObject.FindGameObjectsWithTag("Player");
GameObject closest=null;
for(int i=0;i<100;i++){}
float distance = Mathf.Infinity;
Vector3 position = transform.position;
foreach(var go in gos){
var diff = (go.transform.position-position);
var curDistance=diff.sqrMagnitude;
if(curDistance<distance)
{
closest=go;
distance=curDistance;
}
}
}
With this function, i have <15fps, without >200;
What Should I do? I think i shoud change this: gos = GameObject.FindGameObjectsWithTag(“Player”);, but i have not any ideas how to do it.
(I make RTS game^ and this cannon’s(Turrel) script(part of one)
The best way depends on how many enemies / objects will be there, what’s the max-distance and how are they distributed across the map.
Use Physics.OverlapSphere with your max distance to get a list of all objects within the max distance. Make sure you put all units that should be selectable on a seperate unit-layer. Every enemy need to have a collider attached. This function is way faster then any of Unitys search-functions, especially when you have many objects and a max detection distance.
Another way is that each enemy register / unrigisters itself by some kind of static Unit-Manager. This manager always holds an updated list of all units. If the map is really big and you have a lot of enemies (200+), it’s better to use the first version.
I know in a lot questions people suggest to use FindGameObjectsWithTag, but keep in mind it has to iterate through all objects so it’s quite slow. OverlapSphere uses the physics system which probably uses some kind of space partitioning which minimize the overhead.
It certainly better to get the reference to things like player once at the start. Whether its going to make a lot of difference to your routine I don’t know.
So do something like:
private var playerObject: GameOject;
function Startup()
{
playerObject= GameObject.FindGameObjectsWithTag("Player");
}
function DoWhatever()
{
// You can use "playerObject" instead of doing a find on it
}
Call it once every second or two. It will even make the enemies look cooler (when a better target comes closer, they’ll sometimes take a few shots at the old target before noticing.)
“Easy” way is a counter: if(AIcheck>0) AIcheck--; else { AIcheck=100; // do long loop.
Slightly CPU faster way is a coroutine, called at start, like:
IEnumerator getClosestRunner() {
// each enemy waits a random amount, so they don't all check same frame:
yield return new WaitForSeconds(Random.Range(0.0f, 1.5f);
while(true) {
getClosest();
yield return new WaitForSeconds(1.5f);
}
}
getClosest is what you already have. Maybe it runs GameObject.Find, or uses a predone list. It’s split out of the “every 1.5 secs” loop so you can call it when your current target dies, or when you’re first spawned.