React Router 6的学习

安装react-router-dom

npm i react-router-dom

支持不同的路由创建

createBrowserRouter

特点

  1. 推荐使用的方式,基于 HTML5 的 History API
  2. 支持用户友好的 URL,无需 #
  3. 适用于生产环境的绝大多数场景。

适用

使用现代浏览器,支持 pushState 和 replaceState 的情况

应用

import { createBrowserRouter, RouterProvider } from "react-router-dom";
import Home from "../views/Home";
import About from "../views/About";
import ErrorPage from "../views/ErrorPage";const router = createBrowserRouter([{ path: "/", element: <Home /> },{ path: "/about", element: <About /> },{path: "*", // 404 页面element: <ErrorPage />,},
]);function App() {return <RouterProvider router={router} />;
}export default App;

tips

  1. 需要服务器支持,确保非根路径的页面请求被正确处理(常见问题是刷新后出现 404)
  2. 部署时,服务器通常需要配置以处理 SPA 应用

createHashRouter

特点

  1. URL 使用 #,如 http://example.com/#/about
  2. 不依赖服务器配置,适合快速开发和简单项目。
  3. 不推荐,因为 URL 不够优雅,且 SEO 支持较差

场景

  1. 静态站点部署(如 GitHub Pages)。
  2. 不支持 History API 的环境。

应用

import { createHashRouter, RouterProvider } from "react-router-dom";
import Home from "../views/Home";
import About from "../views/About";const router = createHashRouter([{ path: "/", element: <Home /> },{ path: "/about", element: <About /> },
]);function App() {return <RouterProvider router={router} />;
}export default App;

tips

  1. 无需服务器配置,但不适合对 URL 有美观或 SEO 要求的项目

createMemoryRouter

特点

  1. 不依赖浏览器环境,维护自己的内存中的路由栈。
  2. URL 不显示在浏览器地址栏
  3. 常用于非浏览器环境(如 React Native)或开发工具(如 Storybook)

场景

  1. 测试环境。
  2. 开发工具或需要自定义路由栈的应用

应用

import { createMemoryRouter, RouterProvider } from "react-router-dom";
import Home from "../views/Home";
import About from "../views/About";const router = createMemoryRouter([{ path: "/", element: <Home /> },{ path: "/about", element: <About /> },
]);function App() {return <RouterProvider router={router} />;
}export default App;

tips

  1. 开发中调试时有用,但生产环境通常不使用。
  2. URL 不可见,不适合需要 SEO 的场景

createStaticRouter

特点

  1. 用于服务器端渲染(SSR)。
  2. 基于预定义的静态路由配置。
  3. 无需用户交互,生成的路由用于服务端直接渲染

场景

  1. SSR(服务器端渲染)应用,例如使用 React Router 与 Next.js 或 Remix

应用

import { createStaticRouter, StaticRouterProvider } from "react-router-dom/server";
import Home from "../views/Home";
import About from "../views/About";const router = createStaticRouter([{ path: "/", element: <Home /> },{ path: "/about", element: <About /> },
]);function App() {return <StaticRouterProvider router={router} />;
}export default App;

tips

  1. createStaticRouter需要结合服务器环境。
  2. 主要用于预渲染生成的 HTML,例如静态页面生成器或 SSR 服务。

总结

路由方式特点适用场景优缺点
createBrowserRouter基于 HTML5 History API,推荐使用现代浏览器应用,生产环境URL 美观,需服务器支持
createHashRouterURL 包含 #,不依赖服务器快速开发、简单项目,静态站点部署URL 不优雅,SEO 支持较差
createMemoryRouter路由存储在内存中,无需浏览器环境测试环境、开发工具,无浏览器场景不适用于生产环境,URL 不可见
createStaticRouter静态路由,用于 SSR 和静态页面生成服务器端渲染依赖服务端,无客户端交互

路由配置

  1. 使用 RoutesRoute 代替 React Router 5 的 Switch 和路由定义方式。
  2. 通过嵌套路由支持嵌套组件,可以实现更清晰的层次结构。
  3. path 属性定义路由路径,element 属性指定要渲染的组件
// App.jsximport { 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="/user/:id" element={<User />} /></Routes></Router>);
}function Home() {return <h1>Home Page</h1>;
}function About() {return <h1>About Page</h1>;
}function User({ id }) {return <h1>User Page for {id}</h1>;
}

