前端代码分割与懒加载:让你的应用飞起来

前端代码分割与懒加载:让你的应用飞起来

毒舌时刻

代码分割和懒加载?听起来就像是前端工程师为了掩饰自己代码写得太烂而发明的借口。你写的代码那么大,加载时间那么长,不分割能行吗?

你以为随便分割一下代码就能解决性能问题?别做梦了!如果分割策略不合理,反而会导致更多的网络请求,让应用变得更慢。

为什么你需要这个

  1. 减少初始加载时间:通过代码分割,只加载当前页面所需的代码,减少初始加载时间,提高用户体验。
  2. 优化资源利用:只加载用户需要的代码,避免加载不必要的资源,优化内存和带宽使用。
  3. 提高首屏渲染速度:快速加载首屏所需的代码,让用户尽快看到页面内容。
  4. 支持大型应用:对于大型应用,代码分割可以避免打包后的文件过大,导致加载时间过长。

反面教材

// 这是一个典型的不使用代码分割的应用 import React from 'react'; import ReactDOM from 'react-dom'; import Home from './pages/Home'; import About from './pages/About'; import Contact from './pages/Contact'; import Dashboard from './pages/Dashboard'; import Settings from './pages/Settings'; import UserProfile from './pages/UserProfile'; import AdminPanel from './pages/AdminPanel'; import { BrowserRouter as Router, Routes, Route } from 'react-router-dom'; function App() { return ( <Router> <Routes> <Route path="/" element={<Home />} /> <Route path="/about" element={<About />} /> <Route path="/contact" element={<Contact />} /> <Route path="/dashboard" element={<Dashboard />} /> <Route path="/settings" element={<Settings />} /> <Route path="/user-profile" element={<UserProfile />} /> <Route path="/admin" element={<AdminPanel />} /> </Routes> </Router> ); } ReactDOM.render(<App />, document.getElementById('root')); 

问题

  • 所有页面的代码都打包在一个文件中,导致初始加载时间过长
  • 即使用户只访问首页,也会加载所有页面的代码
  • 大型应用的打包文件会变得非常大,影响加载速度
  • 无法实现按需加载,浪费带宽和内存

正确的做法

React中的代码分割与懒加载

// 使用React.lazy和Suspense实现懒加载 import React, { lazy, Suspense } from 'react'; import ReactDOM from 'react-dom'; import { BrowserRouter as Router, Routes, Route } from 'react-router-dom'; // 懒加载页面组件 const Home = lazy(() => import('./pages/Home')); const About = lazy(() => import('./pages/About')); const Contact = lazy(() => import('./pages/Contact')); const Dashboard = lazy(() => import('./pages/Dashboard')); const Settings = lazy(() => import('./pages/Settings')); const UserProfile = lazy(() => import('./pages/UserProfile')); const AdminPanel = lazy(() => import('./pages/AdminPanel')); // 加载中组件 const Loading = () => <div>Loading...</div>; function App() { return ( <Router> <Suspense fallback={<Loading />}> <Routes> <Route path="/" element={<Home />} /> <Route path="/about" element={<About />} /> <Route path="/contact" element={<Contact />} /> <Route path="/dashboard" element={<Dashboard />} /> <Route path="/settings" element={<Settings />} /> <Route path="/user-profile" element={<UserProfile />} /> <Route path="/admin" element={<AdminPanel />} /> </Routes> </Suspense> </Router> ); } ReactDOM.render(<App />, document.getElementById('root')); 

Webpack中的代码分割

// webpack.config.js module.exports = { entry: { main: './src/index.js', }, output: { filename: '[name].[contenthash].js', chunkFilename: '[name].[contenthash].chunk.js', }, optimization: { splitChunks: { chunks: 'all', cacheGroups: { vendor: { test: /[\\/]node_modules[\\/]/, name: 'vendors', chunks: 'all', }, }, }, }, }; 

基于路由的代码分割

