- Published on
next 页面性能优化
- Authors
- Name
- AgedCoffee
- @__middle__child
开启 gzip 或者 brotli 压缩
一般如果动静分离已经将静态资源上传到了 CDN,正常 CDN 都是开启了以上的压缩方案的
如果没有开启需要开启相关能力
图片压缩
尽量使用 webp 格式图片和压缩图片,可以减少图片的请求大小,从而减少加载时间,提升性能。
如果是部署于 vercel 平台的服务图片默认开启相关的优化逻辑
图片懒加载
图片懒加载是一种常见的提升网页性能的技术,它会等到图片进入视窗再开始加载它们,可以避免一开始加载过多的资源。
<Image loading="lazy" src={renderItem.img} alt="" fill />
添加 dns-prefetch、preconnect meta 标签
DNS 预获取 (dns-prefetch) 和预连接 (preconnect) 是两种让浏览器提前进行域名解析和建立 TCP 握手的技术
在加载一个新的页面时,浏览器需要花费一定的时间来解析域名和建立 TCP 握手。我们可以通过添加 dns-prefetch 和 preconnect meta 标签来预先完成这些操作,从而显著提高页面加载速度
<Head>
<link rel="dns-prefetch" href="//example.com" />
<link rel="preconnect" href="//example.com" />
</Head>
自部署 next 服务的自定义图片 loader 优化
下面的代码展示了一个自定义的图片 loader,它可以将图片优化的功能外包到另一个服务。
'use client'
import React, { ComponentProps } from 'react';
import Image from 'next/image';
export const CustomLoaderImg = (props: ComponentProps<typeof Image>) => {
return (
<Image
{...props}
loader={({ src, width, quality }) => {
// 确认一个可用的图片优化服务之后替换
return `https://optimize-img.com?url=${src}?w=${width}&q=${quality || 75}`;
}}
/>
);
};
部分逻辑组件和渲染内容独立于服务端渲染
有些组件作为用户交互之后才需要展示的内容,例如预留好插槽的 Modal 组件,类似这样的组件没有服务器端渲染的必要,可以考虑修改他的实现为仅客户端渲染。同样,如果你正在渲染和时间有关的内容,由于服务器和客户端的时间可能存在微小差异,也可能导致错误。
这可以通过以下方法实行:
- 使用 next 提供的 dynamic
import dynamic from 'next/dynamic';
//仅在客户端动态导入并渲染 Modal 组件
const DynamicModal = dynamic(
() => import('../components/Modal'),
{ ssr: false } // 这将禁用服务器端渲染(SSR)
);
// 仅在客户端动态导入并渲染 Time 相关组件
const DynamicTime = dynamic(
() => import('../components/Time'),
{ ssr: false } // 这将禁用服务器端渲染(SSR)
);
- useEffect 中动态引入
import React, { useState, useEffect } from 'react';
function YourComponent() {
const [DynamicComponent, setDynamicComponent] = useState(null);
useEffect(() => {
import('../components/Modal').then((Modal) => {
setDynamicComponent(() => Modal.default); // 注意 Modal.default
});
}, []);
if (!DynamicComponent) return <p>Loading...</p>;
return <DynamicComponent />;
}
- 首帧之后才实际渲染最终内容
import React, { useState, useEffect } from 'react';
import Modal from '../components/Modal';
function YourComponent() {
const [isClient, setIsClient] = useState(false);
useEffect(() => {
// 在 useEffect 中设置标志表示现在是在客户端了
setIsClient(true);
}, []);
if (!isClient) return <p>Loading...</p>;
return <Modal />;
}
打包静态资源上传到 CDN 的插件
以下代码展示了如何将打包后的静态文件上传到 CDN,这样可以显著减少服务器的负载,同时由于 CDN 通常都具备很好的地理分布和加载优化,所以使用 CDN 可以进一步提升用户的加载体验。
/* eslint-disable react-func/max-lines-per-function */
/* eslint-disable max-depth */
const request = require('request');
function getCdnUrlPrefix(options) {
return `${options.packageVersion}/${options?.isEnvProduction ? 'prod' : 'dev'}`;
}
function UploadCDN(options) {
this.isEnvProduction = options?.isEnvProduction ?? false;
this.options = options;
}
function upload(options) {
return new Promise((resolve, reject) => {
request(options, function (error, response) {
if (error) reject(error);
resolve();
});
});
}
UploadCDN.prototype.apply = function (compiler) {
// eslint-disable-next-line react-func/max-lines-per-function
compiler.hooks.emit.tapAsync('UploadCDN', async (compilation, callback) => {
// formData list
let optionsMap = {};
let commonOptions = {
method: 'POST',
url: 'https://your-cdn-serverice/api/v1/files',
headers: {
module: this.options.module,
_dir: '/',
'Content-Type': 'application/x-www-form-urlencoded',
},
formData: {},
};
for (let fileName in compilation.assets) {
if (!fileName.endsWith('map')) {
const parts = fileName.split('/');
const filename = parts.pop();
const _dir = `${getCdnUrlPrefix(this.options)}/_next/${parts.join('/')}`;
const formData = {
value: compilation.assets[fileName].source(),
options: {
filename,
contentType: null,
},
};
if (!optionsMap[_dir]) {
optionsMap[_dir] = {
...JSON.parse(JSON.stringify(commonOptions)),
};
optionsMap[_dir].headers._dir = _dir;
}
optionsMap[_dir].formData[filename] = formData;
}
}
const optionsArr = Object.keys(optionsMap).map(key => optionsMap[key]);
optionsArr.forEach(async options => await upload(options));
callback();
});
};
module.exports = {
UploadCDN,
};
// next.config.js
// ...
assetPrefix: isProd ? `https://your-cdn-serverice/${yourModule}/${moduleName}/${packageVersion}/prod` : undefined,
webpack: (config, context) => {
isProd &&
config.plugins.push(
new UploadCDN({
module: moduleName,
packageVersion,
isEnvProduction: isProd,
}),
);
return config;
// ...