How to Optimize Website Images: The Complete 2025 Guide
Images engage users, drive clicks, and generally make everything better–except for performance. Images are giant blobs of bytes that are usually the slowest part of your website. This guide has everything you need to know to optimize them for fast and responsive images for a fast, responsive website.

Why optimized images matter for web performance
Images are big. Really big. The bytes required for an image dwarf most site’s CSS and JavaScript assets. Slow images will damage your Core Web Vitals, impacting your SEO and costing you traffic. Images are usually the element driving Largest Contentful Paint and load delays can increase your Cumulative Layout Shift. If you’re not familiar with these metrics, check them out in the Definitive Guide to Measuring Web Performance.
To make your images fast, you need to make them as small as possible. The trick is making them small, while keeping them looking sharp and nice. This guide is for you! We’ll cover all the ways to make your images smaller:
Website image optimization checklist:
- Choose the right image format – Use JPG for photos, PNG for graphics, and modern formats like WebP or AVIF for better compression and quality.
- Optimize image size and metadata – Compress images with tools like TinyPNG and ImageMin, and remove unnecessary metadata to reduce file size.
- Lazy-load below-the-fold images – Use the loading=”lazy” attribute to delay loading non-critical images until they are needed, improving page speed.
- Use responsive images – Implement srcset and picture elements to serve different image sizes based on the user’s screen resolution.
- Switch to images optimized for mobile – Use media queries and aspect-ratio adjustments to serve mobile-friendly images without unnecessary large file sizes.
- Embed critical images – Convert small, essential images like icons and logos into Base64 or inline SVG to eliminate extra network requests.
- Cache and host images efficiently - Host your images from a CDN that serves with HTTP/2 or HTTP/3 and proper caching headers.
Want faster load times? Follow these steps and check your real-world performance with Request Metrics.
1. Choose the right image format
There are several different ways an image file can be formatted, like JPG
or PNG
. These formats are not interchangeable, and they were designed for different kinds of images. Using the wrong format can make a big image even worse. Let’s look an example, check out this image of Sam the Sloth:

This image is over twice as large when formatted as a JPG! The PNG format is much better for illustrations like this. A good general rule is to use JPG for photographs and use PNG for graphics. Let’s have a look at another image, a photograph:

Notice how graphic illustrations tend to be significantly smaller than photographs. Be sure to consider this when designing the look and feel of your pages.
PNG vs JPG vs WebP vs AVIF
Choosing the right image format helps balance quality and performance. Each format has strengths and trade-offs, so the best choice depends on your image type:
- JPG (JPEG) – Best for photos. JPG uses lossy compression, making it ideal for photographs and images with lots of colors and gradients. It offers smaller file sizes but may lose some detail if compressed too much.
- PNG – Best for graphics and transparency. PNG supports lossless compression and transparency, making it perfect for logos, icons, and detailed graphics. However, PNG files can be significantly larger than JPGs for detailed images like photography.
- WebP – Modern alternative for smaller files. WebP offers both lossy and lossless compression, often producing smaller file sizes than JPG and PNG while maintaining high quality. It’s widely supported in modern browsers and recommended for most images, although can be difficult to manipulate.
- AVIF – Best for maximum compression and quality. AVIF provides even better compression than WebP while preserving image quality. It supports transparency and HDR but isn’t yet universally supported across all browsers.
Best practice: Keep your original images in JPG or PNG, then use a build tool or CDN (like BunnyCDN) to automatically convert and serve them as WebP or AVIF for modern browsers.
Vector images (AKA SVG)
Vector images are a little different. Rather than storing the contents of the image, they store the instructions for how to draw it in XML. For simple illustrations, this tends to be very efficient, and enables a large image to be shown very efficiently. Here’s an example:

Because a vector describes how to draw an image, it can be shown at any size without any increase in size or reduction in quality.
A common mistake with vectors is trying to convert existing images. Exporting a PNG illustration as a SVG does not convert it to a vector! Likely, you just embedded a PNG inside the SVG XML and it will be slightly larger than the original file. SVGs have to be created from original paths and lines using a tool like Adobe Illustrator.
Get your free speed check to see your website performance for real users.
Free speed test

2. Optimize image quality
Aside from image format, there are often settings to adjust the quality of the image using lossy compression. These are algorithms that remove parts of an image that you wouldn’t notice, but still take up space. Check out this example:

