前端路由原理及react-router的常用组件

在react中,通常都是使用单页面应用(SPA),即整个页面只有一个html,然后通过不同的url地址进行组件的匹配和切换。

我们看到的url地址可能会有两种形式,一种是 localhost:3000/home,一种是 localhost:3000/#/home,两种地址的区别在于有无#,有#的是根据hash来进行匹配,即url中的锚点,本质上是通过location.hash来改变href,hash后的内容是不会发送给服务器的,没有#是通过html5的history来进行跳转,两者跳转后都不会进行刷新。

具体来说,通过#来区分路由的实现原理是监听hash的变化,这里监听的方式是使用hashchange 事件,当url地址发生了变化时,通过匹配当前url的地址来进行展示,代码如下所示

<body>
    <div>
       <a href="#/home">首页</a>
       <a href="#/about">关于</a>
    </div>
    <div id="content"></div>

  <script>
    window.addEventListener('hashchange',()=>{
      const content = document.getElementById('content')
      switch(location.hash){
        case '#/home':
          content.innerHTML = '首页'
          break
        case '#/about':
          content.innerHTML = '关于'
          break  
        default:  
          content.innerHTML = ''
      }
    })
  </script>
</body>

通过html5的history来实现跳转无刷新,就需要阻止a标签的默认行为,再通过history的pushState这一方法实现url地址的替换,同样是监听url地址的变化,这里使用popState方法,当url地址发生了变化之后,展示对应的内容。

<body>
  <div>
    <div>
      <a href="/home">首页</a>
      <a href="/about">关于</a>
    </div>
    <div id="content">
   </div>
</body>

<script>
  const content = document.getElementById('content')
  const aEles = document.getElementsByTagName('a')

  for (let el of aEles) {
    el.addEventListener('click', (event) => {
      event.preventDefault()
      const href = el.getAttribute("href")
      history.pushState({}, '', href)
      urlchange()
    })
  }

  window.addEventListener('popstate', () => {
    urlchange()
  })

  function urlchange() {
    switch (location.pathname) {
      case '/home':
        content.innerHTML = '首页'
        break
      case '/about':
       content.innerHTML = '关于'
        break
      default:
        content.innerHTML = ''
    }
  }
</script>

以上的锚点和history分别对应了react-router中的HashRouter和BrowserRouter,在react-router中,想要使用路由组件必须在最外层包裹一层HashRouter或者BrowserRouter,来选择想要使用的路由类型,然后才能使用react-router提供的其它组件,react-route中用于web端的库为react-router-dom,以下所有的组件都是从react-router-dom中导出。

下面来说说react-router的常用组件

1、<Link>和<NavLink>,这两个标签都是由a标签的封装,通过to属性指定跳转的链接地址,<NavLink>比<Link>多的是可以指定选中的样式和类名,如

<Link to="/about">关于</Link>

通过<Link>标签,点击之后就可以跳转到指定的地址,此时即使外层包裹的是HashRouter,也不需要自己加上#,react-router会帮我们在url上加上#


hashRouter.png

2、当使用<Link>定义了跳转的url地址后,此时需要指定匹配跳转该url地址时需要显示的内容,此时使用的是<Route>,通过path属性指定url地址,component属性指定渲染的组件,格式如

 <Route path="/about" component={About} />

加上了之后, /about这个地址显示的就是About这个组件里的内容。<Route>进行的是模糊匹配,一个url路径可能可以匹配多个Route,如果需要严格匹配的话,可以增加一个属性 exact,适合没有二级路由的时候开启。

3、在有很多的Route的情况下,即使在第一个Route匹配到合适的之后,仍然会继续向下匹配,直到最后一个,所以在所有的<Route>外包裹一个<Switch>标签可以让它进行唯一的匹配,匹配到合适的之后就不继续匹配了。

4、当所有的<Route>都无法匹配到url上的地址时,可以定义 <Redirect>组件直接重定向到一个页面,通过to来指定路由地址,这个组件要放置到<Route>的最后面,因为它和<Link>不同,<Link>是点击了之后才会跳转对应的地址,而<Redirect>会直接执行并跳转

<Redirect to="/about"/>

5、通过路由来匹配的组件称为路由组件,<Route path="/about" component={About} />,这里的About就是路由组件,和其它的组件是不一样的,路由组件的props里有一些数据,其中包括三大属性,history、location和match,history可以自定义页面的跳转,location用来获取url地址相关的信息,match可以用作动态路由的匹配。


props里传递的属性.png

但一般的组件是没有这些props属性的,如果一般的组件也需要这样一些属性的话,可以通过一个高阶组件 withRouter。如

class myCom extends PureComponent { }
export default withRouter(myCom)

再来说说路由传参

有时候,我们希望在链接上带一个id值或者两个页面之间跳转的时候传递一些参数,这时候有三种路由传参方式

1、params传参

<NavLink to="/detail/1">商品详情</NavLink>
<Route path="/detail/:id" component={Detail}/>

// 如果是自行定义跳转的地址可以通过 this.props.history.push("/detail/1")

此时的id就是动态的,可以在id这个位置传递任意的数值或者字符串,然后通过props里的match对象中的params获取动态匹配的内容

2、search传参

<NavLink to="/detail?id=1">商品详情</NavLink>
<Route path="/detail" component={Detail}/>

// 如果是自行定义跳转的地址可以通过 this.props.history.push("/detail?id=1")

此时相当于在url上添加一个问号进行拼接,需要把地址拼成一种键值对的形式,通过 props里的location对象中的search属性获取从问号开始的匹配内容,这一种路由的匹配方式需要自行解析字符串

3、state传参

<NavLink to={ pathname: "/detail", state: { id: 1} }>商品详情</NavLink>
<Route path="/detail" component={Detail}/>

// 如果是自行定义跳转的地址可以通过 this.props.history.push("/detail", { id: 1})

这样的传递参数方式通过props里的location对象中的state属性来获取传递的值,这种方式的可以直接以对象的形式传递,并且可传递的数据更多,这些参数不会显示在url上

用一个小的组合案例展示以上内容

import React, { PureComponent } from 'react'
import { NavLink, Route, Switch, withRouter, Redirect } from "react-router-dom"
import Home from "./pages/Home"
import About from "./pages/About"
import Detail from "./pages/Detail"
import Product from './pages/Product'

class App extends PureComponent {
  jumpToProduct(){
    this.props.history.push('/product')
  }
  render() {
    return (
      <div>
        <NavLink exact to="/">首页</NavLink>
        <NavLink to="/about">关于</NavLink>
        <NavLink to="/detail/1">详情</NavLink>
        <button onClick={e=>this.jumpToProduct()}>商品</button>
        <Switch>
          <Route exact path="/" component={Home} />
          <Route path="/about" component={About} />
          <Route path="/detail/:id" component={Detail} />
          <Route path="/product" component={Product}/>
          <Redirect to="/"/>  
        </Switch>
      </div>
    );
  }
}

export default withRouter(App)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容