react路由守卫(路由拦截)

react路由守卫(路由拦截)react不同于vue,通过在路由里设置meta元字符实现路由拦截。在使用Vue,框架提供了路由守卫功能,用来在进入某个路有前进行一些校验工作,如果校验失败,就跳转到404或者登陆页面,比如Vue中的beforeEnter函数:…router.beforeEach(async(to,from,next)=>{consttoPath=to.path;constfromPath=from.path;})…react实现路由拦截的基

大家好,又见面了,我是你们的朋友全栈君。

欢迎大家进群,一起探讨学习

欢迎大家进群,一起讨论学习

微信公众号,每天给大家提供技术干货

在这里插入图片描述

博主技术笔记

博主网站地址1

博主网站地址2

博主开源微服架构前后端分离技术博客项目源码地址,欢迎各位star

react不同于vue,通过在路由里设置meta元字符实现路由拦截。在使用 Vue ,框架提供了路由守卫功能,用来在进入某个路有前进行一些校验工作,如果校验失败,就跳转到 404 或者登陆页面,比如 Vue 中的 beforeEnter 函数:

...
router.beforeEach(async(to, from, next) => { 
   
    const toPath = to.path;
    const fromPath = from.path;
})
...

react实现路由拦截的基本思路还是利用Route 的render函数。通过判断拦截条件来实现不同的组件的跳转,从而实现拦截。在之前的版本中,React Router 也提供了类似的 onEnter 钩子,但在 React Router 4.0 版本中,取消了这个方法。React Router 4.0 以后采用了声明式的组件,路由即组件,要实现路由守卫功能,就得我们自己去写了。
如果不使用路由守卫,Router 组件是这样子的:

import * as React from 'react';
import { 
    HashRouter,Switch,Route,Redirect } from 'react-router-dom';
 
import Index from "./page/index";
import Home from "./page/home";
import ErrorPage from "./page/error";
import Login from "./page/login";
 
export const Router = () => (
    <HashRouter>
        <Switch>
            <Route path="/" exact component={ 
   Index}/>
            <Route path="/login" exact component={ 
   Login}/>
            <Route path="/home" exact component={ 
   Home}/>
            <Route path="/404" exact component={ 
   ErrorPage}/>
            <Redirect to="/404" />
        </Switch>
    </HashRouter>
);

上面的 Router 组件,包含了三个页面:

登陆
主页
404 页面
以及四个路由:
根路由
登陆路由
主页路由
404 路由
其中,根路由和 /home 路由,都定向到了主页路由。
以上是一个基本的路由定义,可以在登陆/主页和 404 页面之间来回跳转,但也有一些问题:
非登陆状态下,可以直接跳转到主页
登陆状态下,也可以输入 /login 路由跳转到登录页
现在,我们想完成这样的功能:
非登陆状态下,无法直接跳转到主页,如果在非登陆状态下进行主页跳转,需要重定向至登陆路由
登陆状态下,无法跳转至登录页,如果在登陆状态下进行登陆页跳转,需要重定向至主页路由
要完成这个功能,有两种方案:
在每个组件中,根据 props 上的 history 对象来进行跳转
进行全局的路由守卫处理

首先在跟目录src下创建一个routerMap.js文件,代码如下:
将 auth 设置为 true,表示该路由需要权限校验。

/** * 定义路由组件,将 auth 设置为 true,表示该路由需要权限校验 */

import Admin from "./pages/Admin";
import Login from "./pages/Login";
import Error from "./pages/Error";

export const routerMap = [
    { 
   path: "/", name: "admin", component: Admin, auth: true},
    { 
   path: "/login", name: "Login", component: Login},
    { 
   path: "/error", name: "error", component: Error},
];

所有的路由跳转,都交给 FrontendAuth 高阶组件代理完成。下面是 FrontendAuth.js 组件的实现:

/** * 路由守卫校验 */
import React, { 
   Component} from "react";
import { 
   Route, Redirect} from "react-router-dom";

class FrontendAuth extends Component { 
   
    // eslint-disable-next-line no-useless-constructor
    constructor(props) { 
   
        super(props);
    }