// 使用React Router和React.lazy实现基于路由的代码分割 import React, { lazy, Suspense } from 'react'; import { BrowserRouter as Router, Routes, Route, Navigate } from 'react-router-dom'; // 懒加载路由组件 const Home = lazy(() => import('./pages/Home')); const About = lazy(() => import('./pages/About')); const Contact = lazy(() => import('./pages/Contact')); const Dashboard = lazy(() => import('./pages/Dashboard')); const Settings = lazy(() => import('./pages/Settings')); const UserProfile = lazy(() => import('./pages/UserProfile')); const AdminPanel = lazy(() => import('./pages/AdminPanel')); // 加载中组件 const Loading = () => ( <div className="loading"> <div className="loading-spinner"></div> <p>Loading...</p> </div> ); // 错误边界组件 class ErrorBoundary extends React.Component { constructor(props) { super(props); this.state = { hasError: false }; } static getDerivedStateFromError(error) { return { hasError: true }; } componentDidCatch(error, errorInfo) { console.error('Error:', error); console.error('Error Info:', errorInfo); } render() { if (this.state.hasError) { return <div>Something went wrong. Please try again later.</div>; } return this.props.children; } } function App() { return ( <Router> <ErrorBoundary> <Suspense fallback={<Loading />}> <Routes> <Route path="/" element={<Home />} /> <Route path="/about" element={<About />} /> <Route path="/contact" element={<Contact />} /> <Route path="/dashboard" element={<Dashboard />} /> <Route path="/settings" element={<Settings />} /> <Route path="/user-profile" element={<UserProfile />} /> <Route path="/admin" element={<AdminPanel />} /> <Route path="*" element={<Navigate to="/" replace />} /> </Routes> </Suspense> </ErrorBoundary> </Router> ); } export default App; 

基于条件的代码分割

// 基于条件懒加载组件 import React, { lazy, Suspense, useState } from 'react'; // 懒加载重型组件 const HeavyComponent = lazy(() => import('./HeavyComponent')); function App() { const [showHeavyComponent, setShowHeavyComponent] = useState(false); return ( <div> <h1>Home Page</h1> <button onClick={() => setShowHeavyComponent(true)}> Load Heavy Component </button> {showHeavyComponent && ( <Suspense fallback={<div>Loading heavy component...</div>}> <HeavyComponent /> </Suspense> )} </div> ); } export default App; 

毒舌点评

代码分割和懒加载确实能提高应用的性能,但我见过太多开发者滥用这个特性,导致应用变得更加复杂。

想象一下,当你为了分割代码而分割代码,把原本简单的应用拆分成无数个小模块,结果导致网络请求变得更多,加载时间反而增加了。这不是得不偿失吗?

还有那些过度使用懒加载的开发者,连一些轻量级的组件也懒加载,结果导致用户点击按钮后还要等待组件加载,影响用户体验。

所以,在使用代码分割和懒加载时,一定要把握好度。不要为了分割而分割,要根据实际情况来决定哪些代码需要分割,哪些不需要。

当然,对于大型应用来说,代码分割和懒加载是必不可少的。但对于小型应用,过度的代码分割反而会增加开发成本和维护难度。

最后,记住一句话:代码分割和懒加载的目的是为了提高应用的性能和用户体验,而不是为了炫技。如果你的代码分割策略导致应用变得更慢,那你就失败了。

Read more

Submodular function次模函数 概念——AI学习

Submodular function次模函数 概念——AI学习

论文名称:Submodularity In Machine Learning and Artificial Intelligence 一、综述论文 这篇文章是一篇 综述论文(survey)。 核心目标是: 介绍 Submodular functions(次模函数) 以及它们在 机器学习与人工智能中的应用。 作者想说明一个非常重要的观点: 很多机器学习问题其实是“离散优化问题”。 例如: * Feature Selection:属于数据预处理问题,旨在从原始特征中筛选出最相关、最有信息量的子集,以降低维度、提升模型性能与可解释性。 * Dataset Subset Selection:属于数据采样或核心集选择问题,旨在从大规模数据中选取一个具有代表性的子集,以降低计算和存储成本,同时保持模型性能。 * Active Learning:属于机器学习训练策略问题,通过让模型主动选择最有价值的数据进行标注,以最少的标注成本最大化模型性能。 * Clustering:属于无监督学习问题,旨在根据数据的内在相似性,将未标记的数据自动分组为不同的类别或簇。 * Data

