路由

安装

输入以下命令进行安装:

  1. # npm
  2. npm install react-router-dom
  3. # yarn
  4. yarn add react-router-dom

react-router相关标签

react-router常用的组件有以下八个:

  1. import {
  2. BrowserRouter,
  3. HashRouter,
  4. Route,
  5. Redirect,
  6. Switch,
  7. Link,
  8. NavLink,
  9. withRouter,
  10. } from 'react-router-dom'

简单路由跳转

实现一个简单的一级路由跳转

  1. import {
  2. BrowserRouter as Router,
  3. Route,
  4. Link
  5. } from 'react-router-dom'
  6. import Home from './home'
  7. import About from './about'
  8. function App() {
  9. return (
  10. <div className="App">
  11. <Router>
  12. <Link to="/home" className="link">跳转Home页面</Link>
  13. <Link to="/about" className="link">跳转About页面</Link>
  14. <Route path="/home" component={Home}/>
  15. <Route path="/about" component={About}/>
  16. </Router>
  17. </div>
  18. );
  19. }
  20. export default App;

效果如下:
image.gifReact Router - 图2
要点总结:

  1. Route组件必须在Router组件内部
  2. Link组件的to属性的值为点击后跳转的路径
  3. Route组建的path属性是与Link标签的to属性匹配的; component属性表示Route组件匹配成功后渲染的组件对象

    嵌套路由跳转

    React 的路由匹配层级是有顺序的
    例如,在 App 组件中,设置了两个路由组件的匹配路径,分别是 /home/about,代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. Link,
    5. } from 'react-router-dom'
    6. import Home from './home'
    7. import About from './about'
    8. function App() {
    9. return (
    10. <div className="App">
    11. <Router>
    12. <Link to="/home">跳转Home页面</Link>
    13. <Link to="/about">跳转About页面</Link>
    14. <Route path="/home" component={Home}/>
    15. <Route path="/about" component={About}/>
    16. </Router>
    17. </div>
    18. );
    19. }
    20. export default App;

    然后 Home 组件中同样也想设置两个路由组件的匹配路径,分别是 /home/one/home/two,此时就可以看出,这个 /home/one/home/two 为上一级路由 /home 的二级嵌套路由,代码如下:

    1. import React from 'react'
    2. import {
    3. Route,
    4. Link,
    5. } from 'react-router-dom'
    6. import One from './one'
    7. import Two from './two'
    8. function Home () {
    9. return (
    10. <>
    11. 我是Home页面
    12. <Link to="/home/one">跳转到Home/one页面</Link>
    13. <Link to="/home/two">跳转到Home/two页面</Link>
    14. <Route path="/home/one" component={One}/>
    15. <Route path="/home/two" component={Two}/>
    16. </>
    17. )
    18. }
    19. export default Home

    :::danger 特别注意: Home 组件中的路由组件 One 的二级路由路径匹配必须要写 /home/one ,而不是 /one ,不要以为 One 组件看似在 Home 组件内就可以简写成 /one :::

    动态链接

    NavLink可以将当前处于active状态的链接附加一个active类名,例如:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink
    5. } from 'react-router-dom'
    6. import Home from './home'
    7. import About from './about'
    8. function App() {
    9. return (
    10. <div className="App">
    11. <Router>
    12. <NavLink to="/home" className="link">跳转Home页面</NavLink>
    13. <NavLink to="/about" className="link">跳转About页面</NavLink>
    14. <Route path="/home" component={Home}/>
    15. <Route path="/about" component={About}/>
    16. </Router>
    17. </div>
    18. );
    19. }
    20. export default App;
    1. /* 设置active类的样式 */
    2. .active {
    3. font-weight: blod;
    4. color: red;
    5. }

    效果如下:
    640.gif

    路由匹配优化

    当点击跳转链接时,会自动去尝试匹配所有的Route对应的路径,如图所示:
    React Router - 图4
    正常情况下,只需匹配到一个规则,渲染即可,即匹配成功一个后,无需进行后续的匹配尝试,此时可以用Switch组件,如下所示:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. } from 'react-router-dom'
    7. import Home from './home'
    8. import About from './about'
    9. function App() {
    10. return (
    11. <div className="App">
    12. <Router>
    13. <NavLink to="/home" className="link">跳转Home页面</NavLink>
    14. <NavLink to="/about" className="link">跳转About页面</NavLink>
    15. <Switch>
    16. <Route path="/home" component={Home}/>
    17. <Route path="/about" component={About}/>
    18. <Route path="/home" component={Home}/>
    19. <Route path="/home" component={Home}/>
    20. {/* 此处省略一万个Route组件 */}
    21. <Route path="/home" component={Home}/>
    22. </Switch>
    23. </Router>
    24. </div>
    25. );
    26. }
    27. export default App;

    效果如下:
    React Router - 图5
    要点总结:

  4. 将多个Route组件同时放在一个Switch组件中,即可避免多次无意义的路由匹配,以此提升性能

    重定向

    当页面跳转时,若跳转链接没有匹配上任何一个 Route 组件,那么就会显示 404 页面,所以需要一个重定向组件 Redirect ,代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. Redirect,
    7. } from 'react-router-dom'
    8. import Home from './home'
    9. import About from './about'
    10. function App() {
    11. return (
    12. <div className="App">
    13. <Router>
    14. <NavLink to="/home" className="link">跳转Home页面</NavLink>
    15. <NavLink to="/about" className="link">跳转About页面</NavLink>
    16. <NavLink to="/shop" className="link">跳转Shop页面</NavLink> {/* 点击,跳转到/shop,但该路径没有设置 */}
    17. <Switch>
    18. <Route path="/home" component={Home}/>
    19. <Route path="/about" component={About}/>
    20. <Redirect to="/home" /> {/* 当以上Route组件都匹配失败时,重定向到/home */}
    21. </Switch>
    22. </Router>
    23. </div>
    24. );
    25. }
    26. export default App;

    效果如下:
    React Router - 图6

    路由传参

    所有路由传递的参数,都会在跳转路由组件的 props 中获取到,每种传参方式接收的方式略有不同
    路由传参的方式一共有三种,依次来看一下

    第一种

    第一种是在 Link 组件的跳转路径上携带参数,并在 Route 组件的匹配路径上通过 :参数名 的方式接收参数,代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. } from 'react-router-dom'
    7. import Home from './home'
    8. import About from './about'
    9. function App() {
    10. return (
    11. <div className="App">
    12. <Router>
    13. {/* 在 /home 的路径上携带了 张三、18 共两个参数 */}
    14. <NavLink to="/home/张三/18" className="link">跳转Home页面</NavLink>
    15. <NavLink to="/about" className="link">跳转About页面</NavLink>
    16. <Switch>
    17. {/* 在 /home 匹配路径上相同的位置接收了 name、age 两个参数 */}
    18. <Route path="/home/:name/:age" component={Home}/>
    19. <Route path="/about" component={About}/>
    20. </Switch>
    21. </Router>
    22. </div>
    23. );
    24. }
    25. export default App;

    尝试跳转,并打印一下路由组件的 props
    React Router - 图7
    可以看到,第一种方式的参数是通过 props.match.params 来获取的

    第二种

    第二种方式就是通过在 Link 组件的跳转链接后面跟上以 ? 开头,类似 ?a=1&b=3 这样的参数进行传递,代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. } from 'react-router-dom'
    7. import Home from './home'
    8. import About from './about'
    9. function App() {
    10. return (
    11. <div className="App">
    12. <Router>
    13. {/* 在跳转路径后面以?开头传递两个参数,分别为name=张三、age=18 */}
    14. <NavLink to="/home?name=张三&age=18" className="link">跳转Home页面</NavLink>
    15. <NavLink to="/about" className="link">跳转About页面</NavLink>
    16. <Switch>
    17. {/* 此处无需做接收操作 */}
    18. <Route path="/home" component={Home}/>
    19. <Route path="/about" component={About}/>
    20. </Switch>
    21. </Router>
    22. </div>
    23. );
    24. }
    25. export default App;

    尝试跳转,并打印一下路由组件的 props
    React Router - 图8
    可以看到,第二种方式的参数是通过 props.location.search 来获取的,不过这里的参数需要自己简单做进一步转化。

    第三种

    第三种方式就是以对象的形式编写 Link 组件的 to 跳转属性,并通过 state 属性来传递参数,代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. } from 'react-router-dom'
    7. import Home from './home'
    8. import About from './about'
    9. function App() {
    10. return (
    11. <div className="App">
    12. <Router>
    13. {/* 以对象的形式描述to属性,路径属性名为pathname,参数属性名为state */}
    14. <NavLink to={{pathname: "/home", state: {name: '张三', age: 18}}} className="link">跳转Home页面</NavLink>
    15. <NavLink to="/about" className="link">跳转About页面</NavLink>
    16. <Switch>
    17. {/* 此处无需特地接收属性 */}
    18. <Route path="/home" component={Home}/>
    19. <Route path="/about" component={About}/>
    20. </Switch>
    21. </Router>
    22. </div>
    23. );
    24. }
    25. export default App;

    尝试跳转,并打印一下路由组件的 props
    React Router - 图9
    可以看到,第三种方式的参数是通过 props.location.state 来获取的

    函数式路由

    以上主要都是通过 react-router-dom 中的 Link 组件来往某个路由组件跳转
    但有时,需要更灵活的方式进行跳转路由,例如通过调用一个函数,随时随地进行路由跳转,这就叫函数式路由
    函数式路由用到的方法有以下 5 个(下方截图来自路由组件props
    React Router - 图10
    5 个方法分别是 pushreplacegoForwardgoBackgo,接下来按顺序介绍一下这几个方法

    push

    push 方法就是使页面跳转到对应路径,并在浏览器中留下记录(即可以通过浏览器的回退按钮,返回上一个页面)
    举个例子:在路由组件 Home 中设置一个按钮 button ,点击后调用 push 方法,跳转到 /about 页面

    1. import React from 'react'
    2. function Home (props) {
    3. let pushLink = () => {
    4. props.history.push('/about')
    5. }
    6. return (
    7. <div className="a">
    8. 我是Home页面
    9. <button onClick={pushLink}>跳转到about页面</button>
    10. </div>
    11. )
    12. }
    13. export default Home

    跳转效果如下:
    React Router - 图11
    可以看到,通过 push 方法跳转以后,可以通过浏览器的回退按钮,返回上一个页面

    replace

    replace 方法与 push 方法类似,不一样的地方就是,跳转后不会在浏览器中保存上一个页面的记录(即无法通过浏览器的回退按钮,返回上一个页面)
    改动一下代码

    1. import React from 'react'
    2. function Home (props) {
    3. let replaceLink = () => {
    4. props.history.replace('/about')
    5. }
    6. return (
    7. <div className="a">
    8. 我是Home页面
    9. <button onClick={replaceLink}>跳转到about页面</button>
    10. </div>
    11. )
    12. }
    13. export default Home

    跳转效果如下:
    React Router - 图12
    可以看到,刚开始的路径是 ‘/‘ ,然后跳转到 ‘/home’ ,再点击按钮,通过 replace 方法跳转到 /about 页面。最后通过浏览器的回退按钮返回到了 / 页面,说明中间的 /home 没有被存在浏览器的记录里

    goForward

    调用 goForward 方法,就相当于点击了浏览器的返回下一个页面按钮,如下图所示:
    React Router - 图13

    goBack

    调用 goBack 方法,就相当于点击了浏览器的返回上一个页面的按钮,如下图所示:
    React Router - 图14

    go

    go 方法顾名思义,是用于跳转到指定路径的。
    该方法接受一个参数(参数类型为 Number),情况如下:

  5. 当参数为正数 n 时,表示跳转到下 n 个页面。例如 go(1) 相当于调用了一次 goForward 方法

  6. 当参数为负数 n 时,表示跳转到上 n 个页面。例如 go(-3) 相当于调用了三次 goBack 方法
  7. 当参数为 0 时,表示刷新当前页面

    普通组件使用路由

    这里区分两个概念,分别为 普通组件路由组件
    通过 Route 组件渲染的组件为路由组件 ,其余的基本上都为 普通组件
    例如,下方代码中:Home 组件为路由组件 ; App 组件为普通组件

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. } from 'react-router-dom'
    7. import Home from './home'
    8. export default function App() {
    9. return (
    10. <div className="App">
    11. <Router>
    12. <NavLink to='/home' className="link">跳转Home页面</NavLink>
    13. <Switch>
    14. <Route path="/home" component={Home}/>
    15. </Switch>
    16. </Router>
    17. </div>
    18. );
    19. }

    然后,路由组件跟普通组件最大的区别就是,组件的 props 属性中是否有下图所示的内容:(前者有,后者无)
    React Router - 图15
    此时,react-router-dom 提供了一个 withRouter 方法,可以使普通组件也能像路由组件一样有那些方法或数据可以使用
    使用方法如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. NavLink,
    5. Switch,
    6. withRouter, // 1. 引入 witRouter
    7. } from 'react-router-dom'
    8. import About from './about'
    9. function App(props) {
    10. console.log(props); // 3. 尝试打印普通组件App的props,发现此时props中已有内容了,即普通组件也能拥有跟路由组件一样类似的功能
    11. return (
    12. <div className="App">
    13. <Router>
    14. <NavLink to="/about" className="link">跳转About页面</NavLink>
    15. <Switch>
    16. <Route path="/about" component={About}/>
    17. </Switch>
    18. </Router>
    19. </div>
    20. );
    21. }
    22. export default withRouter(App); // 2. 通过withRouter方法对普通组件做一层包装处理

    补充

    replace

    在函数式路由里跳转类型主要有两种,分别是 pushreplace,那么在非函数式路由中,同样也可以自定义跳转类型,具体的实现代码如下:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. Link
    5. } from 'react-router-dom'
    6. import Home from './home'
    7. import About from './about'
    8. function App() {
    9. return (
    10. <div className="App">
    11. <Router>
    12. <Link to="/home" className="link">跳转Home页面</Link>
    13. <Link to="/about" className="link">跳转About页面</Link>
    14. <Route path="/home" component={Home} replace={true}/> {/* replace为true,跳转类型为replace */}
    15. <Route path="/about" component={About} replace={false}/> {/* replace为false,跳转类型为push */}
    16. </Router>
    17. </div>
    18. );
    19. }
    20. export default App;

    Route 组件上有个 replace 属性可以设定跳转类型,当值为 true 时,跳转类型为 replace ; 为 false 时,跳转类型为 push

    excat

    路由的匹配默认是模糊匹配的,举个例子:

    1. import {
    2. BrowserRouter as Router,
    3. Route,
    4. Link,
    5. } from 'react-router-dom'
    6. import Home from './home'
    7. import About from './about'
    8. function App() {
    9. return (
    10. <div className="App">
    11. <Router>
    12. <Link to="/home/abc">跳转Home页面</Link> {/* 跳转到/home/abc,但实际home下没有abc这个路由组件 */}
    13. <Link to="/about/abc">跳转About页面</Link> {/* 跳转到/about/abc,但实际home下也没有abc这个路由组件 */}
    14. <Route path="/home" component={Home} /> {/* 路由匹配规则为/home,没有设置exact属性,当前为模糊匹配 */}
    15. <Route path="/about" component={About} exact/> {/* 路由匹配规则为/about,设置了exact属性,当前为精准匹配 */}
    16. </Router>
    17. </div>
    18. );
    19. }
    20. export default App;

    效果如下:
    React Router - 图16
    图中看出,因为跳转 /home/abc 时,第一个 Route 组件是模糊匹配的,所以先匹配到了 /home,因此 Home 组件渲染了 ; 而跳转 /about/abc 时,第二个 Route 组件是精准匹配的,即 /about/abc 不等于 /about,所以 About 组件也没有渲染
    总结:

  8. 如果想要精准匹配的话,只需要将 Route 组件的 exact 属性设置为 true 即可

  9. 精准匹配要谨慎使用,因为可能会影响嵌套路由的使用