动态路由

  1. 动态路由通过 :paramName 定义路径参数。
  2. 使用 useParams 钩子获取参数值。
  3. 动态路由使得可以根据 URL 参数渲染不同的内容,useParams 是获取路径参数的关键。
import { useParams } from 'react-router-dom';function User() {const { id } = useParams(); // 获取路径中的 id 参数return <h1>User ID: {id}</h1>;
}

导航

  1. 使用 useNavigate 替代旧版本中的 useHistory
  2. useNavigate 提供更灵活的导航方式。
  3. navigate 可以接受路径字符串,也可以接受对象形式,适合处理编程式导航
import { useNavigate } from 'react-router-dom';function Home() {const navigate = useNavigate();const goToAbout = () => {navigate('/about');};return (<div><h1>Home Page</h1><button onClick={goToAbout}>Go to About</button></div>);
}

嵌套路由

  1. 支持组件内部嵌套路由。
  2. 子路由通过 Outlet 渲染。
  3. Outlet 是子路由的占位符,嵌套路由更适合复杂布局。
import { Outlet, Link } from 'react-router-dom';function Dashboard() {return (<div><h1>Dashboard</h1><nav><Link to="profile">Profile</Link><Link to="settings">Settings</Link></nav><Outlet /> {/* 渲染子路由 */}</div>);
}function App() {return (<Router><Routes><Route path="/" element={<Home />} /><Route path="dashboard" element={<Dashboard />}><Route path="profile" element={<Profile />} /><Route path="settings" element={<Settings />} /></Route></Routes></Router>);
}

重定向

  1. 使用 Navigate 组件进行页面重定向。
  2. 可在组件渲染时动态重定向。
  3. Navigate 是一个 JSX 组件,用于在路由中实现条件跳转
import { Navigate } from 'react-router-dom';function ProtectedRoute({ isAuth, children }) {return isAuth ? children : <Navigate to="/login" />;
}function App() {const isAuth = false; // 假设用户未登录return (<Router><Routes><Routepath="/protected"element={<ProtectedRoute isAuth={isAuth}><ProtectedPage /></ProtectedRoute>}/><Route path="/login" element={<Login />} /></Routes></Router>);
}

数据加载

  1. React Router 6.4+ 引入了 loaderuseLoaderData,用于预加载路由数据。
  2. 支持异步加载,优化页面渲染体验。
  3. loader 提供异步数据加载,提升数据获取的灵活性,useLoaderData 获取加载的数据。
import { createBrowserRouter, RouterProvider, useLoaderData } from 'react-router-dom';function UserProfile() {const data = useLoaderData(); // 获取预加载的数据return <h1>User Name: {data.name}</h1>;
}const router = createBrowserRouter([{path: "/user/:id",element: <UserProfile />,loader: async ({ params }) => {const response = await fetch(`/api/user/${params.id}`);return response.json();},},
]);function App() {return <RouterProvider router={router} />;
}

路由懒加载

  1. 减少首屏加载时间,提高性能;按需加载组件,降低内存占用;提升用户体验,特别是对于大型应用。
  2. 首次加载某个路由可能会有短暂延迟;如果 fallback 设计不当,可能导致用户体验下降;对 SEO 不友好,需配合服务器端渲染(SSR)解决。
  3. React.lazy(() => import('./ComponentPath')) 实现组件按需加载。
  4. Suspense用于显示加载状态(如 Loading...)直到懒加载组件加载完成
