If a third-party script is slowing down your page load, you have two options to improve performance:
- Remove it if it doesn’t add clear value to your site.
- Optimize the loading process.
This post explains how to optimize the loading process of third-party scripts with the following techniques:
- Using the
- Establishing early connections to required origins
- Lazy loading
- Optimizing how you serve third-party scripts
Because synchronous scripts delay DOM construction and rendering, you should always load third-party scripts asynchronously unless the script has to run before the page can be rendered.
defer attributes tell the browser that it may go on parsing the HTML while loading the script in the background, and then execute the script after it loads. This way, script downloads don’t block DOM construction and page rendering. The result is that the user can see the page before all scripts have finished loading.
<script async src="script.js"> <script defer src="script.js">
The difference between
defer is when they start executing the scripts.
Scripts with the
async attribute execute at the first opportunity after they finish downloading and before the window’s load event. This means it’s possible (and likely) that
async scripts will not be executed in the order in which they appear in the HTML. It also means they can interrupt DOM building if they finish downloading while the parser is still at work.
Scripts with the
defer attribute execute after HTML parsing is completely finished, but before the
defer guarantees scripts will be executed in the order they appear in the HTML and will not block the parser.
asyncif it’s important to have the script run earlier in the loading process.
deferfor less critical resources. A video player that’s below-the-fold, for example.
Using these attributes can significantly speed up page load. For example, Telegraph recently deferred all of their scripts, including ads and analytics, and improved the ad loading time by an average of four seconds.
Establish early connections to required origins #
You can save 100–500 ms by establishing early connections to important third-party origins.
<link> types can help here:
<link rel="preconnect"> informs the browser that your page intends to establish a connection to another origin, and that you’d like the process to start as soon as possible. When the request for a resource from the pre-connected origin is made, the download starts immediately.
<link rel="preconnect" href="https://cdn.example.com">
<link rel="dns-prefetch> handles a small subset of what is handled by
<link rel="preconnect">. Establishing a connection involves the DNS lookup and TCP handshake, and for secure origins, TLS negotiations.
dns-prefetch instructs the browser to only resolve the DNS of a specific domain before it has been explicitly called.
preconnect hint is best used for only the most critical connections; for less critical third-party domains use
<link rel="dns-prefetch" href="http://example.com">
Browser support for
dns-prefetch is slightly different from
preconnect support, so
dns-prefetch can serve as a fallback for browsers that don’t support
preconnect. Use separate link tags to implement this safely:
<link rel="preconnect" href="http://example.com"> <link rel="dns-prefetch" href="http://example.com">
Lazy-load third-party resources #
Embedded third-party resources can be a big contributor to slow page speed when constructed poorly. If they aren’t critical or are below the fold (that is, if users have to scroll to view them), lazy loading is a good way to improve page speed and paint metrics. This way, users will get the main page content faster and have a better experience.
One effective approach is to lazy-load third-party content after the main page content loads. Ads are a good candidate for this approach.
Ads are an important source of income for many sites, but users come for the content. By lazy loading ads and delivering the main content faster, you can increase the overall viewability percentage of an ad. For example, MediaVine switched to lazy-loading ads and saw a 200% improvement in page load speed. DoubleClick have guidance on how to lazy-load ads in their official documentation.
An alternative approach is to load third-party content only when users scroll down to that section of the page.
iframes. It supports YouTube embeds and widgets. It also has optional support for IntersectionObserver.
Optimize how you serve third-party scripts #
Third-party CDN hosting #
But because they are not on the same origin as the rest of your resources, loading files from a public CDN comes with a network cost. The browser needs to perform a DNS lookup, establish a new HTTP connection, and, on secure origins, perform an SSL handshake with the vendor’s server.
When you use files from third-party servers, you rarely have control over caching. Relying on someone else’s caching strategy might cause scripts to be unnecessarily re-fetched from the network too often.
Self-host third-party scripts #
Self-hosting third-party scripts is an option that gives you more control over a script’s loading process. By self-hosting you can:
- Reduce DNS lookup and round-trip times.
- Improve HTTP caching headers.
- Take advantage of HTTP/2, or the newer HTTP/3.
For example, Casper managed to shave 1.7 seconds off load time by self-hosting an A/B testing script.
Self-hosting comes with one big downside though: scripts can go out of date and won’t get automatic updates when there’s an API change or a security fix.
Use service workers to cache scripts from third-party servers #
An alternative to self-hosting that allows you greater control over caching while still getting the third-party CDN benefits is using service workers to cache scripts from third-party servers. This gives you control over how often scripts are re-fetched from the network and makes it possible to create a loading strategy that throttles requests for non-essential third-party resources until the page reaches a key user moment. Using
preconnect to establish early connections in this case can also mitigate the network costs to an extent.
source : web.dev