Lo struct SpinWait
permette di mettere in pausa un Thread utilizzando lo spinning, quindi facendogli fare del lavoro inutile in loop.
Effettuando un wait con spinning usando un classico while
può portare a numerosi problemi:
- Il thread consumerà tutti i core della CPU fino a che
condition
non saràtrue
; - Gli altri Thread dell’applicazione rallenteranno (starvation);
- Perfino il Thread che dovrà mettere
condition
atrue
rallenterà, portando ad un loop di rallentamento orribile (chiamato priority inversion); - In caso di CPU single core la priority inversion è pressoché certa.
La struct SpinWait
risolve questi problemi in due modi:
- Limita il numero di spinning CPU intensive ad un massimo numero di iterazioni: dopo aver raggiunto tale numero massimo fa lo yield del time slice a lui assegnato utilizzando
Thread.Sleep
oThread.Yield
(ricordo che il thread scheduler del sistema operativo assegna dei time slice ad ogni thread e rapidamente continua a cambiare l’esecuzione, aumento inoltre il parallelismo in base al numero di CPU a disposizione); - Rileva se sta lavorando su una CPU single core e in quel caso fa lo
yield
ad ogni ciclo.
SpinWait
si può utilizzare con la sua classe statica in questo modo:
che di fatto aspetta che myPredicate()
diventi true per al massimo 1000 ms.
Aspettare facendo spinning per 1000ms non ha senso, potrebbe essere una idea più interessante provare lo spinning per, esempio, 10ms e successivamente un Thread.sleep
classico.
Una alternativa è utilizzare il metodo SpinOnce()
all’interno di un while
che esce su una determinata condition
Oppure è possibile utilizzare la property NextSpinWillYield
per poter essere intenzionale nel block:
E’ estremamente interessante e ben commentato il codice ufficiale di Microsoft, quindi consiglio di darci una letta.