资讯专栏INFORMATION COLUMN

企业级React项目的个人构建总结

simpleapples / 3044人阅读

摘要:隔壁老王今日行程不同的内容使用这种方法就可以将我们得所有路由写在一起了,可能有人觉得每次都要写引入很麻烦,有没有其他更好用的办法,在讲的时候会说到这里就先跳过。

前言

距离上篇文章已经好长一段时间了,这两个星期公司派驻到京东方这边出差负责入驻项目团队的前端工作。这段时间从零搭建一下前端项目,这次给的时间比较充裕,思考的也比较多。以前也常有搭过前端项目,但是给的时间都比较紧,因此很多问题都忽略掉了。这次正好对以前的进行一次优化,并总结了一些经验分想给大家。如果大家有更好的想法,欢迎留言交流。

温馨提示:

这个项目是以PC端前端项目为视角,移动端前端项目并不完全适用。这点各位小伙们还需要注意一下。

该项目已分不同方向去维护,每个分支与之对应的方向可在CONTRIBUTING.md里查看

项目说明

项目地址: https://github.com/ruichengpi...

该项目可以用我自己写的脚手架工具asuna-cli完成项目构建,我自己写的脚手架工具地址如下:

https://github.com/ruichengpi...

以上是示例项目的目录结构,下面我们将逐一进行分析**
build

这个文件主要放了一些与webpack打包的相关文件。

build.js ---- webpack打包脚本,用于构建生产环境的包

check-versions.js ---- 主要检测当前打包环境的node以及npm的版本是否符合要求

utils.js ---- webpack打包所需要的一些工具库

webpack.base.conf.js ---- webpack的一些基础配置,不同环境的webpack配置都是基于此

webpack.dev.conf.js ---- 开发环境的webpack配置

webpack.prod.conf.js ---- 生产环境的webpack配置

这个项目的webpack配置我是在vue-cli的项目上进行修改的,可以用于React的项目构建。目前只要开发环境和生产环境这两个环境,可能一些公司有多个环境,每个环境下webpack的配置还不同,此时可以根据不同环境建一个文件名格式为“webpack.<环境名>.conf.js”的webpack配置使用。webpack.base.conf.js里面有一些基本配置比如rulesinputoutput的等配置,一般来说每个环境下这些大致都是相同,一些不同之处可以用webpack-merge插件进行合并。一般来说大多数项目来说开发环境和生产环境两个webpack配置足够了。

config

这里存放着不同环境webpack所需要的配置参数。

dev.env.js ---- 向外暴露开发环境下的环境变量NODE_ENV

index.js ---- 存放不同环境的配置参数

prod.env.js ---- 向外暴露生产环境下的环境变量NODE_ENV

如果你需要再加一个环境的话,可以建一个文件名为“<环境名>.env.js”并向外暴露环境变量NODE_ENV,然后在index.js中导入,进行相关参数设置。

mock

这里是用来做接口的mock的,可能很多公司都不太用,我在工作也很少去mock。这里介绍一下自己的接口mock思路,这里我选择mockjs加上json-server的组合。二者具体的使用,大家可以查看其官方文档。

api ---- 存放不同api所对应的数据

index.js ---- json-server的主文件

routes.json ---- 路由的映射

package.json我配置一个script,如下:

 "mock": "json-server mock/index.js  --port 3000 --routes mock/routes.json"

控制台执行“npm run mock“即可。

src api

url.js

export default {
  fetchUserInfo:{
    method:"get",
    url:"/api/user"
  },
  fetchAuthorInfo:{
    method:"get",
    url:"/api/author"
  },
  fetchUserList:{
    method:"get",
    url:"/api/userList"
  }
}

index.js

import _ from "lodash"
import http from "@/utils/http"
import API_URL from "./url";

function mapUrlObjToFuncObj(urlObj){
  const API = {};
  _.keys(urlObj).forEach((key)=>{
    const item = urlObj[key]
    API[key]=function(params){
      return http[item.method](item.url,params)
    }
  });
  return API;
}

