React Router路由详解
- 一、基础
- 1、安装使用
- 2、React Router 中通用的组件
- 路由组件 BrowserRouter 和 HashRouter
- 路径匹配组件: Route 和 Switch
- 导航组件: Link 和 NavLink
- Redirect 重定向
- withRouter
- 编程式导航 - history 对象
- 路由过渡动画
- 打包部署的路由配置
- 3、路由配置实例
一、基础
1、安装使用
React Router 是一个基于 React 之上的强大路由库,它可以让你向应用中快速地添加视图和数据流,同时保持页面与 URL 间的同步。
安装
npm install react-router-dom
实例
import React, { Component, Fragment } from 'react';
import { Provider } from 'react-redux';
import { BrowserRouter, Route } from 'react-router-dom';
import store from './store';import Header from './common/header';
import Home from './pages/home';
import Detail from './pages/detail';
import Login from './pages/login';class App extends Component {render() {return (<Provider store={store}><Fragment><BrowserRouter><div><Header /><Route path='/' exact component={Home}></Route><Route path='/login' exact component={Login}></Route><Route path='/detail/:id' exact component={Detail}></Route></div></BrowserRouter></Fragment></Provider>)}
}export default App;
2、React Router 中通用的组件
有三种:
- 路由组件(作为根组件):
BrowserRouter
(history
模式) 和HashRouter
(hash
模式) - 路径匹配组件:
Route
和Switch
- 导航组件:
Link
和NavLink
路由组件 BrowserRouter 和 HashRouter
BrowserRouter
(history
模式) 和 HashRouter
(hash
模式)作为路由配置的最外层容器,是两种不同的模式,可根据需要选择。
关于路由组件,如果我们的应用有服务器响应web的请求,建议使用<BrowserRouter>
组件; 如果使用静态文件服务器,建议使用<HashRouter>
组件
history 模式:
class App extends Component {render() {return (<BrowserRouter><Header /><Route path='/' exact component={Home}></Route><Route path='/login' exact component={Login}></Route><Route path='/detail/:id' exact component={Detail}></Route></BrowserRouter>)}
}
hash 模式:
class App extends Component {render() {return (<HashRouter><Header /><Route path='/' exact component={Home}></Route><Route path='/login' exact component={Login}></Route><Route path='/detail/:id' exact component={Detail}></Route></HashRouter>)}
}
路径匹配组件: Route 和 Switch
【Route】用来控制路径对应显示的组件
Route
参数:
path
指定路由跳转路径exact
精确匹配路由component
路由对应的组件
import About from './pages/container/about';
<Route path='/about' exact component={About}></Route>
render
通过写render函数返回具体的dom
<Route path='/about' exact render={() => (<div>about</div>)}></Route>// 或者直接返回 About 组件
<Route path='/about' exact render={() => <About /> }></Route>// 这样写的好处是,不仅可以通过 render 方法传递 props 属性,并且可以传递自定义属性
<Route path='/about' exact render={(props) => {return <About {...props} name={'cedric'} />
}}></Route>// render 方法也可用来进行权限认证
<Route path='/user' exact render={(props) => {// isLogin 从 redux 中拿到, 判断用户是否登录return isLogin ? <User {...props} name={'cedric'} /> : <div>请先登录</div>
}}></Route>
location
将与当前历史记录位置以外的位置相匹配,则此功能在路由过渡动效中非常有用sensitive
是否区分路由大小写strict
是否配置路由后面的 ‘/’
Switch
【Switch 】 渲染与该地址匹配的第一个子节点
<Route>
或者<Redirect>
。
类似于选项卡,只是匹配到第一个路由后,就不再继续匹配
<Switch> <Route path='/home' component={Home}></Route><Route path='/login' component={Login}></Route> <Route path='/detail' exact component={detail}></Route> <Route path='/detail/:id' component={detailId}></Route> <Redirect to="/home" from='/' />
</Switch>当路由为/detail/1时,只会访问匹配组件detail, 所以需要在detail路由上加上exact
注意:如果路由 Route 外部包裹 Switch 时,路由匹配到对应的组件后,就不会继续渲染其他组件了。但是如果外部不包裹 Switch 时,所有路由组件会先渲染一遍,然后选择到匹配的路由进行显示。
导航组件: Link 和 NavLink
Link 和 NavLink 都可以用来指定路由跳转,NavLink 的可选参数更多。
Link
两种配置方式:
- 通过字符串执行跳转路由
<Link to='/login'><span>登录</span></Link>
- 通过对象指定跳转路由
pathname
表示要链接到的路径的字符串search
表示查询参数的字符串形式hash
放入网址的 hash,例如 #a-hashstate
状态持续到 location。通常用于隐式传参(埋点),可以用来统计页面来源
<Link to={{pathname: '/login',search: '?name=cedric',hash: '#someHash',state: { fromWechat: true }}}><span>登录</span>
</Link>点击链接 进入 Login 页面后,就可以在this.props.location.state中看到 fromWechat: true
NavLink
可以看做 一个特殊版本的 Link,当它与当前 URL 匹配时,为其渲染元素添加样式属性。
<Link to='/login' activeClassName="selected"><span>登录</span>
</Link><NavLinkto="/login"activeStyle={{fontWeight: 'bold',color: 'red'}}
><span>登录</span>
</NavLink>
exact
如果为 true,则仅在位置完全匹配时才应用 active 的类/样式strict
当为 true,要考虑位置是否匹配当前的URL时,pathname 尾部的斜线要考虑在内location
接收一个location对象,当url满足这个对象的条件才会跳转isActive
接收一个回调函数,只有当 active 状态变化时才能触发,如果返回false则跳转失败
const oddEvent = (match, location) => {if (!match) {return false}const eventID = parseInt(match.params.eventID)return !isNaN(eventID) && eventID % 2 === 1
}<NavLinkto="/login"isActive={oddEvent}>login</NavLink>
Redirect 重定向
<Redirect>
将导航到一个新的地址
<Redirect>
常用在用户是否登录
<Switch> <Route path='/home' exact component={Home}></Route><Route path='/login' exact component={Login}></Route> <Redirect to="/home" from='/' exact /> // 当访问路由‘/’时,会直接重定向到‘/home’
</Switch>
class Center extends PureComponent {render() {const { loginStatus } = this.props;if (loginStatus) {return (<div>个人中心</div>)} else {return <Redirect to='/login' />}}
}
// 使用对象形式
<Redirectto={{pathname: "/login",search: "?utm=your+face",state: { referrer: currentLocation }}}
/>
withRouter
withRouter 可以将一个非路由组件包裹为路由组件,使这个非路由组件也能访问到当前路由的match, location, history对象
import { withRouter } from 'react-router-dom';class Detail extends Component {render() {··· ···}
}const mapStateToProps = (state) => {return {··· ···}
}const mapDispatchToProps = (dispatch) => {return {··· ···}
}export default connect(mapStateToProps, mapDispatchToProps)(withRouter(Detail));
编程式导航 - history 对象
history 对象的属性和方法:
length
- (number) history 堆栈的条目数
action
- (string) 当前的操作(PUSH, REPLACE, POP
)
location
- (object) 当前的位置。
location 属性:
pathname
- (string) URL 路径search
- (string) URL 中的查询字符串hash
- (string) URL 的哈希片段state
- (object) 提供给例如使用 push(path, state) 操作将 location 放入堆栈时的特定 location 状态。只在浏览器和内存历史中可用。push(path, [state])
- (function) 在 history 堆栈添加一个新条目replace(path, [state])
- (function) 替换在 history 堆栈中的当前条目go(n)
- (function) 将 history 堆栈中的指针调整 ngoBack()
- (function) 等同于 go(-1)goForward()
- (function) 等同于 go(1)block(prompt)
- (function) 阻止跳转。
eg:点击img进入登录页
class Home extends PureComponent {goHome = () => {console.log(this.props);this.props.history.push({pathname: '/login',state: {identityId: 1}})}render() {return (<img className='banner-img' alt='' src="img.png" onClick={this.goHome} />)}
}
只有通过 Route 组件渲染的组件,才能在 this.props 上找到 history 对象。
如果想在路由组件的子组件中使用 history ,需要使用 withRouter 包裹
import React, { PureComponent } from 'react';
import { withRouter } from 'react-router-dom';class 子组件 extends PureComponent {goHome = () => {this.props.history.push('/home')}render() {console.log(this.props)return (<div onClick={this.goHome}>子组件</div>)}
}export default withRouter(子组件);
路由过渡动画
import { TransitionGroup, CSSTransition } from "react-transition-group";<Route render={({location}) => {console.log(location);return (<TransitionGroup><CSSTransitionkey={location.key}classNames='fade'timeout={300}><Switch><Redirect exact from='/' to='/home' /><Route path='/home' exact component={Home}></Route><Route path='/login' exact component={Login}></Route><Route path='/write' exact component={Write}></Route><Route path='/detail/:id' exact component={Detail}></Route><Route render={() => <div>Not Found</div>} /></Switch></CSSTransition></TransitionGroup>)
}}>
</Route>
.fade-enter {opacity: 0;z-index: 1;
}.fade-enter.fade-enter-active {opacity: 1;transition: opacity 300ms ease-in;
}
打包部署的路由配置
如果 react-router 路由 使用了 history 模式(即),那么在 Nginx 配置中必须加上:
location / {··· ···try_files $uri /index.html;··· ···}}
如果 react-router 路由 使用了 hash 模式(即),那么在 Nginx 中不需要上面的配置。
3、路由配置实例
routers/config.js 路由配置文件
import React, { lazy } from 'react';import BasicLayout from '../layouts/BasicLayout/index.jsx';
import BlankLayout from '../layouts/BlankLayout/index.jsx';
import Login from '../pages/Login/index.jsx';
import Welcome from '../pages/Welcome/index.jsx';// 懒加载
const Home = lazy(() =>import(/* webpackChunkName: "Home" */ 'Home/App.jsx')
)
const Mine = lazy(() =>import(/* webpackChunkName: "Mine" */ 'Mine/App.jsx')
)const config = [{path: '/',component: BlankLayout,childRoutes: [{path: '/login', // 路由路径name: '登录页', // 菜单名称 (不设置,则不展示在菜单栏中)component: Login,},{path: '/',component: BasicLayout,childRoutes: [{path: '/welcome',name: '欢迎页',icon: 'logo',component: Welcome,},{path: '/home',name: '首页',component: Home,},{path: '/mine',name: '个人中心',component: Mine,},{ path: '/', exact: true, redirect: '/welcome' },{ path: '*', exact: true, redirect: '/404' },],},],},
];export default config;
main.js 入口文件
import React from 'react';
import ReactDom from 'react-dom';import './styles/main.less';
import AppRouter from './routers/index';ReactDom.render(<AppRouter />, document.getElementById('app'));
routers/index.js 路由渲染文件
import React, { Suspense } from 'react';
import { BrowserRouter as Router, Route, Switch, Redirect } from 'react-router-dom';import Loading from '../components/Loading/index.jsx';
import config from './config';const renderRoutes = (routes) => {if (!Array.isArray(routes)) {return null;}return (<Switch>{routes.map((route, index) => {if (route.redirect) {return (<Redirectkey={route.path || index}exact={route.exact}strict={route.strict}from={route.path}to={route.redirect}/>);}return (<Routekey={route.path || index}path={route.path}exact={route.exact}strict={route.strict}render={() => {const renderChildRoutes = renderRoutes(route.childRoutes);if (route.component) {return (<Suspense fallback={<Loading />}><route.component route={route}>{renderChildRoutes}</route.component></Suspense>);}return renderChildRoutes;}}/>);})}</Switch>);
};const AppRouter = () => <Router>{renderRoutes(config)}</Router>;export default AppRouter;
BlankLayout/index.jsx 空白页布局
import React from 'react';const BlankLayout = ({ children }) => <>{children}</>;export default BlankLayout;
BasicLayout/index.jsx 基本布局
import React from 'react';
import SiderMenu from '../SiderMenu/index.jsx';
import MainHeader from '../MainHeader/index.jsx';
import MainFooter from "../MainFooter/index.jsx";
import styles from './style.less';const BasicLayout = ({ route, children }) => (<div className={styles.container}><MainHeader /><div>{/* 左侧菜单导航 */}<SiderMenu routes={route.childRoutes} />{/* 右侧菜单内容 */}<div className={styles.contant}>{children}</div></div><MainFooter /></div>
);export default BasicLayout;
Welcome/index.jsx contant内容
import React from 'react';import './index.less';const WelcomePage = () => <div className="welcome">welcome</div>;export default WelcomePage;