This reduction is accomplished by pulling out unused colors, or by combining colors and pixels that too similar to notice. But you don’t need to worry about that, most optimization tools can detect the appropriate level of quality for an image. TinyPNG and ImageMin are great for this.
3. Lazy-Load below-the-fold images
Websites have a lot of images that don’t need to be loaded right away. But the browser doesn’t know what images it will need, so its gotta download them all! Downloading the extra images creates lag and network congestion that slows down the critical images.
Instead, tell the browser to be lazy! Lazy-loading is a technique to tell the browser to wait to download certain images until they are visible to the user. This can have huge impacts to Largest Contentful Paint (LCP) performance because the browser can focus on the critical elements to render.
Fortunately, the HTML spec let’s us tell the web browser that an image isn’t needed right away and it can lazy load it, with the loading
attribute. This is really useful for images that are not immediately visible “above the fold”.
<picture>
<img src="some-big-image.png"
alt="Image description" width="1600" height="600"
loading="lazy" />
</picture>
This will instruct the browser to wait to download the image until it is almost visible to the user.
Controlling load order with fetchpriority
You can even more fine-grained control of how images are downloaded using the fetchpriorty
attribute. This tells the browser what order to download images in, separately from how they appear in the markup.
<picture>
<img src="hero-image.png"
alt="Image description" width="1600" height="600"
fetchpriority="high" />
</picture>
For example, if your hero image is also your LCP Element, you probably want to set fetchpriority="high"
, so that it downloads as fast as possible.
Browser-compatibility
Both loading
and fetchpriorty
are supported in the latest major browsers, but if you want to have compatibility with older devices, you can polyfill lazy-loading behavior yourself with some short JavaScript:
<img data-src="some-big-image.png"
alt="Image description" width="1600" height="600" />
<script>
const lazyImages = [...document.querySelectorAll("[data-src]")];
const imageObserver = new IntersectionObserver((entries) => {
for(const entry of entries) {
if (entry.isIntersecting) {
const img = entry.target;
const src = img.getAttribute("data-src");
if (!img.src && src) {
img.src = src;
}
imageObserver.unobserve(img);
}
}
});
for(const img of lazyImages) {
imageObserver.observe(img);
}
</script>
This code looks for images that have a data-src
attribute instead of a src
one, so that the images don’t load right away. Then, it uses IntersectionObserver to detect when the images have intersected the viewport, and switches the attribute to src
, where it will download normally.
Layout sizing
With lazy-loading images, it is even more important to specify the sizing of images to prevent Layout Shift. Layout shifts are when the positioning of elements changes during load because of dynamically sized content. Read more about layout shift and the impact on performance.
You prevent this by specifying a height
and width
attribute on your images.
<img src="picture-1200.jpg"
loading="lazy" class="lazy"
width="1200" height="900" />
Notice that the height and width attributes. It’s not 1200px, it’s just 1200. It also doesn’t do exactly what you’d expect–the size of this element will not necessarily be 1200x900, it will be whatever the CSS styling and layout says it needs to be. But the browser remembers the aspect ratio from these attributes so that the height of an image will be correct given it’s width.
So if your layout only has 800px wide for the image, the browser will know, without downloading the image, to reserve 600px of height as well to maintain the aspect ratio.
4. Use responsive images
Visitors will view your website at different sizes. Some of your users will have a huge 1600px wide display. Others may have a 720px tablet or a 600px phone. A 2800px wide image would need a lot of wasteful bytes for those smaller devices, where the image will get scaled down anyway.