function mapUrlObjToStrObj(urlObj){
  const Url = {};
  _.keys(urlObj).forEach((key)=>{
    const item = urlObj[key]
    Url[key]=item.url
  });
  return Url;
}

export const API = mapUrlObjToFuncObj(API_URL);
export const URL = mapUrlObjToStrObj(API_URL);

这里我们用来放置api的接口地址,为了后续的接口维护,我们在使用的过程中不会直接写死接口地址,而是将接口请求封装成一个个方法。通过对接口的统一维护,我们就可以做到在执行修改接口地址、修改请求方法、新增接口等等操作时,就不用在整个项目里到处找了,只要维护好url.js向外暴露的对象即可。使用方法如下:

import {API} from "@/api"
//params为请求参数
API.fetchUserInfo(params).then(response=>{
    //response为返回值
    ...
})
assets

这里我们会放项目的所需要图片资源,这些图片资源一般来说都是做图标的,都比较小。webpack会将其转化成BASE64去使用。如果你不想以这种方式使用,可以在static目录下存放图片资源。

components

这里存放整个项目所用到的公共组件。定一个组件,这里要求是新建一个文件夹,文件夹名为组件名,另外在这个文件夹下新建index.jsx和style.scss文件。例如做一个HelloWorld组件,则应该是如下结构。

HelloWorld

index.jsx

style.scss //存放组件的样式

index.js

import React from "react";
import "./style.scss";
class HelloWorld extends React.PureComponent{
  render(){
    return (
      

Hello World

) } } export default HelloWorld;

style.scss

.u-text{
  color: red;
}
layouts

这里存放着布局文件。关于这个布局文件我是这么去定义它的,我在开发过程中有一些页面他们的某一部分都是相同,早之前可能大家可能会在一个React组件加去实现这个功能,可以这么干,没毛病。但是这个有一个不好点就是你的路由没法做统一的管理,分散在各个组件中,给后续的维护带来很多问题。为了解决这个,我选择利用props.children结合标签嵌套的方式去完成。举个例子:

先定一个layout(本职也是React组件)BasicLayout.jsx

import React from "react";
class BasicLayout extends React.PureComponent{
    render(){
        const {children} = this.props;
        return (
            
隔壁老王今日行程:
{children}
) } } export default BasicLayout;

定义完之后我们可以这么使用:

import React from "react";
import BasicLayout from ""
class Work extends React.PureComponent{
    render(){
        return (
            
                
今天隔壁老王比较累,不工作!
) } } export default BasicLayout;

最后在的dom结构如下:

隔壁老王今日行程:
今天隔壁老王比较累,不工作!

这样我们可以基于BasicLayout做出很多个像下面的页面。

隔壁老王今日行程:
//<不同的内容>

使用这种方法就可以将我们得所有路由写在一起了,可能有人觉得每次都要写引入BasicLayout很麻烦,有没有其他更好用的办法,在讲App.jsx的时候会说到这里就先跳过。

pages

这里的存放的都是页面级组件,跟react-router对应的路由需要一一对应。每个页面都是一个文件夹,文件名就是页面名称,每个页面都要包含如下几个文件:

components ---- 存放当前页独有的一些组件

redux ---- 存放三个文件actions.jsactionTypes.jsreducer.js,这几个文件应该只与这个页面相关

index.jsx ---- 页面的入口文件

style.scss ---- 页面所需要的样式

具体代码可以自行git clone 项目查看,这里就不贴出来了。

scss

这里存放共有的scss文件,比较一些常用的功能类、@mixin、@function等等。

store

这里有四个文件:

actions.js

actionTypes.js

reducer.js

index.js

我们知道每个页面都有自己的actions.jsactionTypes.jsreducer.js,但是这里是全局的,另外index.js会向外暴露store,然后再main.js中引入使用。