import React from "react";const Home = React.lazy(() => import("../views/Home"));
const About = React.lazy(() => import("../views/About"));
const Dashboard = React.lazy(() => import("../views/Dashboard"));const routes = [{path: "/",element: (<React.Suspense fallback={<div>Loading...</div>}><Home /></React.Suspense>),},{path: "/about",element: (<React.Suspense fallback={<div>Loading...</div>}><About /></React.Suspense>),},{path: "/dashboard",element: (<React.Suspense fallback={<div>Loading...</div>}><Dashboard /></React.Suspense>),},
];export default routes;
import { createBrowserRouter, RouterProvider } from "react-router-dom";
import routes from "./routes";const router = createBrowserRouter(routes);function App() {return <RouterProvider router={router} />;
}export default App;

路由保护

  1. 使用高阶组件模式或条件渲染实现路由保护。
  2. 常与 Navigate认证逻辑结合。
function ProtectedRoute({ isAuth, children }) {return isAuth ? children : <Navigate to="/login" />;
}

错误处理

  1. 支持 errorElement 处理路由级别的错误
const router = createBrowserRouter([{path: "/",element: <Home />,errorElement: <ErrorPage />, // 错误处理组件},
]);function ErrorPage() {return <h1>Something went wrong!</h1>;
}

综合案例

src/
├── router/
│   └── index.jsx  // 路由配置
├── views/
│   ├── Home.jsx   // 首页
│   ├── Login.jsx  // 登录页
│   ├── Content.jsx  // 内容页 (嵌套路由)
│   ├── ErrorPage.jsx // 错误页面
│   └── Protected.jsx // 受保护的页面
├── App.jsx        // 应用入口
└── main.jsx       // 渲染入口
// src/router/index.jsx
import { createBrowserRouter } from "react-router-dom";
import Home from "../views/Home";
import Login from "../views/Login";
import ErrorPage from "../views/ErrorPage";
import Content from "../views/Content";
import Protected from "../views/Protected";
import { Navigate } from "react-router-dom";const router = createBrowserRouter([{path: "/",element: <Navigate to="/home" replace />, // 重定向到 /home},{path: "/home",element: <Home />,errorElement: <ErrorPage />,children: [{path: "content",element: <Content />,},],},{path: "/login",element: <Login />,errorElement: <ErrorPage />,},{path: "/protected",element: <ProtectedRoute><Protected /></ProtectedRoute>,errorElement: <ErrorPage />,},{path: "*", // 404 页面element: <ErrorPage />,},
]);// 模拟受保护的路由
function ProtectedRoute({ children }) {const isAuthenticated = false; // 假设用户未登录return isAuthenticated ? children : <Navigate to="/login" replace />;
}export default router;

页面组件:src/views

// Home.jsx
import { Outlet, Link } from "react-router-dom";function Home() {return (<div><h1>Home Page</h1><nav><Link to="/home/content">Go to Content</Link><Link to="/protected">Go to Protected Page</Link></nav><Outlet /> {/* 嵌套路由 */}</div>);
}export default Home;
//Login.jsx
import { useNavigate } from "react-router-dom";function Login() {const navigate = useNavigate();const handleLogin = () => {// 模拟登录操作alert("Logged in!");navigate("/home"); // 登录后跳转到首页};return (<div><h1>Login Page</h1><button onClick={handleLogin}>Login</button></div>);
}export default Login;
//Content.jsx
function Content() {return <h1>Content Page</h1>;
}export default Content;
//Protected.jsx
function Protected() {return <h1>Protected Page: You are authenticated!</h1>;
}export default Protected;
//ErrorPage.jsx
function ErrorPage() {return <h1>404 Not Found</h1>;
}export default ErrorPage;
// src/App.js
import { RouterProvider } from "react-router-dom";
import router from "./router";function App() {return (<div><RouterProvider router={router} /></div>);
}export default App;
// src/index.js
import React from "react";
import ReactDOM from "react-dom";
import App from "./App";ReactDOM.render(<React.StrictMode><App /></React.StrictMode>,document.getElementById("root")
);

效果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/889272.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微信小程序web-view 嵌套h5界面 实现文件预览效果

实现方法&#xff1a;(这里我是在小程序里面单独加了一个页面用来下载预览文件) 安装 使用方法请参考文档 npm 安装 npm install weixin-js-sdk import wx from weixin-js-sdk预览 h5界面代码 <u-button click"onclick" type"primary" :loading"…

HTTP 状态码大全

常见状态码 200 OK # 客户端请求成功 400 Bad Request # 客户端请求有语法错误 不能被服务器所理解 401 Unauthorized # 请求未经授权 这个状态代码必须和WWW- Authenticate 报头域一起使用 403 Forbidden # 服务器收到请求但是拒绝提供服务 404 Not Found # 请求资源不存…

一文详解TCP协议 [图文并茂, 明了易懂]

欢迎来到啊妮莫的学习小屋! 目录 什么是TCP协议 TCP协议特点✨ TCP报文格式 三次握手和四次挥手✨ 可靠性 效率性 基于字节流✨ 基于TCP的应用层协议 什么是TCP协议 TCP(传输控制协议, Transmission Control Protocol) 是一种面向连接的, 可靠的, 基于字节流的传输层通…

在Linux(ubuntu22.04)搭建rust开发环境

1.安装rust 1.安装curl: sudo apt install curl 2.安装rust最新版 curl --proto ‘https’ --tlsv1.2 https://sh.rustup.rs -sSf | sh 安装完成后出现&#xff1a;Rust is installed now. Great! 重启当前shell即可 3.检验是否安装成功 rustc --version 结果出现&…

UnityShaderLab 实现程序化形状(一)

1.实现一个长宽可变的矩形&#xff1a; 代码&#xff1a; fixed4 frag (v2f i) : SV_Target{return saturate(length(saturate(abs(i.uv - 0.5)-0.13)))/0.03;} 2.实现一个半径可变的圆形&#xff1a; 代码&#xff1a; fixed4 frag (v2f i) : SV_Target{return (distance(a…

如何解决压测过程中JMeter堆内存溢出问题

如何解决压测过程中JMeter堆内存溢出问题 背景一、为什么会堆内存溢出&#xff1f;二、解决堆内存溢出措施三、堆内存参数应该怎么调整&#xff1f;四、堆内存大小配置建议 背景 Windows环境下使用JMeter压测运行一段时间后&#xff0c;JMeter日志窗口报错“java.lang.OutOfMe…

宽字节注入

尽管现在呼吁所有的程序都使用unicode编码&#xff0c;所有的网站都使用utf-8编码&#xff0c;来一个统一的国际规范。但仍然有很多&#xff0c;包括国内及国外&#xff08;特别是非英语国家&#xff09;的一些cms&#xff0c;仍然使用着自己国家的一套编码&#xff0c;比如gbk…

Web APIsPIs第1章

WebApi阶段学习什么&#xff1f; WebApi是浏览器提供的一组接口 使用 JavaScript 去操作页面文档 和 浏览器 什么是 API API: 应用程序接口&#xff08;Application Programming Interface&#xff09; 接口&#xff1a;本质上就是各种函数&#xff0c;无需关心内部如何实现…

android——录制屏幕

录制屏幕 1、界面 2、核心代码 import android.app.NotificationChannel import android.app.NotificationManager import android.app.PendingIntent import android.app.Service import android.content.Context import android.content.Intent import android.graphics.Bi…

【Excel学习记录】01-认识Excel

1.之前的优秀软件Lotus-1-2-3 默认公式以等号开头 兼容Lotus-1-2-3的公式写法&#xff0c;不用写等号 &#xff1a; 文件→选项→高级→勾选&#xff1a;“转换Lotus-1-2-3公式(U)” 备注&#xff1a;对于大范围手动输入公式可以使用该选项&#xff0c;否则请不要勾选&#x…

短视频矩阵抖音SEO源码OEM独立部署

短视频优化矩阵源码涉及对抖音平台上的视频内容进行筛选与排序&#xff0c;目的是增强其在搜索引擎中的可见度&#xff0c;以便更多用户能够浏览到这些视频。而抖音SEO优化系统则是通过构建一个分析框架&#xff0c;来解析抖音上的用户数据、视频信息及标签等元素&#xff0c;并…

MySQL——buffer poll

为什么要有buffer poll&#xff1f; 如果没有buffer poll&#xff0c;每次读取数据的时候都是从磁盘上读的&#xff0c;这样效率是很差的的。 所以有了提高效率的方式&#xff0c;就加上了一个缓存——buffer poll 所以&#xff0c;当我们读取数据的时候就有以下的方式 当读…

生产慎用之调试日志对空间矢量数据批量插入的性能影响-以MybatisPlus为例

目录 前言 一、一些缘由 1、性能分析 二、插入方式调整 1、批量插入的实现 2、MP的批量插入实现 3、日志的配置 三、默认处理方式 1、基础程序代码 2、执行情况 四、提升调试日志等级 1、在logback中进行设置 2、提升后的效果 五、总结 前言 在现代软件开发中&#xff0c;性能优…

元宇宙时代的社交平台:Facebook的愿景与实践

随着科技的不断进步&#xff0c;元宇宙&#xff08;Metaverse&#xff09;这一概念逐渐走进了人们的视野。作为全球最大的社交平台之一&#xff0c;Facebook&#xff08;现Meta&#xff09;在这场元宇宙革命中扮演着重要角色。Meta不仅在不断扩展其社交平台的边界&#xff0c;还…

C# 小案例(IT资产管理系统)

开发工具&#xff1a;visual studio 2022 语言&#xff1a;C# 数据库&#xff1a;Sql Server 2008 页面展示 一、登录 二、主窗体 三、用户管理 四、资产管理 五、关于 Java版地址&#xff1a;基于若依开发物品管理系统(springbootvue)_若依物品管理系统-CSDN博客 Python版…

分布式日志系统设计

一、分布式日志系统定义 分布式日志系统是一种用于收集、存储和分析大规模分布式系统日志的系统。它可以帮助开发人员和系统管理员实时监控和调试系统&#xff0c;提高系统可靠性和可用性&#xff0c;同时也可以用于日志分析和故障排查。 二、简单设计思路 日志收集&#xff…

敏捷开发04:Scrum 中的 Product Backlog(产品待办列表) 详细介绍

Product Backlog 产品待办列表 在计划开发产品功能时&#xff0c;都希望产品功能上线后&#xff0c;用户能够喜欢并经常使用。 因此在开发产品新功能时&#xff0c;就要衡量哪些产品需求是对用户最有价值&#xff0c;这是最应该思考的问题。 然后把这些有价值的需求集合放在一…

vmware vsphere5---部署vCSA(VMware vCenter Server)附带第二阶段安装报错解决方案

声明 因为这份文档我是边做边写的&#xff0c;遇到问题重新装了好几次所以IP会很乱 ESXI主机为192.168.20.10 VCSA为192.168.20.7&#xff0c;后台为192.168.20.7:5480 后期请自行对应&#xff0c;后面的192.168.20.57请对应192.168.20.7&#xff0c;或根据自己的来 第一阶段…

110.【C语言】编写命令行程序(1)

目录 1.前置知识 "命令"的含义 运行C语言程序 2.介绍 main函数的参数 实验1 执行结果 实验2 执行结果 修改代码 实验3 分析 方法:遍历数组argv[]中的所有参数 执行结果 修改代码 执行结果 1.前置知识 "命令"的含义 WINR输入cmd,在cmd窗口下…

Leecode刷题C语言之半有序排列

执行结果:通过 执行用时和内存消耗如下&#xff1a; 代码如下&#xff1a; int semiOrderedPermutation(int* nums, int numsSize) {int first 0, last 0;for (int i 0; i < numsSize; i) {if (nums[i] 1) {first i;}if (nums[i] numsSize) {last i;}}return firs…