Responsive images let you keep that beautiful, high resolution image for visitors with large displays, and efficiently scale it down or replace the image for other devices. You can do this with the picture
and source
elements. Here’s an example where a 1600px wide image can be served at smaller 1200px or 700px sizes, depending on the size of the visitor’s screen:
<picture>
<source
sizes="(max-width: 1600px) 100vw, 1600px"
srcset="hero-desktop-700.png 700w,
hero-desktop-1200.png 1200w,
hero-desktop-1600.png 1600w" />
<img src="hero-desktop-1600.png"
alt="Image description" width="1600" height="600" />
</picture>
In the example above, we wrapped the img
tag in a picture
element that let’s us specify alternate sources based on media queries. The sizes
attribute says that for all screen sizes up to 1600px, use an image that is 100% of the viewport width. The srcset
attribute defines all the sizes of the image that we have. The first part is the URL and the second is the width of the image.
Most image tools, like Photoshop, Gimp, and Paint.NET, can export images at multiple resolutions. Your native image viewer can probably do some limited resizing as well. To automate this on a large scale, you may want to consider a command line tool like ImageMagick.
Let’s look at a real example. A visitor loads the size with a screen 800px wide, the browser loo would look for an image at least 800px, aad the first one it finds is hero-desktop-1200.png
, so it loads that one.
5. Switch to images optimized for mobile
For some websites, you may want to show a different image on mobile that has a different aspect ratio. Or maybe you want to hide the image altogether? Setting a display:none
style on the image isn’t very helpful because the browser will still waste time and bytes downloading the image. Instead, you can use multiple source
elements to tell the browser what image to show, and when.
<picture>
<source
media="(max-width: 700px)"
sizes="(max-width: 300px) 0, 100vw"
srcset="hero-mobile-400.png 400w,
hero-mobile-600.png 500w,
hero-mobile-700.png 700w" />
<source
media="(min-width: 701px)"
sizes="(max-width: 1600px) 100vw, 1600px"
srcset="hero-desktop-700.png 700w,
hero-desktop-1200.png 1200w,
hero-desktop-1600.png 1600w" />
<img src="hero-desktop-1600.png"
alt="Image description" width="1600" height="600" />
</picture>
In this second example, the first source
applies for screens less than 700px wide because of the media
attribute. For those screens, if it is less than 300px, the width of the space is 0, so the browser knows not to bother downloading anything. For anything greater than 300, its 100% of the viewport, and it has a set of hero-mobile-X.png
images to choose from.
Once the screen is greater than 700px, the next source element applies and shows a hero-desktop-X.png
image for the correct width.
6. Embed critical images
Sometimes an image is essential for a webpage to be useful, such as a button, logo, or icon. Once you’ve optimized it as small as you can make it, the only way to go faster is to embed the image on the page. Embedding an image saves the initial network request time and shows an image as soon as the document starts rendering.
You embed an image by converting it into a base64 string and putting it right in the html tag, like this:
<img src=""
alt="my awesome picture" />
This may look strange, but it’s 100% supported as a data url. The src
defines the format as a PNG image thats base64 encoded. The remainder is the actual contents of the image, in this case a small blue square.
Pop open devtools and look at the source of the image above, it’s embedded! It’s also quite long, over 3,000 characters. It will download faster than an image reference, but it also slows down the document itself, so use this technique sparingly.
Google uses embedded images frequently in the display of search results. One of the many reasons Google feels so fast. Check out our review of Google’s performance and what you can learn from it.
Here’s a handy web tool for converting your images to base64.
7. Cache and host images efficiently
Even perfectly optimized images can slow down your site if they aren’t delivered efficiently. Using modern protocols and caching strategies ensures your images load instantly for repeat visitors and across multiple pages.
Use HTTP/2 and HTTP/3 for faster image delivery
Older versions of HTTP load assets sequentially, meaning each image request has to wait in line. HTTP/2 and HTTP/3 allow multiple images to load simultaneously over a single connection, significantly reducing load times.
Most CDNs and modern web hosts already support HTTP/2, and many are rolling out HTTP/3, which reduces latency even further. If you’re hosting images yourself, check out why HTTP/3 is fast and ensure your server supports it.
Set proper caching headers
Once a visitor loads an image, their browser shouldn’t have to download it again every time they visit. You can control how long images are stored in a user’s cache using HTTP caching headers.
For static images that don’t change often (like logos or product photos), set a long cache expiration (Cache-Control: public, max-age=31536000, immutable). For dynamic images that may update, use ETag or Last-Modified headers to let the browser check for updates without downloading the full image again. Check out all the ways to cache images and other resources with our guide to HTTP Caching.
Want to see how well your site handles caching? Use our Cache Header Check Tool to analyze your headers.
Serve images from a CDN
A Content Delivery Network (CDN) stores images in multiple locations worldwide, so they load from the nearest server instead of your origin. This speeds up delivery and reduces bandwidth costs.
If your images are still served from a single data center, consider moving them to a CDN like Cloudflare, BunnyCDN, or Fastly for global performance gains.
Why HTTP compression doesn’t work for images
Some folks have asked me why they need to do anything of these things. That their whole website is served with Brotli HTTP compression, so shouldn’t the images already be as small as they can be?
It’s a good question, but no. Brotli compression (just like Gzip and Deflate) are text-based compression algorithms. They make dictionaries of common sets of characters to remove repetition. But there’s not a lot of repetition in a binary format like an image. If you’ve ever opened an image in a text editor, you know what I mean.
So instead, we need to reduce the number of bytes before sending them over the wire using the tips mentioned in this post.
Conclusion
These techniques will greatly speed your website and images. The correct format, resolution, quality, and load order for your images can transform the end user experience of your website for the better. Not sure how users see your website today? Try out performance monitoring from Request Metrics to track your real-user performance metrics.