    render() { 
   
        const { 
   routerConfig, location} = this.props;
        const { 
   pathname} = location;
        const isLogin = localStorage.getItem("user");
        console.log(pathname, isLogin);
        console.log(location);
        // 如果该路由不用进行权限校验,登录状态下登陆页除外
        // 因为登陆后,无法跳转到登陆页
        // 这部分代码,是为了在非登陆状态下,访问不需要权限校验的路由
        const targetRouterConfig = routerConfig.find(
            (item) => item.path === pathname
        );
        console.log(targetRouterConfig);
        if (targetRouterConfig && !targetRouterConfig.auth && !isLogin) { 
   
            const { 
   component} = targetRouterConfig;
            return <Route exact path={ 
   pathname} component={ 
   component}/>;
        }
        if (isLogin) { 
   
            // 如果是登陆状态,想要跳转到登陆,重定向到主页
            if (pathname === "/login") { 
   
                return <Redirect to="/"/>;
            } else { 
   
                // 如果路由合法,就跳转到相应的路由
                if (targetRouterConfig) { 
   
                    return (
                        <Route path={ 
   pathname} component={ 
   targetRouterConfig.component}/>
                    );
                } else { 
   
                    // 如果路由不合法,重定向到 404 页面
                    return <Redirect to="/error"/>;
                }
            }
        } else { 
   
            // 非登陆状态下,当路由合法时且需要权限校验时,跳转到登陆页面,要求登陆
            if (targetRouterConfig && targetRouterConfig.auth) { 
   
                return <Redirect to="/login"/>;
            } else { 
   
                // 非登陆状态下,路由不合法时,重定向至 404
                return <Redirect to="/error"/>;
            }
        }
    }
}

export default FrontendAuth;

然后,定义 Router 组件,在App.js中,该组件是经过高阶组件包装后的结果:

import './App.less';
import React, { 
   Fragment} from "react";
import { 
   Switch} from 'react-router-dom'
import FrontendAuth from "./FrontendAuth";
import { 
   routerMap} from "./routerMap";

function App() { 
   
    return (
        <Fragment>
            { 
   /*只匹配一个,匹配成功就不往下匹配,效率高*/}
            <Switch>
                <FrontendAuth routerConfig={ 
   routerMap}/>
            </Switch>
        </Fragment>
    );
}

export default App;

测试

在这里插入图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/142935.html原文链接:https://javaforall.net

(0)
上一篇 2022年5月22日 下午5:00
下一篇 2022年5月22日 下午5:00


相关推荐

  • API 与 SDK 之间的区别

    API 与 SDK 之间的区别很多人在软件开发中经常会分不清 SDK 与 API 今天就来浅谈一下两者之间的区别 直白地说 SDK 包含了 API 是一套完整的 能完成更多功能的工具包 无论你想获取什么样的信息 SDK 里总能找到实现的办法 用通俗的话来举个例子 有一杯密封饮料 它的名字叫做 SDK 饮料上插着吸管 吸管的名字叫 API 把你叫做 XX 系统 如果你想喝到 SDK 里的饮料 让系统拥有 SDK 中的功能 你必须通过 API 这根吸管来实现 通过 API 连接你的系统和

    2026年3月17日
    2
  • nc隧道技术

    nc隧道技术使用场景 客户端 嵌入式 Linux 网关服务器 阿里云服务器 1 正向 shell 客户端连接服务器 获取服务器的 shell 服务器 root VM 0 14 centos mkfifo tmp fifo root VM 0 14 centos cat tmp fifo bin bash i2 gt amp 1 nc l10000 gt tmp

    2026年3月17日
    1
  • FormData上传文件以及其他参数

    FormData上传文件以及其他参数FormData 上传文件参数

    2026年3月19日
    2
  • 三维点在平面上投影

    三维点在平面上投影三维空间平面的一般方程为 Ax By Cz D 0Ax By Cz D 0Ax By Cz D 0 1 假定不在平面上的三维空间点坐标为 x o y o z o 其在平面上的投影点坐标为 x p y p z p 因为投影点到当前点与平面垂直 根据垂直约束条件 易知 y p 与 z p 满足如下条件 yp BA xp xo yoy p frac B A x p x o y oyp AB xp xo yo 2

    2026年3月17日
    2
  • pycharm ssh_云终端机安装方法

    pycharm ssh_云终端机安装方法1、主题  如何使用Pycahrm内置终端以及远程SSH工具。  2、准备工作  Pycharm版本为3.0或更高  连接SSH服务器  3、使用SSH客户端  4、开启连接  选择Tools|StartSSHSession…的主菜单命令,单击Editcredentials:  5、提供连接信息  在Sessio

    2022年8月28日
    5
  • SpringMvc—拦截器「建议收藏」

    SpringMvc—拦截器「建议收藏」SpringMvc拦截器作用拦截器是用来拦截经过dispatcherServlet【请求控制器】的请求。它用来拦截控制器方法的执行。拦截器通过实现接口HandlerInterceptor并在S

    2022年7月1日
    28

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号