【共享單車】—— React後臺管理系統開發手記:Router 4.0路由實戰演練

前言:如下內容基於React全家桶+AntD實戰課程的學習實踐過程記錄。最終成果github地址:https://github.com/66Web/react-antd-manager,歡迎star。javascript


 1、React Router 4.0核心概念css

     4.0版本中已不須要路由配置,一切皆組件html

  • react-router:基礎路由包
  1. 提供了一些router的核心api,包括Router,Route,Switch等
  • react-router-dom:基於瀏覽器的路由(包含react-router)
  1. 提供了BrowerRouter,HashRouter,Route,Link,NavLink
  2. 安裝:
    npm install react-router-dom --save
    //或
    yarn add react-router-dom  

     react-router-dom核心用法java

  • HashRouter和BrowserRouter區別
  1. HashRouter
    http://localhost:3000/#/admin/buttons  
  2. BrowserRouter
    http://localhost:3000/admin/buttons
    
  • Route用法
  1. path屬性:路由地址
  2. component屬性:路由對應要跳轉到的組件
  3. exact屬性:表明精準匹配,必須path徹底匹配才能夠加載對應組件
  4. render方法:渲染一個組件,其中包裹一個子路由
    <Route path="/admin/ui/buttons" component={Button} /> 
    
    <Route path="/admin" render={() => 
    
         <Admin>
              <Route path="/admin/home"  component={Home} />
         </Admin>
    }/>
    
  • 導航
  1. NavLink:菜單裏的導航;Link:超連接導航
  2. to:能夠傳一個字符串,也能夠傳一個location對象
    import {Link} from 'react-router-dom';
    
    const Header=()=>{
        <header>
            <nav>
                <li><Link to='/'>Home</Link></li>
                <li><Link to='/about'>About</Link></li>
                <li><Link to='/three'>Three</Link></li>
            </nav>
        </header>
    }    
    
    <Link to={{pathname:'/three/7'}}>Three #7</Link> 
  3. 定義:<Route path="/three/:number" /> react

  4. 取值:this.props.match.params.numbergit

  5. 一個基本的location對象: {  pathname: '/',  search:'',  hash:'',  key:'abc123',  state:{} }github

  • Switch 選擇web

  1. 選擇符合要求的Route,自上至下找到一個能夠匹配的內容則不繼續加載其餘 npm

  2. <Switch>是惟一的由於它僅僅只會渲染一個路徑
    <Switch>
        <Route path='/admin/ui/buttons' component={Buttons}/>
        <Route path='/admin/ui/models' component={Models}/>
        <Route path='/admin/ui/loading' component={Loading}/>
    </Switch>
    
  • Redirect 重定向api

    <Redirect to="/admin/home">

2、React Router4.0 Demo介紹

