Un'introduzione all'API delle prestazioni

Pubblicato: 2022-08-10

L'API Performance misura la reattività della tua applicazione Web live su dispositivi di utenti reali e connessioni di rete. Può aiutare a identificare i colli di bottiglia nel codice lato client e lato server con:

  • tempistica utente: misurazione personalizzata delle prestazioni della funzione JavaScript lato client
  • tempo di pittura: metriche di rendering del browser
  • temporizzazione delle risorse: caricamento delle prestazioni delle risorse e delle chiamate Ajax
  • tempi di navigazione: metriche di caricamento della pagina, inclusi reindirizzamenti, ricerche DNS, disponibilità DOM e altro

L'API risolve diversi problemi associati alla tipica valutazione delle prestazioni:

  1. Gli sviluppatori spesso testano le applicazioni su PC di fascia alta collegati a una rete veloce. DevTools può emulare dispositivi più lenti, ma non metterà sempre in evidenza i problemi del mondo reale quando la maggior parte dei client utilizza un cellulare di due anni connesso al Wi-Fi dell'aeroporto.
  2. Le opzioni di terze parti come Google Analytics sono spesso bloccate, portando a risultati e ipotesi distorte. Potresti anche riscontrare implicazioni sulla privacy in alcuni paesi.
  3. L'API Performance può misurare con precisione varie metriche meglio di metodi come Date() .

Vuoi saperne di più sull'utilizzo dell'API Performance? Inizia qui... Clicca per twittare
Le sezioni seguenti descrivono i modi in cui è possibile utilizzare l'API Performance. Si consiglia una certa conoscenza di JavaScript e delle metriche di caricamento delle pagine.

Disponibilità dell'API delle prestazioni

