资讯专栏INFORMATION COLUMN

react搭建后台管理(react初窥)

wangjuntytl / 706人阅读

摘要:前言以前一直是用进行的开发于是决定年后弄一弄所以年后这段时间也就一直瞎弄可算是看到成果了本来是想写一个类似仿今日头条那样的项目来入手后来又寻思还不如写个后台管理呢。于是乎自己便着手简单的搭建了一个集中设置的版本。

前言

以前一直是用vue进行的开发, 于是决定年后弄一弄react, 所以年后这段时间也就一直瞎弄react, 可算是看到成果了

本来是想写一个 类似 Vue仿今日头条 那样的项目来入手, 后来又寻思还不如写个后台管理呢。
于是乎便开始捣鼓起来了。

用到react相关的生态链模块:

react

react-dom

react-router-dom: react-router4以后 好像都是用这个东西了

react-transition-group: 用来做动画的

redux: 用来管理全局状态

react-redux: 用来管理全局状态

redux-actions: 用来创建action的,而且生成相关reducers的时候也不要写 switch/case 或 if/else 了,主要是方便。

redux-thunk: redux的中间件, 用来处理我们异步action

antd: 随便找的一个比较常用的react-UI库

跟react相关的主要就是这个几个了
至于webpack 配置,基本跟以前配置vue的基本没多大区别。

文件目录讲解:

build: 用来放置关于webpack的配置

config: 项目配置

src: 源码

static: 静态资源

.babelrc: babel配置

postcss.config.js: css配置

别的目录就不说了,主要介绍一个src下的目录结构

actions: 放redux中action相关的地方

reducers: 放redux中reducer相关的地方

assets: 项目静态资源

components: 常用的公共组件

router: 路由相关的配置

store: redux的配置

styles: 公共样式文件

utils: 工具类的封装

view: 所有页面的主体结构

main.js: 项目入口文件

config.js: 公共属性配置

1. react 的 几种书写方式

React.createClass

import React from "react"
const MyComponent = React.createClass({
   render () {
       return (
           

我是React.createClass生成的组件

) } })

React.createClass会自绑定函数方法(不像React.Component只绑定需要关心的函数)导致不必要的性能开销,增加代码过时的可能性

React.createClass的mixins不够自然、直观;

React.Component

import React from "react"
class MyComponent from React.Component {
    render () {
        return (
            

我是React.Component生成的组件

) } }

需要手动绑定this指向

React.Component形式非常适合高阶组件(Higher Order Components--HOC),它以更直观的形式展示了比mixins更强大的功能,并且HOC是纯净的JavaScript,不用担心他们会被废弃

无状态函数式组件

import React from "react"
 const MyComponent = (props) => (
     

我是无状态函数式组件

) ReactDOM.render(, mountNode)

无状态组件的创建形式使代码的可读性更好,并且减少了大量冗余的代码,精简至只有一个render方法,大大的增强了编写一个组件的便利

组件不会被实例化,整体渲染性能得到提升

组件不能访问this对象

组件无法访问生命周期的方法

无状态组件只能访问输入的props,同样的props会得到同样的渲染结果,不会有副作用

2. 路由拦截

路由拦截这块费了挺长时间,本来是想找个类似vue的beforeRouter这个种钩子函数,发现没有。

然后后面找到history模块,发现有个这东西有个监听路由的方法,最开始就用这它,但是我突然切成hash模式进行开发的时候,发现通过history.push(path, [state])设置state属性的时候出了问题,这东西好像只能给history模式设置state属性,但是我有部分东西是通过设置state属性来进来的,于是便放弃了这个方法寻找新的方法。

后面发现可以通过监听根路径的 componentWillReceiveProps 钩子函数 便可以达到监听的效果。

这钩子函数只要props改变便会触发,因为每次切换路由 locationpathname总是不同的,所有只要切换路径便会触发这个这个钩子函数。这东西容易触发死循环,所以记得做好判断。

