手动实现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的路由拦截大同小异,主要核心是在返回页面组件前先进行一些状态判断,根据判断的结果来进行路由跳转,从而实现路由拦截。

相关推荐
涔溪43 分钟前
Ecmascript(ES)标准
前端·elasticsearch·ecmascript
榴莲千丞1 小时前
第8章利用CSS制作导航菜单
前端·css
奔跑草-1 小时前
【前端】深入浅出 - TypeScript 的详细讲解
前端·javascript·react.js·typescript
羡与1 小时前
echarts-gl 3D柱状图配置
前端·javascript·echarts
guokanglun1 小时前
CSS样式实现3D效果
前端·css·3d
咔咔库奇1 小时前
ES6进阶知识一
前端·ecmascript·es6
渗透测试老鸟-九青2 小时前
通过投毒Bingbot索引挖掘必应中的存储型XSS
服务器·前端·javascript·安全·web安全·缓存·xss
龙猫蓝图2 小时前
vue el-date-picker 日期选择器禁用失效问题
前端·javascript·vue.js
fakaifa2 小时前
CRMEB Pro版v3.1源码全开源+PC端+Uniapp前端+搭建教程
前端·小程序·uni-app·php·源码下载
夜色呦2 小时前
掌握ECMAScript模块化:构建高效JavaScript应用
前端·javascript·ecmascript