This skill should be used when the user asks about "Unity performance", "optimization", "GC allocation", "object pooling", "caching", "Update loop optimization", "memory management", "profiling",...
Essential performance optimization techniques for Unity games, covering memory management, CPU optimization, rendering, and profiling strategies.
Performance is critical for Unity games across all platforms. Poor performance manifests as low framerates, stuttering, long load times, and crashes. This skill covers proven optimization techniques that apply to all Unity projects.
Core optimization areas:
The most common Unity performance mistake is repeated expensive lookups. Cache all references to avoid redundant operations.
Never call GetComponent repeatedly - cache results in Awake:
// ❌ BAD - GetComponent every frame
private void Update()
{
GetComponent<Rigidbody>().velocity = Vector3.forward; // SLOW!
}
// ✅ GOOD - Cache once
private Rigidbody rb;
private void Awake()
{
rb = GetComponent<Rigidbody>();
}
private void Update()
{
rb.velocity = Vector3.forward; // FAST
}
Performance impact: GetComponent is 10-100x slower than cached reference.
Cache transform access, especially for frequently accessed GameObjects:
// ❌ BAD - Property access overhead
private void Update()
{
transform.position += Vector3.forward * Time.deltaTime;
transform.rotation = Quaternion.identity;
}
// ✅ GOOD - Cache transform reference
private Transform myTransform;
private void Awake()
{
myTransform = transform;
}
private void Update()
{
myTransform.position += Vector3.forward * Time.deltaTime;
myTransform.rotation = Quaternion.identity;
}
Why: transform property has overhead. Cached reference eliminates repeated lookups.
Never use Find methods in Update - cache results:
// ❌ BAD - Find every frame (EXTREMELY SLOW)
private void Update()
{
GameObject player = GameObject.Find("Player");
Transform target = GameObject.FindWithTag("Enemy").transform;
}
// ✅ GOOD - Cache in Start
private GameObject player;
private Transform target;
private void Start()
{
player = GameObject.Find("Player");
target = GameObject.FindWithTag("Enemy")?.transform;
}
private void Update()
{
// Use cached references
}
Performance impact: Find methods scan entire scene hierarchy. 100-1000x slower than cached references.
Access renderer.material creates new Material instance - cache to avoid leaks:
// ❌ BAD - Creates new Material every frame (MEMORY LEAK)
private void Update()
{
GetComponent<Renderer>().material.color = Color.red; // Creates new Material!
}
// ✅ GOOD - Cache material reference
private Material material;
private void Awake()
{
material = GetComponent<Renderer>().material;
}
private void Update()
{
material.color = Color.red; // Modifies cached Material
}
private void OnDestroy()
{
// Clean up instantiated material
if (material != null)
Destroy(material);
}
Critical: Accessing .material creates new instance. Use .sharedMaterial for read-only access to avoid instantiation.
Instantiate and Destroy are expensive. Reuse objects instead of creating/destroying repeatedly.
public class ObjectPool : MonoBehaviour
{
[SerializeField] private GameObject prefab;
[SerializeField] private int initialSize = 10;
private Queue<GameObject> pool = new Queue<GameObject>();
private void Awake()
{
// Pre-instantiate objects
for (int i = 0; i < initialSize; i++)
{
GameObject obj = Instantiate(prefab);
obj.SetActive(false);
pool.Enqueue(obj);
}
}
public GameObject Get()
{
if (pool.Count > 0)
{
GameObject obj = pool.Dequeue();
obj.SetActive(true);
return obj;
}
// Pool exhausted - create new
return Instantiate(prefab);
}
public void Return(GameObject obj)
{
obj.SetActive(false);
pool.Enqueue(obj);
}
}
Use for:
Performance gain: 10-50x faster than Instantiate/Destroy, eliminates GC spikes.
public class BulletSpawner : MonoBehaviour
{
[SerializeField] private ObjectPool bulletPool;
[SerializeField] private Transform firePoint;
private void Fire()
{
// Get from pool
GameObject bullet = bulletPool.Get();
bullet.transform.position = firePoint.position;
bullet.transform.rotation = firePoint.rotation;
// Return after 3 seconds
StartCoroutine(ReturnToPoolAfterDelay(bullet, 3f));
}
private IEnumerator ReturnToPoolAfterDelay(GameObject obj, float delay)
{
yield return new WaitForSeconds(delay);
bulletPool.Return(obj);
}
}
Update, FixedUpdate, and LateUpdate are called frequently - minimize work done in these methods.
// ❌ BAD - Empty methods still have overhead
private void Update() { }
private void FixedUpdate() { }
// ✅ GOOD - Remove unused methods
// (No Update/FixedUpdate if not needed)
Performance: Unity calls all Update methods even if empty. Remove to reduce overhead.
Not all logic needs to run every frame:
// ❌ BAD - Expensive check every frame
private void Update()
{
CheckForNearbyEnemies(); // Expensive raycast/distance checks
}
// ✅ GOOD - Check every N frames
private int frameCounter = 0;
private const int checkInterval = 10;
private void Update()
{
frameCounter++;
if (frameCounter >= checkInterval)
{
frameCounter = 0;
CheckForNearbyEnemies();
}
}
// ✅ BETTER - Use InvokeRepeating or Coroutine
private void Start()
{
InvokeRepeating(nameof(CheckForNearbyEnemies), 0f, 0.2f); // Every 0.2 seconds
}
Alternative: Coroutines
private void Start()
{
StartCoroutine(CheckEnemiesRoutine());
}
private IEnumerator CheckEnemiesRoutine()
{
while (true)
{
CheckForNearbyEnemies();
yield return new WaitForSeconds(0.2f);
}
}
Replace polling with events:
// ❌ BAD - Poll for state change every frame
private bool wasGrounded;
private void Update()
{
bool grounded = IsGrounded();
if (grounded != wasGrounded)
{
OnGroundedChanged(grounded);
}
wasGrounded = grounded;
}
// ✅ GOOD - Event-driven
public event Action<bool> OnGroundedChanged;
private bool isGrounded;
private void SetGrounded(bool grounded)
{
if (isGrounded != grounded)
{
isGrounded = grounded;
OnGroundedChanged?.Invoke(grounded);
}
}
Avoid allocations in frequently-called methods to prevent GC spikes.
// ❌ BAD - Allocates strings every frame
private void Update()
{
string message = "Health: " + health; // String allocation
scoreText.text = "Score: " + score; // String allocation
}
// ✅ GOOD - Use StringBuilder or string interpolation
private StringBuilder sb = new StringBuilder();
private void UpdateUI()
{
sb.Clear();
sb.Append("Health: ").Append(health);
healthText.text = sb.ToString();
}
// ✅ ALTERNATIVE - Cache formatted strings
private void UpdateHealth(int newHealth)
{
health = newHealth;
healthText.text = health.ToString(); // Less allocation than concatenation
}
// ❌ BAD - Allocates new list every frame
private void Update()
{
List<Enemy> nearbyEnemies = new List<Enemy>(); // GC allocation!
FindNearbyEnemies(nearbyEnemies);
}
// ✅ GOOD - Reuse list
private List<Enemy> nearbyEnemies = new List<Enemy>();
private void Update()
{
nearbyEnemies.Clear(); // Reuse existing list
FindNearbyEnemies(nearbyEnemies);
}
// ❌ BAD - ToArray allocates
private void Update()
{
GameObject[] enemies = enemyList.ToArray(); // GC allocation!
}
// ✅ GOOD - Iterate list directly
private void Update()
{
for (int i = 0; i < enemyList.Count; i++)
{
Enemy enemy = enemyList[i];
// Process enemy
}
}
// ✅ GOOD - Use foreach (no allocation for List)
private void Update()
{
foreach (var enemy in enemyList)
{
// Process enemy
}
}
// ❌ BAD - Allocates WaitForSeconds every call
private IEnumerator DelayedAction()
{
yield return new WaitForSeconds(1f); // New allocation each time
}
// ✅ GOOD - Cache WaitForSeconds
private WaitForSeconds oneSecondWait = new WaitForSeconds(1f);
private IEnumerator DelayedAction()
{
yield return oneSecondWait; // Reuse cached wait
}
// ❌ BAD - Multiple GetComponent calls
private void OnTriggerEnter(Collider other)
{
if (other.GetComponent<Enemy>() != null)
{
other.GetComponent<Enemy>().TakeDamage(10); // Called twice!
}
}
// ✅ GOOD - Single GetComponent with pattern matching
private void OnTriggerEnter(Collider other)
{
if (other.TryGetComponent<Enemy>(out var enemy))
{
enemy.TakeDamage(10); // Called once
}
}
// ❌ BAD - GetComponent on every collision
private void OnTriggerEnter(Collider other)
{
var damageable = other.GetComponent<IDamageable>();
if (damageable != null)
damageable.TakeDamage(10);
}
// ✅ GOOD - Cache component on trigger enter
private Dictionary<Collider, IDamageable> damageableCache = new Dictionary<Collider, IDamageable>();
private void OnTriggerEnter(Collider other)
{
if (!damageableCache.TryGetValue(other, out var damageable))
{
damageable = other.GetComponent<IDamageable>();
damageableCache[other] = damageable; // Cache for future collisions
}
damageable?.TakeDamage(10);
}
private void OnTriggerExit(Collider other)
{
damageableCache.Remove(other); // Clean up cache
}
Configure Physics Layer Collision Matrix to prevent unnecessary collision checks:
Edit > Project Settings > Physics > Layer Collision Matrix
// Setup layers
Layer 8: Player
Layer 9: Enemies
Layer 10: Projectiles
Layer 11: Environment
// Disable unnecessary collisions:
- Player vs Player (disabled)
- Enemies vs Enemies (disabled)
- Projectiles vs Projectiles (disabled)
Performance gain: 30-50% reduction in physics overhead.
// ❌ BAD - Raycast checks everything
bool hit = Physics.Raycast(origin, direction, out RaycastHit hitInfo);
// ✅ GOOD - Layer mask limits checks
int layerMask = 1 << LayerMask.NameToLayer("Enemy");
bool hit = Physics.Raycast(origin, direction, out RaycastHit hitInfo, maxDistance, layerMask);
// ✅ BETTER - Cache layer mask
private int enemyLayerMask;
private void Awake()
{
enemyLayerMask = 1 << LayerMask.NameToLayer("Enemy");
}
private void Fire()
{
bool hit = Physics.Raycast(origin, direction, out RaycastHit hitInfo, maxDistance, enemyLayerMask);
}
Let Rigidbody sleep when not moving:
// Rigidbody automatically sleeps when velocity < threshold
// Configure in Edit > Project Settings > Physics
// Wake up manually when needed
private void ApplyForce()
{
if (rb.IsSleeping())
rb.WakeUp();
rb.AddForce(force);
}
Measure before optimizing. Use Unity Profiler to identify actual bottlenecks.
Open Profiler: Window > Analysis > Profiler
CPU Usage:
Memory:
Enable Deep Profile for detailed call stack (impacts performance):
Warning: Deep Profile slows game significantly. Use for small scenes or targeted profiling.
Measure specific code sections:
using Unity.Profiling;
public class AIController : MonoBehaviour
{
private static readonly ProfilerMarker s_PathfindingMarker = new ProfilerMarker("AI.Pathfinding");
private static readonly ProfilerMarker s_DecisionMarker = new ProfilerMarker("AI.DecisionMaking");
private void Update()
{
s_PathfindingMarker.Begin();
CalculatePath();
s_PathfindingMarker.End();
s_DecisionMarker.Begin();
MakeDecision();
s_DecisionMarker.End();
}
private void CalculatePath() { }
private void MakeDecision() { }
}
Shows custom markers in Profiler for precise measurement.
Set performance targets for each system:
Target: 60 FPS (16.67ms per frame)
Monitor with Profiler and optimize systems exceeding budget.
Key concerns:
Mobile-specific optimizations:
More headroom but still optimize:
For detailed performance techniques, consult:
references/memory-optimization.md - Advanced GC reduction, allocation patternsreferences/rendering-optimization.md - Draw call batching, GPU optimization, shadersreferences/physics-optimization.md - Collision optimization, Rigidbody best practicesreferences/profiling-guide.md - Complete profiling workflows, tools, analysisCaching priorities:
Avoid in Update:
Always profile before optimizing:
Apply these performance practices consistently for smooth, responsive Unity games across all platforms.