import {createStore,combineReducers,applyMiddleware} from "redux";
import thunk from "redux-thunk";
import API from "@/api";
import user from "./reducer";
import author from "@/pages/PageOne/redux/reducer";
const rootReducer = combineReducers({
    user,
    author
  });
const store=createStore(
  rootReducer,
  applyMiddleware(thunk.withExtraArgument({
    API
  }))
)
export default store;

这里有一个小细节,redux-thunk是可以携带一些额外的对象或者方法的,这里,我携带API对象。当我们需要在actions.js里面使用API对象时,就不需要再import导入进来。下面我们做个对比:

修改前

import * as actionTypes from "./actionTypes";
import API from "../api";

export const fecthUserName=(params)=> async (dispatch,getState)=>{
  const response =await API.fetchUserInfo(params);
  const {success,data} = response;
  if(success){
    dispatch({
      type:actionTypes.CHANGE_USER_NAME,
      payload:data
    });
  }
}

修改后

import * as actionTypes from "./actionTypes";

export const fecthUserName=(params)=> async (dispatch,getState,{API})=>{
  const response =await API.fetchUserInfo(params);
  const {success,data} = response;
  if(success){
    dispatch({
      type:actionTypes.CHANGE_USER_NAME,
      payload:data
    });
  }
}
utils

这里会存放一些自己的封装的js工具文件,比如我在项目基于axios封装了一个http.js,简化了axios的操作。

router/index.js

这里以配置化的防止去注册路由,并app.js里面去渲染路由标签。

import Loadable from "react-loadable";
import createHistory from "history/createBrowserHistory";
import BasicLayout from "@/layouts/BasicLayout";
import NavTwoLayout from "@/layouts/NavTwoLayout";
import Loading from "@/components/Loading";
import NotFound from "@/pages/Exception/404";


const Home = Loadable({loader: () => import("@/pages/Home"),loading: Loading});
const Teachers = Loadable({loader: () => import("@/pages/Teachers"),loading: Loading});

export const history = createHistory();

export const routes = [
  {
    path:"/",
    redirect:"/navone/home"
  },
  {
    path:"/navone",
    redirect:"/navone/home",
    children:[{
      path:"/home",
      layout:BasicLayout,
      component:Home
    }]
  },
  {
    path:"/navtwo",
    redirect:"/navtwo/teachers",
    children:[{
      path:"/teachers",
      layout:NavTwoLayout,
      component:Teachers
    }]
  },
  {
    path:"*",
    component:NotFound
  }
]
App.js

这里根据路由配置用来渲染路由标签,先放代码:

import React from "react";
import {Router} from "react-router-dom";
import {Switch, Route ,Redirect} from "react-router";
import {history,routes} from "@/router";



function getRouterByRoutes(routes){
  const renderedRoutesList = [];
  const renderRoutes = (routes,parentPath)=>{
    Array.isArray(routes)&&routes.forEach((route)=>{
      const {path,redirect,children,layout,component} = route;
      if(redirect){
        renderedRoutesList.push()
      }
      if(component){
        renderedRoutesList.push(
          layout?React.createElement(layout,props,React.createElement(component,props))} />:
          )
      }
      if(Array.isArray(children)&&children.length>0){
        renderRoutes(children,path)
      }
    });
  }  
  renderRoutes(routes,"")
  return renderedRoutesList;
 
}
class App extends React.PureComponent{
  render(){
    return (
      
        
          {getRouterByRoutes(routes)}
        
      
    )
  }
}
export default App;

这里我们需要重点讲的是之间在layouts中我们跳过的内容,能不能不每次都用layout组件去包裹代码,答案是可以的。这里我选择中的render属性。

main.js

webpack入口文件,主要一些全局js或者scss的导入,并执行react-dom下的render方法,代码如下:

import React from "react";
import {render} from "react-dom";
import {Provider} from "react-redux";
import store from "@/store";
import App from "@/App";
import "@/scss/reset.scss";
import "@/scss/base.scss";


render(
  
    
  ,
  document.getElementById("app")
)
static