嵌套路由 動態路由

     混合組件化 -- 將Route和Link放到同一個頁面 

  • HashRouter將Link和Router進行包裹,其內部必須只能有一個子節點
  • 渲染對應component中的組件
  1. 經過Link組件的to屬性設置路由地址
  2. 經過Route組件的path屬性匹配路由地址
  • 注意:Route添加exact屬性能夠作到【路徑精準匹配】,不然/about既能夠匹配 也能夠匹配 /about 等路由地址

     Demo代碼

  • Home.js:顯示整個頁面內容
    import React from 'react';
    import {HashRouter,Route,Link,Switch} from 'react-router-dom'
    import Main from './Main';
    import About from './About';
    import Topics from './Topics';
    
    class Home extends React.Component{
      render(){
          return(
              <HashRouter>
                  <div>
                      <ul>
                          <li>
                              <Link to="/">Home</Link>
                          </li>
                          <li>
                              <Link to="/about">About</Link>
                          </li>
                          <li>
                              <Link to="/topics">Topics</Link>
                          </li>
                      </ul>
                      <hr></hr>
                      <Switch>
                          <Route path="/" exact={true} component={Main}></Route>
                          <Route path="/about" component={About}></Route>
                          <Route path="/topics" component={Topics}></Route>
                      </Switch>
                  </div>   
              </HashRouter>
          );
      }
    }
    export default Home; 

    Main.js :路由組件 (其它組件同Main.js)

    import React,{Component} from 'react';
    class Main extends Component{
      render(){
          return(
              <div>
                  this is Main.
              </div>
          );
      }
    }
    export default Main;

    index.js:將Home組件掛載到根節點上

    import Home from './pages/router_demo/Home'
    
    ReactDOM.render(<Home />, document.getElementById('root'));

     配置化 -- 將Route路由提取到一個單獨的JS文件中 

  • Router、Link和Route
  1. Router是節點 ——下面只能包含一個盒子標籤,如這裏的div,內容爲路由及路由組件
    <Router>
         <div>
            //otherCoding
         </div>
    </Router>  
  2. Link是連接 —— 在html界面中會解析成a標籤,to表明連接地址(一個相對路徑 ) 
    <Router>
         <div>
            <ul>
               <li><link to='/'>首頁</Link></li>
               <li><link to='/other'>其餘頁</Link></li>
            </ul>
         </div>
    </Router>
  3. Route是路由組件 —— path表明路徑,component表明路徑所對應的界面

    <Router>
          <div>
              <ul>
                  <li><Link to="/home">首頁</Link></li>
                  <li><Link to="/other">其餘頁</Link></li>
              </ul>
              <Route path="/home" component={Home}/>
              <Route path="/other" component={Other}/>
          </div>
    </Router>
    
  • 配置化實現路由功能

  1. router.js:根據Home.js中的Link導航配置找到對應的組件
    import React from 'react'
    import {HashRouter as Router,Route} from 'react-router-dom' 
    import Main from './Main';
    import About from './About';
    import Topics from './Topics';
    import Home from './Home'
    
    export default class IRouter extends React.Component{
    
        render(){
            return (
                <Router>
                    <Home>
                        {/*根據導航配置找到對應的組件*/}
                        <Route path="/" exact={true} component={Main}></Route>
                        <Route path="/about" component={About}></Route>
                        <Route path="/topics" component={Topics}></Route>
                    </Home>
                </Router>
            )
        }
    }  
  2. Home.js:①頁面中配置Link導航  ②在router.js中經過Route找到組件  ③呈如今{this.props.children}處

    import React from 'react';
    import {Link} from 'react-router-dom'
    
    class Home extends React.Component{
      render(){
          return( 
                  <div>
                      {/*頁面導航的配置*/}
                      <ul>
                          <li>
                              <Link to="/">Home</Link>
                          </li>
                          <li>
                              <Link to="/about">About</Link>
                          </li>
                          <li>
                              <Link to="/topics">Topics</Link>
                          </li>
                      </ul>
                      <hr />
                      {/*找到路由匹配的組件後呈現的位置*/}
                      {this.props.children}  
                  </div> 
          );
      }
    }
    export default Home;   
  3. index.js:將router組件掛載到根節點上
    import Router from './pages/router_demo/router'
    
    ReactDOM.render(<Router />, document.getElementById('root'));
    
  • 嵌套路由 

  1. router.js:使用render方法渲染Main主頁面極其嵌套路由;注意箭頭函數後不能加{},不然只執行,並不會返回(return)

    import User from './User';
    
    {/*改變Main路由,添加render方法*/}
     <Route path="/main" render={() => 
            <Main>
                  <Route path="/main/user" component={User}></Route>
            </Main>
     }></Route>  
  2. Main.js:使用Link配置子路由導航,同時設置子路由顯示的位置{this.props.children}

    import {Link} from 'react-router-dom'
    
    {/*添加Link配置子路由導航*/}
    <Link to='/main/user'>嵌套路由</Link>
    <hr />
    {this.props.children}  
  3. 坑:由於前面main組件的路由添加了exact={true}精確匹配,會致使/user不被匹配,沒法獲取子路由
  4. 解決:去掉exact屬性,使用/main匹配main組件路由,/main/user匹配user子組件路由
  • 獲取動態路由的值
  1. Main.js:設置跳轉的路由連接
    <Link to='/main/test-id'>動態路由1</Link>
    <br />
    <Link to='/main/456'>動態路由2</Link>
  2. router.js:添加動態路由,即path:"/main/:value"用冒號定義的路由內容

    import Info from './Info';
    
    {/*用冒號定義路由內容*/}
    <Route path="/main" render={() => 
         <Main>
                <Route path="/main/:value" component={Info}></Route>
         </Main>
    }></Route>
  3. Info.js:獲取定義的動態路由內容信息,經過{this.props.match.params.路由的名稱}
    import React,{Component} from 'react';
    class Info extends Component{
      render(){
          return(
              <div>
                  這裏是測試動態路由功能
                  動態路由的值時:{this.props.match.params.value}
              </div>
          );
      }
    }
    export default Info;
    
  • 匹配路由的404頁面

  1. router.js:添加沒有path屬性的Route組件放置Switch組件內部的最後位置,做爲默認路由

    import {HashRouter as Router,Route,Switch} from 'react-router-dom'
    import NoMatch from './NoMatch'
    
    <Switch>
           {/*Switch組件從上到下,只匹配一個路由*/}
           <Route path="/main" render={() => 
                  <Main>
                       <Route path="/main/:value" component={Info}></Route>
                  </Main>
           }></Route>
           <Route path="/about" component={About}></Route>
           <Route path="/topics" component={Topics}></Route>
           <Route component={NoMatch}></Route>
    </Switch>
  2. NoMatch.js

    import React from 'react';
    class NoMatch extends React.Component{
        render(){
            return(
                <div>
                    404 Not Found
                </div>
            );
        }
    }
    export default NoMatch;
    