class MainComponents extends React.Component {
    componentWillMount () { // 第一次进来触发
        this.dataInit(this.props)
    }
    componentWillReceiveProps(nextProps){ // 以后每次变化props都会触发
        // 如果死循环了 可能是某个属性设置会更新props上属性,所以导致一直循环,这个时候记得做好判断
        this.dataInit(nextProps)
    }
    render () {
        // 404
        if (!isExistPath(allRoutes, pathname)) return 
        
        //当前路径路由信息
        let currRoute = getRoute(allRoutes, pathname)

        // 非白名单验证
        if (!whiteList.some(path => path === pathname)) {

            // 登录验证
            if (!Cookie.get("Auth_Token")) {
                return 
            }
            
            // 获取用户信息
            if (!user) {
                this.getUserInfo(() => {
                    this.setRoutesByRole(this.props.user.roles)
                })
            }
        }
        // 401
        if (user && currRoute) {
            if (!isAuth(currRoute.role, user)) return 
        }

        // 网页title
        document.title = currRoute.name
    }
}
3. 路由集中设置

用过vue的都知道我们一般都是通过new Router({routes}) 来集中管理路由表。但是react-router好像不能这么设置。最新的版本好像连嵌套都不行。
于是乎自己便着手简单的搭建了一个集中设置的版本 。不过后面我看到个插件好像是可以管理的 react-router-config,不过我也还没试过,也不知道可不可行。

// 路由表
const allRoutes = [
  {
    path: "/auth",
    login: true,
    layout: true,
    icon: "user",
    name: "权限管理",
    role: ["admin"],
    component: _import_views("Auth")
  },
  {
    path: "/error",
    login: true,
    layout: true,
    icon: "user",
    name: "ErrorPage",
    redirect: "/error/404",
    children: [
        { path: "/error/404", component: _import_views("Error/NotFound"), name: "404"},
        { path: "/error/401", component: _import_views("Error/NotAuth"), name: "401"}
    ]
  }
  ...
]


// 根目录

    


// MainComponents
class MainComponents extends React.Component {
  render () {
    return (
      
          {renderRouteComponent(allRoutes.filter(route => !route.layout))} //不需要侧边栏等公共部分的路由页面
          
      
    )
  }
}

// ComponentByLayout
const ComponentByLayout = ({history}) => (
  
      
          {renderRouteComponent(allRoutes.filter(route => route.layout))}
      
     
)


// 路由渲染
const RouteComponent = route =>  
const renderRouteComponent = routes => routes.map((route, index) => {
    return route.children ? route.children.map(route => RouteComponent(route)) : RouteComponent(route)
})
4. 根据用户权限动态生成路由

我想根据用户不同的权限生成不同的侧边栏。

{
  path: "/auth",
  login: true,
  layout: true,
  icon: "user",
  name: "权限管理",
  role: ["admin"],
  component: _import_views("Auth")
}

根据这个路由role信息 跟用户的role信息匹配进行显示跟隐藏

这样来筛选出符合这个用户的路由表以及侧边栏(侧边栏根据路由表生成)

但是有个问题,因为我们是需要登录才能得知用户的权限信息,所以我们得那个时候才能确定路由是哪些。

但是那个时候路由已经设置完毕了。vue里面的提供了 router.addRoutes这个方法来供我们动态设置路由,react里面我也没找到关于这个api的,于是我便采取所有的路由都注册一遍,但是这样便产生一个问题。

/auth 为例,我本身是没有访问/auth的权限,所以我侧边栏不会生成 /auth这个列表选项。但是我们在地址栏里面 访问 /auth 是能进入这个页面的的 (最好的办法就是压根就不生成这个路由)。所以这个设置其实是有问题,目前我也没知道怎么动态生成路由的办法,暂时也只是在根目录 做了权限处理

5. 按需加载

按需加载的方法也不少,目前只尝试了第一种,因为我写Vue也是用import实现按需加载的,所以也就没去折腾了。

1. import方法
//asyncComponent.js
import React from "react"
export default loadComponent => (
    class AsyncComponent extends React.Component {
        state = {
            Component: null,
        }
        async componentDidMount() {
            if (this.state.Component !== null) return

            try {
                const {default: Component} = await loadComponent()
                this.setState({ Component })
            }catch (err) {
                console.error("Cannot load component in ");
                throw err
            }
        }

        render() {
            const { Component } = this.state
            return (Component) ?  : null
        }
    }
)