这是一个静态资源目录,一般存放一些第三方工具库。这个目录主要两方面考虑:

有些第三方工具库没有npm包,我们无法用npm install 或者 yarn add方式添加

一些比较大的第三方工具库会影响我们的打包速度,可以把它拿出来通过script的方式引入

其实第三方工具库最好的方式是CDN,但是有些公司就是没有,无奈只能如此。你加入的第三工具库都可在当前服务器下”/static/*“路径下获取到。

templates

这里存放着页面和组件级别构建所需要的模板文件,页面级别构建提供了两种模板PageReducer(集成了reducer)和PageSample(不集成reducer),而组件只提供了一种模板ComSample。页面和组件级别的构建是需要配合asuna-cli才能构建,目前项目已经集成了asuna-cli。package.json写了两个script:npm run newPage(页面构建)和npm run newComponent(组件构建)。开发可根据实际需要选择构建,asuna-cli具体使用可以去https://github.com/ruichengpi...查看。

其他文件

.babelrc ---- babel转换的配置文件

.gitignore ---- git操作所需要忽略的文件

.postcssrc.js ---- postcss的配置文件

index.html ---- 模板index.html,webpack会根据此生成新的index.html,配合html-webpack-plugin使用

package.json ---- 家喻户晓的东西

README.md ---- 项目说明

theme.js ---- ant-design的主题色配置文件,具体使用可以参考ant-design

asuna.config.js ---- asuna-cli的配置文件

yarn.lock ---- 锁定包的版本

结语

这个只是个人搭建企业级React项目的一些总结。当然存在不足的地方,后面在工作过程中如果有一些好的想法也会在这上面进行更新。欢迎大家Star关注!如果你也有好的想法欢迎留言交流,希望这篇拙文能给大家一些启发。

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

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

相关文章

  • 利用TypeScript构建优雅React Native项目

    摘要:很长一段时间就想把引入公司的项目,但总是因为一些原因被搁置。接下来有机会重构之前的项目,借此机会正好可以引入,为了使后期的项目架构更加完善,近期我会梳理的一些知识点和新特性。 很长一段时间就想把ts引入公司的项目,但总是因为一些原因被搁置。接下来有机会重构之前的rn项目,借此机会正好可以引入ts,为了使后期的项目架构更加完善,近期我会梳理rn的一些知识点和新特性。 首先来介绍下Type...

    FingerLiu 评论0 收藏0
  • 7月份前端资源分享

    摘要:更多资源请文章转自月份前端资源分享的作用数组元素随机化排序算法实现学习笔记数组随机排序个变态题解析上个变态题解析下中的数字前端开发笔记本过目不忘正则表达式聊一聊前端存储那些事儿一键分享到各种写给刚入门的前端工程师的前后端交互指南物联网世界的 更多资源请Star:https://github.com/maidishike... 文章转自:https://github.com/jsfr...

    pingan8787 评论0 收藏0
  • 方案设计--如何看待前端框架选型 ?

    摘要:纯前端开发主要是针对静态页面。自主权最大,正常是使用进行辅助开发,上线等。大致原因使用是为了和端的保持同步。四总结对于比较正式的项目,前端技术选型策略一定是产品收益最大化,用户在首位。 对于前端团队,可以实现企业受益最大化要点。 一、技术选型的策略 1、保证产品质量 (1)功能稳健:网页不白屏,不错位,不卡死;操作正常;数据精准。 (2)体验优秀:加载体验,交互体验,视觉体验,无障碍访...

    gnehc 评论0 收藏0
  • 计划在2021年进行响应式开发?但不确定应该选择哪种技术来快速且低成本开发应用程序?一文给你解决问

    摘要:与此同时,因新冠疫情的影响使得用户对移动应用程序的需求激增。调查报告显示年移动应用程序已经产生了亿美元的收入,预计到年将产生亿美元的收入。 引言 计划在2021年进...

    Codeing_ls 评论0 收藏0

发表评论

0条评论

simpleapples

|高级讲师

TA的文章

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