La maggior parte dei browser moderni supporta l'API Performance, inclusi IE10 e IE11 (anche IE9 ha un supporto limitato). Puoi rilevare la presenza dell'API utilizzando:

 if ('performance' in window) { // use Performance API }

Non è possibile riempire completamente l'API con Polyfill, quindi fai attenzione ai browser mancanti. Se il 90% dei tuoi utenti naviga felicemente con Internet Explorer 8, misureresti solo il 10% dei client con applicazioni più capaci.

L'API può essere utilizzata in Web Workers, che forniscono un modo per eseguire calcoli complessi in un thread in background senza interrompere le operazioni del browser.

La maggior parte dei metodi API può essere utilizzata in Node.js lato server con il modulo perf_hooks standard:

 // Node.js performance import { performance } from 'node:perf_hooks'; // or in Common JS: const { performance } = require('node:perf_hooks'); console.log( performance.now() );

Deno fornisce l'API Performance standard:

 // Deno performance console.log( performance.now() );

Dovrai eseguire script con l'autorizzazione --allow-hrtime per abilitare la misurazione del tempo ad alta risoluzione:

 deno run --allow-hrtime index.js

Le prestazioni lato server sono generalmente più facili da valutare e gestire perché dipendono da carico, CPU, RAM, dischi rigidi e limiti del servizio cloud. Gli aggiornamenti hardware o le opzioni di gestione dei processi come PM2, clustering e Kubernetes possono essere più efficaci del refactoring del codice.

Le sezioni seguenti si concentrano sulle prestazioni lato client per questo motivo.

Misurazione personalizzata delle prestazioni

L'API Performance può essere utilizzata per cronometrare la velocità di esecuzione delle funzioni dell'applicazione. Potresti aver usato o incontrato funzioni di temporizzazione usando Date() :

 const timeStart = new Date(); runMyCode(); const timeTaken = new Date() - timeStart; console.log(`runMyCode() executed in ${ timeTaken }ms`);

L'API Performance offre due vantaggi principali:

  1. Migliore precisione: Date() misura il millisecondo più vicino, ma l'API Performance può misurare frazioni di millisecondo (a seconda del browser).
  2. Migliore affidabilità: l'utente o il sistema operativo possono modificare l'ora del sistema, quindi le metriche basate su Date() non saranno sempre accurate. Ciò significa che le tue funzioni potrebbero apparire particolarmente lente quando gli orologi avanzano!

L'equivalente di Date() è performance.now() che restituisce un timestamp ad alta risoluzione che viene impostato a zero all'avvio del processo responsabile della creazione del documento (la pagina è stata caricata):

 const timeStart = performance.now(); runMyCode(); const timeTaken = performance.now() - timeStart; console.log(`runMyCode() executed in ${ timeTaken }ms`);

Una proprietà performance.timeOrigin non standard può anche restituire un timestamp dal 1° gennaio 1970 sebbene questo non sia disponibile in IE e Deno.

performance.now() diventa impraticabile quando si effettuano più di poche misurazioni. L'API Performance fornisce un buffer in cui è possibile registrare l'evento per un'analisi successiva passando il nome di un'etichetta a performance.mark() :

 performance.mark('start:app'); performance.mark('start:init'); init(); // run initialization functions performance.mark('end:init'); performance.mark('start:funcX'); funcX(); // run another function performance.mark('end:funcX'); performance.mark('end:app');

È possibile estrarre un array di tutti gli oggetti mark nel buffer Performance utilizzando:

 const mark = performance.getEntriesByType('mark');

Esempio di risultato:

 [ { detail: null duration: 0 entryType: "mark" name: "start:app" startTime: 1000 }, { detail: null duration: 0 entryType: "mark" name: "start:init" startTime: 1001 }, { detail: null duration: 0 entryType: "mark" name: "end:init" startTime: 1100 }, ... ]

Il metodo performance.measure() calcola il tempo tra due contrassegni e lo memorizza anche nel buffer Performance. Si passa un nuovo nome di misura, il nome del segno iniziale (o null per misurare dal caricamento della pagina) e il nome del segno finale (o null per misurare l'ora corrente):

 performance.measure('init', 'start:init', 'end:init');

Un oggetto PerformanceMeasure viene aggiunto al buffer con la durata calcolata. Per ottenere questo valore, puoi richiedere un array di tutte le misure:

 const measure = performance.getEntriesByType('measure');

o richiedere una misura con il suo nome:

 performance.getEntriesByName('init');

Esempio di risultato:

 [ { detail: null duration: 99 entryType: "measure" name: "init" startTime: 1001 } ]

Utilizzo del buffer delle prestazioni

Oltre a indicatori e misure, il buffer delle prestazioni viene utilizzato per registrare automaticamente i tempi di navigazione, i tempi delle risorse e i tempi di disegno (di cui parleremo più avanti). È possibile ottenere un array di tutte le voci nel buffer:

 performance.getEntries();

Per impostazione predefinita, la maggior parte dei browser fornisce un buffer che memorizza fino a 150 metriche di risorse. Questo dovrebbe essere sufficiente per la maggior parte delle valutazioni, ma puoi aumentare o diminuire il limite del buffer se necessario:

 // record 500 metrics performance.setResourceTimingBufferSize(500);

I segni possono essere cancellati per nome oppure puoi specificare un valore vuoto per cancellare tutti i segni:

 performance.clearMarks('start:init');

Allo stesso modo, le misure possono essere cancellate per nome o un valore vuoto per cancellare tutto:

 performance.clearMeasures();

Monitoraggio degli aggiornamenti del buffer delle prestazioni

Un PerformanceObserver può monitorare le modifiche al buffer delle prestazioni ed eseguire una funzione quando si verificano eventi specifici. La sintassi risulterà familiare se hai utilizzato MutationObserver per rispondere agli aggiornamenti DOM o IntersectionObserver per rilevare quando gli elementi vengono fatti scorrere nel viewport.

È necessario definire una funzione osservatore con due parametri:

  1. un array di voci di osservatore che sono state rilevate, e
  2. l'oggetto osservatore. Se necessario, il suo metodo disconnect() può essere chiamato per fermare l'osservatore.
 function performanceCallback(list, observer) { list.getEntries().forEach(entry => { console.log(`name : ${ entry.name }`); console.log(`type : ${ entry.type }`); console.log(`start : ${ entry.startTime }`); console.log(`duration: ${ entry.duration }`); }); }

La funzione viene passata a un nuovo oggetto PerformanceObserver. Il suo metodo observe() riceve un array di entryType del buffer delle prestazioni da osservare:

 let observer = new PerformanceObserver( performanceCallback ); observer.observe({ entryTypes: ['mark', 'measure'] });

In questo esempio, l'aggiunta di un nuovo contrassegno o misura esegue la funzione performanceCallback() . Sebbene qui registri solo i messaggi, potrebbe essere utilizzato per attivare un caricamento di dati o eseguire ulteriori calcoli.

Misurazione delle prestazioni della vernice

L'API Paint Timing è disponibile solo in JavaScript lato client e registra automaticamente due metriche importanti per Core Web Vitals:

  1. first-paint: il browser ha iniziato a disegnare la pagina.
  2. first-contentful-paint: il browser ha dipinto il primo elemento significativo del contenuto DOM, come un'intestazione o un'immagine.

Questi possono essere estratti dal buffer delle prestazioni in un array:

 const paintTimes = performance.getEntriesByType('paint');

Fai attenzione a non eseguirlo prima che la pagina sia completamente caricata; i valori non saranno pronti. Attendi l'evento window.load o usa un PerformanceObserver per monitorare i tipi di input di paint .

Lottando con tempi di inattività e problemi con WordPress? Kinsta è la soluzione di hosting progettata per farti risparmiare tempo! Scopri le nostre caratteristiche

Esempio di risultato:

 [ { "name": "first-paint", "entryType": "paint", "startTime": 812, "duration": 0 }, { "name": "first-contentful-paint", "entryType": "paint", "startTime": 856, "duration": 0 } ]

Una prima vernice lenta è spesso causata da CSS o JavaScript che bloccano il rendering. Il divario rispetto alla prima vernice di contenuto potrebbe essere ampio se il browser deve scaricare un'immagine di grandi dimensioni o eseguire il rendering di elementi complessi.

Misurazione delle prestazioni delle risorse

I tempi di rete per risorse come immagini, fogli di stile e file JavaScript vengono registrati automaticamente nel buffer delle prestazioni. Anche se c'è poco che puoi fare per risolvere i problemi di velocità della rete (oltre a ridurre le dimensioni dei file), può aiutare a evidenziare problemi con risorse più grandi, risposte Ajax lente o script di terze parti con prestazioni scadenti.

Una matrice di metriche PerformanceResourceTiming può essere estratta dal buffer utilizzando:

 const resources = performance.getEntriesByType('resource');

In alternativa, puoi recuperare le metriche per una risorsa passando il suo URL completo:

 const resource = performance.getEntriesByName('https://test.com/script.js');

Esempio di risultato:

 [ { connectEnd: 195, connectStart: 195, decodedBodySize: 0, domainLookupEnd: 195, domainLookupStart: 195, duration: 2, encodedBodySize: 0, entryType: "resource", fetchStart: 195, initiatorType: "script", name: "https://test.com/script.js", nextHopProtocol: "h3", redirectEnd: 0, redirectStart: 0, requestStart: 195, responseEnd: 197, responseStart: 197, secureConnectionStart: 195, serverTiming: [], startTime: 195, transferSize: 0, workerStart: 195 } ]

Si possono esaminare le seguenti proprietà:

  • nome : URL risorsa
  • entryType : "risorsa"
  • initiatorType : come è stata avviata la risorsa, ad esempio "script" o "link"
  • serverTiming : una matrice di oggetti PerformanceServerTiming passati dal server nell'intestazione HTTP Server-Timing (l'applicazione lato server potrebbe inviare le metriche al client per ulteriori analisi)
  • startTime : timestamp di inizio del recupero
  • nextHopProtocol : protocollo di rete utilizzato
  • workerStart : timestamp prima dell'avvio di un ruolo di lavoro del servizio app Web progressivo (0 se la richiesta non viene intercettata da un ruolo di lavoro del servizio)
  • redirectStart : timestamp quando è iniziato un reindirizzamento
  • redirectEnd : timestamp dopo l'ultimo byte dell'ultima risposta di reindirizzamento
  • fetchStart : timestamp prima del recupero della risorsa
  • domainLookupStart : timestamp prima di una ricerca DNS
  • domainLookupEnd : timestamp dopo la ricerca DNS
  • connectStart : timestamp prima di stabilire una connessione al server
  • connectEnd : timestamp dopo aver stabilito una connessione al server
  • secureConnectionStart : timestamp prima dell'handshake SSL
  • requestStart : timestamp prima che il browser richieda la risorsa
  • responseStart : Timestamp quando il browser riceve il primo byte di dati
  • responseEnd : timestamp dopo la ricezione dell'ultimo byte o la chiusura della connessione
  • duration : la differenza tra startTime e responseEnd
  • transferSize : la dimensione della risorsa in byte inclusa l'intestazione e il corpo compresso
  • encodedBodySize : il corpo della risorsa in byte prima della decompressione
  • decodedBodySize : il corpo della risorsa in byte dopo la decompressione

Questo script di esempio recupera tutte le richieste Ajax avviate dall'API Fetch e restituisce la dimensione e la durata totali del trasferimento:

 const fetchAll = performance.getEntriesByType('resource') .filter( r => r.initiatorType === 'fetch') .reduce( (sum, current) => { return { transferSize: sum.transferSize += current.transferSize, duration: sum.duration += current.duration } }, { transferSize: 0, duration: 0 } );

Misurazione delle prestazioni di navigazione

I tempi di rete per scaricare la pagina precedente e caricare la pagina corrente vengono registrati automaticamente nel buffer Performance come un singolo oggetto PerformanceNavigationTiming .

Estrailo in un array usando:

 const pageTime = performance.getEntriesByType('navigation');

...o passando l'URL della pagina a .getEntriesByName() :

 const pageTiming = performance.getEntriesByName(window.location);

Le metriche sono identiche a quelle per le risorse ma includono anche valori specifici della pagina:

  • entryType : Es. "navigazione"
  • digita : "naviga", "ricarica", "indietro_avanti" o "prerenderizzare"
  • redirectCount : il numero di reindirizzamenti
  • unloadEventStart : timestamp prima dell'evento di scaricamento del documento precedente
  • unloadEventEnd : timestamp dopo l'evento di scaricamento del documento precedente
  • domInteractive : Timestamp quando il browser ha analizzato l'HTML e costruito il DOM
  • domContentLoadedEventStart : Timestamp prima dell'attivazione dell'evento DOMContentLoaded del documento
  • domContentLoadedEventEnd : Timestamp dopo il completamento dell'evento DOMContentLoaded del documento
  • domComplete : timestamp dopo il completamento della costruzione del DOM e degli eventi DOMContentLoaded
  • loadEventStart : timestamp prima dell'attivazione dell'evento di caricamento della pagina
  • loadEventEnd : timestamp dopo l'evento di caricamento della pagina e tutte le risorse sono disponibili

I problemi tipici includono:

  • Un lungo ritardo tra unloadEventEnd e domInteractive . Ciò potrebbe indicare una risposta del server lenta.
  • Un lungo ritardo tra domContentLoadedEventStart e domComplete . Ciò potrebbe indicare che gli script di avvio della pagina sono troppo lenti.
  • Un lungo ritardo tra domComplete e loadEventEnd . Ciò potrebbe indicare che la pagina ha troppe risorse o che molte richiedono troppo tempo per essere caricate.

Registrazione e analisi delle prestazioni

L'API Performance ti consente di raccogliere dati sull'utilizzo del mondo reale e caricarli su un server per ulteriori analisi. Puoi utilizzare un servizio di terze parti come Google Analytics per archiviare i dati, ma c'è il rischio che lo script di terze parti venga bloccato o introduca nuovi problemi di prestazioni. La tua soluzione può essere personalizzata in base alle tue esigenze per garantire che il monitoraggio non influisca su altre funzionalità.

Fai attenzione alle situazioni in cui non è possibile determinare le statistiche, forse perché gli utenti utilizzano vecchi browser, bloccano JavaScript o sono dietro un proxy aziendale. Capire quali dati mancano può essere più fruttuoso che formulare ipotesi basate su informazioni incomplete.

Idealmente, gli script di analisi non influiranno negativamente sulle prestazioni eseguendo calcoli complessi o caricando grandi quantità di dati. Prendi in considerazione l'utilizzo di web worker e la riduzione al minimo dell'uso di chiamate localStorage sincrone. È sempre possibile elaborare in batch i dati grezzi in un secondo momento.

Infine, fai attenzione ai valori anomali come dispositivi e connessioni molto veloci o molto lenti che influiscono negativamente sulle statistiche. Ad esempio, se nove utenti caricano una pagina in due secondi ma il decimo sperimenta un download di 60 secondi, la latenza media arriva a quasi 8 secondi. Una metrica più realistica è la cifra mediana (2 secondi) o il 90° percentile (9 utenti su 10 sperimentano un tempo di caricamento di 2 secondi o meno).

Riepilogo

Le prestazioni Web rimangono un fattore critico per gli sviluppatori. Gli utenti si aspettano che i siti e le applicazioni siano reattivi sulla maggior parte dei dispositivi. Anche l'ottimizzazione per i motori di ricerca può essere influenzata dal downgrade dei siti più lenti in Google.
Tutto ciò che devi sapere per iniziare con l'API Performance è proprio qui Fai clic per twittare
Esistono molti strumenti di monitoraggio delle prestazioni, ma la maggior parte valuta le velocità di esecuzione lato server o utilizza un numero limitato di client in grado di valutare il rendering del browser. L'API delle prestazioni fornisce un modo per raccogliere le metriche degli utenti reali che non sarebbe possibile calcolare in nessun altro modo.