// index.js
import asyncComponent from "./asyncComponent.js"
const _import_ = file => asyncComponent(() => import(file))
_import_("components/Home/index.js")

原理很简单:

import()接受相应的模块然后返回Promise对象

asyncComponent 接收一个函数,且这个函数返回promise对象

在componentDidMount钩子函数通过 async/await 执行接受进来的loadComponent方法,得到import返回的结果,赋值给state.Component,

因为我们import的是一个React组件,所以我们得到的也是React组件,到时候只需要把该组件 render出去就行了

2. Bundle组件 + import(跟第一种感觉差不多) 3. react-loadable 4. bundle-loader 6. request

我这里用到的是axios, 用axios做了个简单的拦截器

import axios from "axios"
import qs from "qs"


axios.defaults.withCredentials = true 

// 发送时
axios.interceptors.request.use(config => {
    // 发起请求,可以进行动画啥的
    return config
}, err => {
    return Promise.reject(err)
})

// 响应时
axios.interceptors.response.use(response => response, err => Promise.resolve(err.response))

// 检查状态码
function checkStatus(res) { 
    // 得到返回结果,结束动画啥的
    if (res.status === 200 || res.status === 304) {
        return res.data
    }
    return {
        code: 0,
        msg: res.data.msg || res.statusText,
        data: res.statusText
    }
    return res
}


// 检查CODE值
function checkCode(res) {
    if (res.code === 0) {
        throw new Error(res.msg)
    }
    
    return res
}

export default {
    get(url, params) {
        if (!url) return
        return axios({
            method: "get",
            url: url,
            params,
            timeout: 30000
        }).then(checkStatus).then(checkCode)
    },
    post(url, data) {
        if (!url) return
        return axios({
            method: "post",
            url: url,
            data: qs.stringify(data),
            timeout: 30000
        }).then(checkStatus).then(checkCode)
    }
}
7. redux

这里主要用了 redux-actions 来创建action的 ,
原生写法

// action
const addTodo = text => ({
    type: "ADD_TODO",
    payload: {
      text,
      completed: false
    }
})

// reducer
const todos = (state = [], action) => {
    switch(action.type) {
        case "ADD_TODO":
            return [...state, action.payload]
        ...
        default:
            return state
    }
}

用了 redux-actions的写法

import { createAction, handleActions } from "redux-actions"

// action
const addTodo = createAction("ADD_TODO")

// reducer
const todos = handleActions({
    ADD_TODO: (state, action) => {
        return [...state, action.payload]
    }
    ...
}, [])

// 用redux-actions简单明了

8. connect

用了redux,这东西基本就不能少了, connect主要是用来 连接 组件redux store的, 就是让组件能获取redux store里面的 方法
connect([mapStateToProps], [mapDispatchToProps], [mergeProps],[options])

一般只用到前两个参数

mapStateToProps(state, ownProps): 获取store里面state指定数据,然后传递到指定组件, ownProps 组件本身的 props

mapDispatchToProps: 这个是获取store里面的action方法, 然后传入指定组件

用法

import toggleTodo from "actions/todo"
const mapStateToProps = state => ({
    active: state.active
})
const mapDispatchToProps = {
    onTodoClick: toggleTodo
}
connect(mapStateToProps, mapDispatchToProps)(Component)
// 在Component组件中, 便能在 props 里面获取到 active 数据, 跟 onTodoClick 这个方法了

connect很多地方基本都要用到
所以也进行了封装

// connect.js
import actions from "src/actions" // 所有action
import {connect} from "react-redux" 
import {bindActionCreators} from "redux"
export default connect(
    state => ({state}), // 偷懒了, 每次把state里面所有的数据都返回了
    dispatch => bindActionCreators(actions, dispatch) //合并所有action,并且传入dispatch, 那样我们在组件里面调用action,就不在需要dispatch了
)

bindActionCreators

然后我们把 connect.js 文件通过 webpack 的alias属性来进行配置

//配置别名映射
alias: {
    "src": resolve("src"),
    "connect": resolve("src/utils/connect")
}

然后我们就可以在文件中如下引用

import React from "react"
import connect from "connect"

@connect // 通过装饰器调用
class Component extends React.Component {
  componentWillMount () {
    const {state, onTodoClick} = this.props
    console.log(state, onTodoClick)
  }
}