3、項目路由實戰開發

  • index.js:掛載Router組件到根節點
    // import Admin from './admin'
    import Router from './router'
    
    ReactDOM.render(<Router />, document.getElementById('root'));
    
  • App.js:項目路由入口文件

  1. 用戶訪問項目時輸入url對應的做爲整個文件輸出,一共有三種狀況:登陸、詳情頁、首頁
  2. App.js內部什麼都沒有,只有{this.props.children},主要用來存放子組件(即上述三個頁面),不然沒有顯示位置
    /**路由入口組件 */
    
    import React, { Component } from 'react';
    import './App.css';
    
    class App extends Component {
      render() {
        return (
            <div>
               {this.props.children}
            </div>
        );
      }
    }
    
    export default App;
    
  3. 總結:想利用Route顯示組件信息,則必須有調動{this.props.children}顯示其頁面的組件

  • src->router.js:有三個頁面就須要有三個路由

  1. path爲/login渲染登陸組件

  2. path爲/admin渲染首頁:①用render函數返回子路由Admin組件 ②定義嵌套路由/admin/ui/buttons顯示組件按鈕 ③無path顯示404頁面 ④外層用Switch包裹保證只顯示其中第一個匹配的Route
  3. path爲/order/detail渲染詳情頁

    import React from 'react'
    import {HashRouter, Route, Switch} from 'react-router-dom'
    import App from './App'
    import Login from './pages/Login'
    import NoMatch from './pages/NoMatch'
    import Admin from './admin'
    import Buttons from './pages/ui/buttons'
    
    export default class IRouter extends React.Component{
    
        render() {
            return (
               <HashRouter>
                   <App>
                      <Route path="/login" component={Login}></Route>
                      <Route path="/admin" render={() => 
                          <Admin>
                              <Switch>
                                <Route path="/admin/ui/buttons" component={Buttons}></Route>
                                <Route component={NoMatch}></Route>
                              </Switch>
                          </Admin>
                      }></Route>
                     <Route path="/order/detail" component={Login}></Route>
                   </App>
               </HashRouter>
            )
        }
    }
  • src->admin.js:content部分使用{this.props.children}顯示在router.js中Route獲得的頁面
    <Row className="content">
         {/* <Home /> */}
         {this.props.children}   
    </Row> 
  • NevLeft->index.js:經過左側導航欄進行跳轉
  1. 有Route就必定要有Link指定路由地址
  2. 利用react-router-dom的NavLink設置路由地址,NavLink組件顯示的內容爲{item.title},to跳轉的地址爲{item.key}
    import {NavLink} from 'react-router-dom'
    
    return <Menu.Item title={item.title} key={item.key}>
                     <NavLink to={item.key}>{item.title}</NavLink>
           </Menu.Item>
    
  • pages->ui->button.js:匹配的是/admin/ui/buttons則將Button組件渲染到Admin組件的content中

    import React,{Component} from 'react';
    class Buttons extends Component{
      render(){
          return(
              <div>
                  this is Buttons.
              </div>
          );
      }
    }
    export default Buttons;
  • pages->NoMatch->index.js:沒有對應的路由匹配則將404頁面渲染到Admin組件的content中

    import React,{Component} from 'react';
    class NoMatch extends Component{
      render(){
          return(
              <div style={{textAlign: 'center', fontSize:'24'}}>
                  404 No Found !!!
              </div>
          );
      }
    }
    export default NoMatch;

注:項目來自慕課網

相關文章
相關標籤/搜索