75元!复刻Moji 2.0 小智 AI 桌面机器人,基于乐鑫ESP32开发板,内置DeepSeek、Qwen大模型

文末联系小编,获取项目源码 Moji 2.0 是一个栖息在你桌面上的“有灵魂的伴侣”,采用乐鑫 ESP32-C5开发板,配置 1.5寸 360x360 高清屏,FPC 插接方式,支持 5G Wi-Fi 6 极速连接,内置小智 AI 2.0 系统,主要充当智能电子宠物的角色,在你工作学习枯燥时,通过圆形屏幕上的动态表情包卖萌解压,提供情绪陪伴;同时它也是功能强大的AI 语音助手,支持像真人一样流畅的连续对话,随时为你查询天气、解答疑惑或闲聊解闷,非常适合作为极客桌搭或嵌入式学习的开源平台。 🛠️ 装配进化 告别手焊屏幕的噩梦。全新设计的 FPC 插座连接,排线一插即锁,将复刻门槛降至最低。 🚀 性能进化 主控升级为 ESP32-C5。支持 5GHz Wi-Fi 6,

RISC-V开源处理器实战:从Verilog RTL设计到FPGA原型验证

RISC-V开源处理器实战:从Verilog RTL设计到FPGA原型验证

引言:开源浪潮下的RISC-V处理器设计 在芯片设计领域,RISC-V架构正以其开源免授权、模块化扩展和极简指令集三大优势重塑行业格局。与传统闭源架构不同,RISC-V允许开发者自由定制处理器核,从嵌入式微控制器到高性能服务器芯片均可覆盖。本文以Xilinx Vivado 2025工具链和蜂鸟E203处理器为核心,完整呈现从Verilog RTL设计到FPGA原型验证的全流程,为嵌入式工程师和硬件爱好者提供一套可复现的实战指南。 项目目标与技术栈 * 核心目标:基于RISC-V RV32I指令集,设计支持五级流水线的32位处理器核,实现基础算术运算、逻辑操作及访存功能,并在Xilinx Artix-7 FPGA开发板验证。 * 工具链:Xilinx Vivado 2025(逻辑设计、综合实现)、ModelSim(功能仿真)、Xilinx Artix-7 XC7A35T FPGA开发板(硬件验证)。 * 参考案例:蜂鸟E203处理器(芯来科技开源RISC-V核,已在Xilinx FPGA上完成移植验证,最高运行频率50MHz)。 一、数字系统设计流程:从需求到架构 1.

轻小说机翻机器人:5分钟打造你的日语小说翻译神器

轻小说机翻机器人:5分钟打造你的日语小说翻译神器 【免费下载链接】auto-novel轻小说机翻网站,支持网络小说/文库小说/本地小说 项目地址: https://gitcode.com/GitHub_Trending/au/auto-novel 轻小说机翻机器人是一款开源的日语小说翻译工具,支持网络小说、文库小说和本地小说的全自动翻译处理。作为专业的轻小说翻译解决方案,它能自动抓取日本主流平台内容,提供多引擎翻译服务,并构建完整的阅读生态,让日语阅读不再受语言障碍困扰。 🚀 核心价值:为什么选择轻小说机翻机器人? 全自动小说采集系统 内置对Kakuyomu、小説家になろう等6大日本小说平台的支持,只需输入小说名称或URL,系统即可智能抓取内容并完成翻译。通过crawler/src/lib/domain/目录下的平台适配代码(如kakuyomu.ts、syosetu.ts),实现对不同网站结构的精准解析。 多引擎翻译切换 集成百度翻译、有道翻译、OpenAI类API、Sakura等多种翻译器,满足从快速浏览到深度阅读的不同需求。翻译引擎实现代码位于web/src/do