手动实现react-router路由拦截

其实这个话题在网上找已经能找到很多解决方案了,之所以写这篇文章主要也是让自己对路由拦截加深点印象吧(毕竟自己没实现过路由拦截。。。)。用vue时已经有导航守卫,在公司里做项目路由拦截的部分也早已经写好,那就只能自己做demo练习一下了。react-router本身没提供导航守卫,按作者的说法是可以自己来实现,那就自己来实现一下吧。

我在上家公司用的是react-router4,现在已经有react-router6可以使用了,我们先使用react-router4写一遍再来使用react-router6,会使用一个再来使用另一个也不难了,现在先定义最简单的需求:

  1. 定义三个页面,登录页,首页,关于我们页面
  2. 登录页和关于我们是免登录页面。要进入首页必须状态认证通过,否则会跳转到登录页

react-router4:

xml 复制代码
<HashRouter>
  <Switch>
    <Route exact path="/" component={Home} />
    <Route exact path="/aboutUs" component={AboutUs} />
    <Route exact path="/login" component={Login} />
  </Switch>
</HashRouter>

定义好路由后,通过修改地址栏,就能匹配路径渲染相应的组件。所以要做路由拦截,我们可以选择不要直接渲染Route组件,可以自定义AuthRoute组件,组件最终会返回Route组件,在返回Route组件之前我们就可以判断状态不通过时跳转登录页的操作,从而实现路由拦截。

方法一:自定义权限组件

  1. 自定义AuthRoute组件,AuthRoute组件会返回Route组件
  2. AuthRoute返回Route之前进行状态认证,不通过会跳转到登录页
javascript 复制代码
<HashRouter>
  <Switch>
    <AuthRoute exact path="/" component={Home} />
    <Route exact path="/aboutUs" component={AboutUs} />
    <Route exact path="/login" component={Login} />
  </Switch>
</HashRouter>
    
const AuthRoute = (props) => {
    const auth = null

    if (!auth) {
        alert('跳转到登录页')
        return <Redirect to="/login" />
    }

    return (
        <Route {...props} />
    )
}

效果:

方法二:使用Route自带的render属性

xml 复制代码
<HashRouter>
  <Switch>
    <Route exact path="/" render={() => auth ? <Home /> : <Redirect to="/login" /> } />
    <Route exact path="/aboutUs" component={AboutUs} />
    <Route exact path="/login" component={Login} />
  </Switch>
</HashRouter>

这种方法也可以通过判断状态来渲染不同的组件,认证不通过则跳转到登录页。但是需要路由拦截的页面可能会有很多个,而且需要认证的状态也可能不止一个,所以最好还是封装一个自定义组件来实现。

react-router6:

我们先试下同样使用自定义组件来进行路由拦截。

xml 复制代码
<HashRouter>
    <Routes>
      <AuthRoute exact path="/" element={<Home />} />
      <Route exact path="/login" element={<Login />}></Route>
      <Route exact path="/aboutUs" element={<AboutUs />}></Route>
    </Routes>
</HashRouter>

效果:

结果报了错,在Routes组件里只能是Route组件不能是自定义组件,所以稍微改造一下,Routes里只放Route组件,把自定义组件放到element属性上。

自定义权限组件

javascript 复制代码
const AuthRoute = (props) => {
    const auth = null

    if (!auth) {
        return <Navigate to="login" />
    }

    return props.component
}

<Routes>
  <Route exact path="/" element={<AuthRoute component={<Home />} />} />
  ...
</Routes>

效果:

现在路由拦截完成了,不过Route组件可以嵌套,所以可以在优化下Route的写法,外层Route是自定义组件,内层Route是页面组件,这样代码更清晰

javascript 复制代码
AuthRoute组件:
import { Navigate, Outlet } from 'react-router-dom'

const AuthRoute = (props) => {
    const auth = null

    if (!auth) {
        return <Navigate to="login" />
    }

    return <Outlet />
}
export default AuthRoute


<HashRouter>
    <Routes>
      <Route path="/" element={<AuthRoute />}>
        <Route path='/' element={<Home/>}/>
      </Route>
      <Route path="/login" element={<Login />}></Route>
    </Routes>
</HashRouter>

其中Outlet组件就是内层的Route组件,这样就可以把所有需要状态认证的路由都放到内层的Route组件。

总结:react-router4和react-router6的路由拦截大同小异,主要核心是在返回页面组件前先进行一些状态判断,根据判断的结果来进行路由跳转,从而实现路由拦截。

相关推荐
小白学习日记34 分钟前
【复习】HTML常用标签<table>
前端·html
丁总学Java1 小时前
微信小程序-npm支持-如何使用npm包
前端·微信小程序·npm·node.js
yanlele1 小时前
前瞻 - 盘点 ES2025 已经定稿的语法规范
前端·javascript·代码规范
懒羊羊大王呀1 小时前
CSS——属性值计算
前端·css
xgq2 小时前
使用File System Access API 直接读写本地文件
前端·javascript·面试
用户3157476081352 小时前
前端之路-了解原型和原型链
前端
永远不打烊2 小时前
librtmp 原生API做直播推流
前端
北极小狐2 小时前
浏览器事件处理机制:从硬件中断到事件驱动
前端
无咎.lsy2 小时前
vue之vuex的使用及举例
前端·javascript·vue.js
fishmemory7sec2 小时前
Electron 主进程与渲染进程、预加载preload.js
前端·javascript·electron