为了省事,我把store里面所有的数据 和 action都返回了。

9. cssModules

vue 中 我们一般都是通过设置 style标签的 scoped 属性来做到css模块化
但是在 react 中,我采用的 cssModules 来做css模块化

通过webpack设置 css-loadermodules来开启css的模块化

{
    loader: "css-loader",
    options: {
      modules: true, //是否开启
      localIdentName: "[name]__[local]___[hash:base64:5]"  // 转化出来的class名字结构
    }
},

引入css, 并通过对象的赋值方式添加className

import styles from "./styles.css"

export default () => (
  
) //styles.css .a { color: #ff4747; }

或者可以通过 react-css-modules 来更方便的控制class类名

import styles from "./styles.css"
import CSSModules from "react-css-modules"

class Component extends React.Component {
  render () {
    return (
      
) } } export default CSSModules(Component, styles, { allowMultiple: true //允许多个class一起使用 }) //styles.css .a { color: #ff4747; } .b { background: #f00; }

这样我们就可以通过字符串的方式传入 class类名. 注意: 我们添加时 不再使用 className 了, 而是使用 styleName

10. 双向绑定的实现
class Bingding extends React.Component {
  state = {
    value: ""
  }
  handleInput = value => {
    this.setState({
      value
    })
  }
  render () {
    return (
       {this.handleInput(e.target.value)}}/>
      
{this.state.value}
) } }

就是通过 onChange 事件 来触发 this.setState 重新渲染 render 方法

还有一些知识点
包括 动画生命周期 等等
就不过多介绍了。这些项目中基本多多少少都参和了一点。
开发中遇到的问题挺多的,最主要是react-router配置的问题,怎么配置都感觉不太好。
也同时希望有人推荐几个全面的尤其是最新版本的react开源项目。

项目启动步骤

npm/yarn run dll (DllPlugin打包,只需打包一次就够了)

npm/yarn run dev (开发模式)

npm/yarn run build (生产模式)

小结

国内比较火的两个框架,也勉强算是都接触了下,vue我是一直在用的,react算是年后刚接触的。
从我目前来看,vuereact开发起来确实要方便很多(可能用的比较多吧)。
因为vue很多常用的都是内置的。而react基本都要自己去寻找对应的模块。本身就只提供UI, 其他基本得自力更生。
主要是你经常一找能找着多个模块,你就不知道用哪个,还得一个个试水。当然,react的社区强大,这么都不是什么大问题。

在线观看地址

博客地址

github

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/93427.html

相关文章

  • 初窥基于 react-art 库的 React Native SVG

    摘要:语法更近似于移动端。当参数为两个时,等同于,绘制光滑二次贝塞尔曲线。有些精通的同学这时候可能就要问我了,不对啊,二次贝塞尔曲线和光滑三次贝塞尔曲线的参数都是个,你这里没有光滑三次啊因为开发的同学留坑没写了呀微笑。和则是用于指定旋转的原点。 技术背景 在移动应用的开发过程中,绘制基本的二维图形或动画是必不可少的。然而,考虑到Android和iOS均有一套各自的API方案,因此采用一种更普...

    xiaowugui666 评论0 收藏0
  • 好程序员React精品项目全集:商城管理后台(视频+源码+笔记)

    摘要:今天给大家带来了好程序员实战项目商城管理后台。配合项目学习会让你更快掌握它的使用方法下面就来看看好程序员这套实战项目课程介绍好程序员项目本项目是一个使用开发的商城系统的管理后台,里面登录判断,接口调用,数据展示和编辑,文件上传等后台功能。 众所周知,项目经验对于一个程序员变得越来越重要。在面...

    李世赞 评论0 收藏0
  • React项目实践系列一

    摘要:在此我们选用用友的公共静态资源库。因此打算建立远程的其实还有个关键是我使用用友配的电脑开发,在本地部署的话电脑配置。。。不过此步骤我们也可以省略了,用友的大前端技术团队提供了平台。 数据分析平台-实践系列一 项目创建于2018年1月底,到现在已经接近半年,在此写下半年来项目的实践过程以及自己对前端的学习与体悟。 技术选型 框架: React 路由: React-Router 4 状态管...

    DC_er 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<