成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

webpack4 中的最新 React全家桶實(shí)戰(zhàn)使用配置指南!

Towers / 608人閱讀

摘要:安裝配置加載器配置配置文件配置支持自定義的預(yù)設(shè)或插件只有配置了這兩個(gè)才能讓生效,多帶帶的安裝是無意義的。

想閱讀更多優(yōu)質(zhì)文章請猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你!

最新React全家桶實(shí)戰(zhàn)使用配置指南

這篇文檔 是呂小明老師結(jié)合以往的項(xiàng)目經(jīng)驗(yàn) 加上自己本身對react webpack redux理解寫下的總結(jié)文檔,總共耗時(shí)一周總結(jié)下來的,希望能對讀者能夠有收獲, 我是在這基礎(chǔ)多些加工!

目錄

1.版本說明

2.目錄結(jié)構(gòu)

3.初始化項(xiàng)目

4.webpack

5.react

6.配置loader(sass,jsx))

7.引入babel

8.使用HtmlWebpackPlugin

9.redux

10.使用webpack-dev-server

11.多入口頁面配置

12.如何理解entry point(bundle),chunk,module

13.多入口頁面html配置

14.模塊熱替換(Hot Module Replacement)

15.使用ESLint

16.使用react-router

17.使用redux-thunk

18.使用axios和async/await

19.Code Splitting

20.使用CommonsChunkPlugin

版本說明

由于構(gòu)建相關(guān)例如webpack,babel等更新的較快,所以本教程以下面各種模塊的版本號為主,切勿輕易修改或更新版本。

"dependencies": {
    "babel-core": "^6.26.3",
    "babel-eslint": "^8.2.3",
    "babel-loader": "^7.1.4",
    "babel-plugin-transform-async-to-generator": "^6.24.1",
    "babel-plugin-transform-runtime": "^6.23.0",
    "babel-preset-es2015": "^6.24.1",
    "babel-preset-react": "^6.24.1",
    "babel-preset-stage-0": "^6.24.1",
    "babel-preset-stage-3": "^6.24.1",
    "css-loader": "^0.28.11",
    "eslint": "^4.19.1",
    "eslint-loader": "^2.0.0",
    "eslint-plugin-react": "^7.9.1",
    "file-loader": "^1.1.11",
    "history": "^4.7.2",
    "html-webpack-plugin": "^3.2.0",
    "react": "^16.4.0",
    "react-dom": "^16.4.0",
    "react-hot-loader": "^4.0.0",
    "react-redux": "^5.0.7",
    "react-router-dom": "^4.3.1",
    "react-router-redux": "^5.0.0-alpha.9",
    "redux": "^4.0.0",
    "sass-loader": "^7.0.3",
    "style-loader": "^0.21.0",
    "url-loader": "^1.0.1",
    "webpack": "^4.12.0",
    "webpack-cli": "^3.0.3",
    "webpack-dev-server": "^3.1.1"
}
目錄結(jié)構(gòu)

開發(fā)和發(fā)布版本的配置文件是分開的,多入口頁面的目錄結(jié)構(gòu)。

react-family/
    |
    |──dist/                                    * 發(fā)布版本構(gòu)建輸出路徑
    |
    |──dev/                                     * 調(diào)試版本構(gòu)建輸出路徑
    |
    |──src/                                     * 工具函數(shù)
    |     |
    |     |—— component/                        * 各頁面公用組件
    |     |
    |     |—— page/                             * 頁面代碼
    |     |      |—— index/                     * 頁面代碼
    |     |      |        |—— Main/             * 組件代碼
    |     |      |        |       |—— Main.jsx  * 組件jsx
    |     |      |        |       |—— Main.scss * 組件css
    |     |      |
    |     |      |—— detail/                    * 頁面代碼
    |     |
    |     |—— static/                           * 靜態(tài)文件js,css
    |
    |
    |──webpack.config.build.js                  * 發(fā)布版本使用的webpack配置文件
    |──webpack.config.dev.js                    * 調(diào)試版本使用的webpack配置文件
    |──.eslint                                  * eslint配置文件
    |__.babelrc                                 * babel配置文件
    
    
    
初始化項(xiàng)目

1.創(chuàng)建文件夾

mkdir react-family-bucket

2.初始化npm

cd react-family-bucket
npm init

如果有特殊需要,可以填入自己的配置,一路回車下來,會生成一個(gè)package.json,里面是你項(xiàng)目的基本信息,后面的npm依賴安裝也會配置在這里。

webpack

1.安裝webpack

npm install [email protected] --save
or
npm install [email protected] --g

--save 是將當(dāng)前webpack安裝到react-family-bucket下的/node_modules。

--g 是將當(dāng)前webpack安裝到全局下面,可以在node的安裝目錄下找到全局的/node_modules。

2.配置webopack配置文件

touch webpack.config.dev.js

3.新建一個(gè)app.js

touch app.js

寫入基本的webpack配置,可以參考這里:

const path = require("path");
const srcRoot = "./src";
module.exports = {

    // 輸入配置
    entry: [
      "./app.js"
    ],,

    // 輸出配置
    output: {
        path: path.resolve(__dirname, "./dev"),

        filename: "bundle.min.js"
    },
};

3.執(zhí)行webpack命令

如果是全局安裝:

webpack --config webpack.config.dev.js

如果是當(dāng)前目錄安裝:

./node_modules/.bin/webpack --config webpack.config.dev.js

為了方便我們使用,可以在package.json中 scripts 添加執(zhí)行命令:

"scripts": {
  "dev": "./node_modules/.bin/webpack --config webpack.config.dev.js",
},

執(zhí)行npm run dev命令之后,會發(fā)現(xiàn)需要安裝webpack-cli,(webpack4之后需要安裝這個(gè))

npm install webpack-cli --save

安裝后,執(zhí)行 npm run dev 會發(fā)現(xiàn)控制臺有個(gè)警告 WARNING in configuration ,去除WARNING in configuration 警告,在webpack.config.dev.js 增加一個(gè)配置即可:

...
mode: "development"
...

成功之后會在dev下面生成bundle.min.js代表正常。

如果想要?jiǎng)討B(tài)監(jiān)聽文件變化需要在命令后面添加 --watch。

react

1.安裝react

npm install react react-dom --save

2.創(chuàng)建page目錄和index頁面文件:

mkdir src
mkdir page
cd page

3.創(chuàng)建index

mkdir index
cd index & touch index.js & touch index.html

index.js

import ReactDom from "react-dom";
import Main from "./Main/Main.jsx";

ReactDom.render(
, document.getElementById("root"));

index.html




    index
    
    



4.創(chuàng)建Main組件

import React from "react";

class Main extends React.Component {

    constructor(props) {
        super(props);
    }

    render() {
        return (
Main
); } } export default Main;

5.修改webpack配置入口文件

entry: [
    path.resolve(srcRoot,"./page/index/index.js")
],

配置loader

1.處理樣式文件需要這些loader:

css-loader

sass-loader

style-loader

npm install css-loader sass-loader style-loader file-loader --save

配置:

module: {
    // 加載器配置
    rules: [
        { test: /.css$/, use: ["style-loader", "css-loader"], include: path.resolve(srcRoot)},
        { test: /.scss$/, use: ["style-loader", "css-loader", "sass-loader"], include: path.resolve(srcRoot)}
    ]
},

2.url-loader 處理處理靜態(tài)文件

npm install url-loader --save

配置:

module: {
    // 加載器配置
    rules: [
        { test: /.(png|jpg|jpeg)$/, use: "url-loader?limit=8192&name=images/[name].[hash].[ext]", include: path.resolve(srcRoot)}
    ]
},

limit:表示超過多少就使用base64來代替,單位是byte

name:可以設(shè)置圖片的路徑,名稱和是否使用hash 具體參考這里

引入babel

bebel是用來解析es6語法或者是es7語法分解析器,讓開發(fā)者能夠使用新的es語法,同時(shí)支持jsx,vue等多種框架。

1.安裝babel

babel-core

babel-loader

npm install babel-core babel-loader --save

配置:

module: {
    // 加載器配置
    rules: [
        { test: /.(js|jsx)$/, use: [{loader:"babel-loader"}] ,include: path.resolve(srcRoot)},
    ]
},

2.babel配置文件:.babelrc

touch .babelrc

配置:

{
    "presets": [
        "es2015",
        "react",
        "stage-0"
    ],
    "plugins": []
}

babel支持自定義的預(yù)設(shè)(presets)或插件(plugins),只有配置了這兩個(gè)才能讓babel生效,多帶帶的安裝babel是無意義的。

presets:代表babel支持那種語法(就是你用那種語法寫),優(yōu)先級是從下往上,state-0|1|2|..代表有很多沒有列入標(biāo)準(zhǔn)的語法回已state-x表示,參考這里

plugins:代表babel解析的時(shí)候使用哪些插件,作用和presets類似,優(yōu)先級是從上往下。

依次安裝:

babel-preset-es2015

babel-preset-react

babel-preset-stage-0

npm install babel-preset-es2015 babel-preset-react babel-preset-stage-0 --save

3.babel-polyfill是什么?

我們之前使用的babel,babel-loader 默認(rèn)只轉(zhuǎn)換新的 JavaScript 語法,而不轉(zhuǎn)換新的 API。例如,Iterator、Generator、Set、Maps、Proxy、Reflect、Symbol、Promise 等全局對象,以及一些定義在全局對象上的方法(比如 Object.assign)都不會轉(zhuǎn)譯。如果想使用這些新的對象和方法,必須使用 babel-polyfill,為當(dāng)前環(huán)境提供一個(gè)墊片。

npm install --save babel-polyfill

4.transform-runtime 有什么區(qū)別?

當(dāng)使用babel-polyfill時(shí)有一些問題:

默認(rèn)會引入所有babel支持的新語法,這樣就會導(dǎo)致你的文件代碼非常龐大。

通過向全局對象和內(nèi)置對象的prototype上添加方法來達(dá)成目的,造成全局變量污染。

這時(shí)就需要transform-runtime來幫我們有選擇性的引入:

npm install --save babel-plugin-transform-runtime

配置文件:

{
  "plugins": [
    ["transform-runtime", {
      "helpers": false,
      "polyfill": false,
      "regenerator": true,
      "moduleName": "babel-runtime"
    }]
  ]
}


使用HtmlWebpackPlugin

記得我們之前新建的index.html么 我們執(zhí)行構(gòu)建命令之后并沒有將index.html打包到dev目錄下 我們需要HtmlWebpackPlugin來將我們output的js和html結(jié)合起來:

npm install html-webpack-plugin --save

配置:

const HtmlWebpackPlugin = require("html-webpack-plugin");
...
plugins: [
    new HtmlWebpackPlugin({
        filename: path.resolve(devPath, "index.html"),
        template: path.resolve(srcRoot, "./page/index/index.html"),
    })
]

filename:可以設(shè)置html輸出的路徑和文件名
template:可以設(shè)置已哪個(gè)html文件為模版

更多參數(shù)配置可以參考這里

redux

關(guān)于redux的使用可以參考阮一峰老師的入門教程

1.安裝redux

redux

react-redux

npm install redux react-redux --save

1.新建reducers,actions目錄和文件

|—— index/                          
|—— Main/                   * 組件代碼
|       |—— Main.jsx        * 組件jsx
|       |—— Main.scss       * 組件css
|
|—— actions/ 
|       |—— actionTypes.js  * action常量
|       |—— todoAction.js   * action
|
|—— reducers/ 
|       |—— todoReducer.js  * reducer
|
|—— store.js
|
|—— index.js

2.修改代碼,引入redux,這里以一個(gè)redux todo為demo例子:

index.js

import ReactDom from "react-dom";
import React from "react";
import Main from "./Main/Main.jsx";
import store from "./store.js";
import { Provider } from "react-redux";

ReactDom.render(
    
        
, document.getElementById("root"));

store.js

import { createStore } from "redux";
import todoReducer from "./reducers/todoReducer.js";

const store = createStore(todoReducer);

export default store;  

tabReducer.js

    import { ADD_TODO } from "../actions/actionTypes.js";
    
    const initialState = {
          todoList: []
    };
    
    const addTodo = (state, action) => {
    
      return { ...state, todoList: state.todoList.concat(action.obj) }
    }
    
    const todoReducer = (state = initialState, action) => {
      switch(action.type) {
        case ADD_TODO: return addTodo(state, action);
        default: return state;
      }
    };
    export default todoReducer;

Main.jsx

import React from "react";
import { connect } from "react-redux";
import { addTodo } from "../actions/todoAction.js";

class Main extends React.Component {

    onClick(){
        let text = this.refs.input;

        this.props.dispatch(addTodo({
            text: text.value
        }))
    }
    render() {
        return (
            
    {this.props.todoList.map((item, index)=>{ return
  • {item.text}
  • })}
); } } export default connect( state => ({ todoList: state.todoList }) )(Main);

todoAction.js

import { ADD_TODO } from "./actionTypes.js";

export const addTodo = (obj) => {
  return {
    type: ADD_TODO,
    obj: obj
  };
};

使用webpack-dev-server

webpack-dev-server是一個(gè)小型的Node.js Express服務(wù)器,它使用webpack-dev-middleware來服務(wù)于webpack的包。

1.安裝

npm install webpack-dev-server --save

修改在package.json中添加的執(zhí)行命令:

"scripts": {
  "dev": "./node_modules/.bin/webpack-dev-server --config webpack.config.dev.js",
},

2.配置webpack配置文件:

devServer: {
    "contentBase": devPath,
    "compress": true,
},

contentBase: 表示server文件的根目錄
compress: 表示開啟gzip

更多的配置文檔參考這里

webpack-dev-server默認(rèn)情況下會將output的內(nèi)容放在內(nèi)存中,是看不到物理的文件的,如果想要看到物理的dev下面的文件可以安裝write-file-webpack-plugin這個(gè)插件。

webpack-dev-server默認(rèn)會開啟livereload功能

3.devtool功能:

具體來說添加了devtool: "inline-source-map"之后,利用source-map你在chrome控制臺看到的source源碼都是真正的源碼,未壓縮,未編譯前的代碼,沒有添加,你看到的代碼是真實(shí)的壓縮過,編譯過的代碼,更多devtool的配置可以參考這里。

多入口文件配置

在之前的配置中,都是基于單入口頁面配置的,entry和output只有一個(gè)文件,但是實(shí)際項(xiàng)目很多情況下是多頁面的,在配置多頁面時(shí),有2中方法可以選擇:

1.在entry入口配置時(shí),傳入對象而不是多帶帶數(shù)組,output時(shí)利用[name]關(guān)鍵字來區(qū)分輸出文件例如:

entry: {
    index: [path.resolve(srcRoot,"./page/index/index1.js"),path.resolve(srcRoot,"./page/index/index2.js")],
    detail: path.resolve(srcRoot,"./page/detail/detail.js"),
    home: path.resolve(srcRoot,"./page/home/home.js"),
},
output: {
    path: path.resolve(__dirname, "./dev"),

    filename: "[name].min.js"
},

2.通過node動態(tài)遍歷需要entry point的目錄,來動態(tài)生成entry:

const pageDir = path.resolve(srcRoot, "page");
function getEntry() {
    let entryMap = {};

    fs.readdirSync(pageDir).forEach((pathname)=>{
        let fullPathName = path.resolve(pageDir, pathname);
        let stat = fs.statSync(fullPathName);
        let fileName = path.resolve(fullPathName, "index.js");

        if (stat.isDirectory() && fs.existsSync(fileName)) {
            entryMap[pathname] = fileName;
        }

    });

    return entryMap;
}
{
    ...
    entry: getEntry()
    ...
}

本demo采用的是第二中寫法,能夠更加靈活。

如何理解entry point(bundle),chunk,module

在webpack中,如何理解entry point(bundle),chunk,module?

根據(jù)圖上的表述,我這里簡單說一下便于理解的結(jié)論:

配置中每個(gè)文件例如index1.js,index2.js,detail.js,home.js都屬于entry point.

entry這個(gè)配置中,每個(gè)key值,index,detail,home都相當(dāng)于chunk。

我們在代碼中的require或者import的都屬于module,這點(diǎn)很好理解。

chunk的分類比較特別,有entry chunk,initial chunk,normal chunk,參考這個(gè)文章

正常情況下,一個(gè)chunk對應(yīng)一個(gè)output,在使用了CommonsChunkPlugin或者require.ensure之后,chunk就變成了initial chunk,normal chunk,這時(shí),一個(gè)chunk對應(yīng)多個(gè)output。

理解這些概念對于后續(xù)使用webpack插件有很大的幫助。

多入口頁面html配置

之前我們配置HtmlWebpackPlugin時(shí),同樣采用的是但頁面的配置,這里我們將進(jìn)行多頁面改造,entryMap是上一步得到的entry:

function htmlAarray(entryMap) {
    let htmlAarray = [];

    Object.keys(entryMap).forEach(function(key){
        let fullPathName = path.resolve(pageDir, key);
        let fileName = path.resolve(fullPathName, key + ".html")
        if (fs.existsSync(fileName)) {
            htmlAarray.push(new HtmlWebpackPlugin({
                chunks: key, // 注意這里的key就是chunk
                filename: key + ".html",
                template: fileName,
                inlineSource:  ".(js|css)"
            }))
        }
    });

    return htmlAarray;

}
修改plugin配置:plugins: [
     ...
].concat(htmlMap)

模塊熱替換(Hot Module Replacement)

模塊熱替換 (Hot Module Replacement 或 HMR)是 webpack 提供的最有用的功能之一。它允許在運(yùn)行時(shí)更新各種模塊,而無需進(jìn)行完全刷新,很高大上有木有!

下面說一下配置方法,它需要結(jié)合devServer使用:

devServer: {
    hot: true // 開啟HMR
},

開啟plugin:

const webpack = require("webpack");
plugins: [
    new webpack.NamedModulesPlugin(),
    new webpack.HotModuleReplacementPlugin(),
].concat(htmlMap)

結(jié)合React一起使用:

1.安裝react-hot-loader

npm install react-hot-loader --save

并新建一個(gè)Container.jsx:

import React from "react";
import Main from "./Main.jsx";
import { hot } from "react-hot-loader"

class Container extends React.Component {

    render() {
        return 
} } export default hot(module)(Container);

結(jié)合redux:如果項(xiàng)目沒有使用redux,可以無需配置后面2步

2.修改store.js新增下面代碼,為了讓reducer也能實(shí)時(shí)熱替換

if (module.hot) {
    module.hot.accept("./reducers/todoReducer.js", () => {
      const nextRootReducer = require("./reducers/todoReducer.js").default;
      store.replaceReducer(nextRootReducer);
    });
}

3.修改index.js

import ReactDom from "react-dom";
import React from "react";
import Container from "./Main/Container.jsx";
import store from "./store.js";

import { Provider } from "react-redux";

ReactDom.render(
    
        
    
, document.getElementById("root"));

當(dāng)控制臺看到[WDS] Hot Module Replacement enabled.代表開啟成功

使用ESLint

ESLint 是眾多 Javascript Linter 中的其中一種,其他比較常見的還有 JSLint 跟 JSHint,之所以用 ESLint 是因?yàn)樗梢宰杂蛇x擇要使用哪些規(guī)則,也有很多現(xiàn)成的 plugin 可以使用,另外他對 ES6 還有 JSX 的支持程度跟其他 linter 相比之下也是最高的。

1.安裝ESLint

npm install eslint eslint-loader babel-eslint --save

其中eslint-loader是將webpack和eslint結(jié)合起來在webpack的配置文件中新增一個(gè)eslint-loader種,修改如下:

{ test: /.(js|jsx)$/, use: [{loader:"babel-loader"},{loader:"eslint-loader"}] ,include: path.resolve(srcRoot)},   


2.新建.eslintrc配置文件,將parser配置成babel-eslint

{
    "extends": ["eslint:recommended"],
    
    "parser": "babel-eslint",

    "globals": {
    },
    "rules": {
    }
} 

3.安裝eslint-plugin-react:

npm install eslint-plugin-react --save

說明一下,正常情況下每個(gè)eslint規(guī)則都是需要在rule下面配置,如果什么都不配置,其實(shí)本身eslint是不生效的。

eslint本身有很多默認(rèn)的規(guī)則模版,可以通過extends來配置,默認(rèn)可以使用eslint:recommended。

在使用react開發(fā)時(shí)可以安裝eslint-plugin-react來告知使用react專用的規(guī)則來lint。

修改.eslintrc配置文件,增加rules,更多rules配置可以參考這里

{
    "extends": ["eslint:recommended","plugin:react/recommended"],
    
    "parser": "babel-eslint",

    "globals": {
        "window": true,
        "document": true,
        "module": true,
        "require": true
    },
    "rules": {
        "react/prop-types" : "off",
        "no-console" : "off"
    }
}
使用react-router

react-router強(qiáng)大指出在于方便代碼管理,結(jié)合redux使用更加強(qiáng)大,同時(shí)支持web,native更多參考這里

1.安裝react-router-dom

npm install react-router-dom --save

2.如果項(xiàng)目中用了redux,可以安裝react-router-redux

npm install react-router-redux@next history --save

3.修改代碼:

index.js

import ReactDom from "react-dom";
import React from "react";
import Container from "./Main/Container.jsx";
import { store, history } from "./store.js";

import { Provider } from "react-redux";

import createHistory from "history/createHashHistory";
import { ConnectedRouter } from "react-router-redux";

const history = createHistory();

ReactDom.render(
    
        
            
        
    
, document.getElementById("root"));

結(jié)合history,react-router一共有3中不同的router:

BrowserRouter通過history/createBrowserHistory引入:當(dāng)切換時(shí),url會動態(tài)更新,底層使用的時(shí)html5的pushState。

HashRouter通過history/createHashHistory引入:當(dāng)切換時(shí),動態(tài)修改hash,利用hashchange事件。

MemoryRouter 通過history/createMemoryHistory引入:將路徑,路由相關(guān)數(shù)據(jù)存入內(nèi)存中,不涉及url相關(guān)更新,兼容性好。

更多配置可以參考這里

4.如果想要在代碼邏輯中獲取當(dāng)前的route路徑需要引入router-reducer:

新建main.js:

import { combineReducers } from "redux";
import { routerReducer } from "react-router-redux";
import todoReducer from "./todoReducer.js";

const reducers = combineReducers({
  todoReducer,
  router: routerReducer
});
export default reducers;

修改store.js:

import { createStore } from "redux";
import mainReducer from "./reducers/main.js";

const store = createStore(mainReducer);

export default store;

然后就可以在this.props.router里面獲取單相關(guān)的路徑信息

5.如果需要自己通過action來觸發(fā)router的跳轉(zhuǎn),需要引入routerMiddleware:

import { createStore,applyMiddleware } from "redux";
import { routerMiddleware } from "react-router-redux";
const middleware = routerMiddleware(history);
const store = createStore(mainReducer,applyMiddleware(middleware));

6.使用Route和Link和withRouter:

先說說都是干嘛的:



export default withRouter(connect(
   state => ({
      todoList: state.todoReducer.todoList
   })
)(Main));

如果你在使用hash時(shí)遇到Warning: Hash history cannot PUSH the same path; a new entry will not be added to the history stack錯(cuò)誤,可以將push改為replace即:

切換到2號

7.設(shè)置初始化路由:

BrowserRouter和HashRouter:

const history = createHistory();
history.push("2");

MemoryRouter:

const history = createMemoryHistory({
    initialEntries: ["/2"]
});


使用redux-thunk

redux-thunk 是一個(gè)比較流行的 redux 異步 action 中間件,比如 action 中有 setTimeout 或者通過 fetch通用遠(yuǎn)程 API 這些場景,那么久應(yīng)該使用 redux-thunk 了。redux-thunk 幫助你統(tǒng)一了異步和同步 action 的調(diào)用方式,把異步過程放在 action 級別解決,對 component 沒有影響。

1.安裝redux-thunk:

npm install redux-thunk --save

2.修改store.js:

import { createStore,applyMiddleware } from "redux";
import thunk from "redux-thunk";
import mainReducer from "./reducers/main";
...
const store = createStore(mainReducer, applyMiddleware(thunk));
...
export default store;

3.在action.js使用redux-thunk:

export const getData = (obj) => (dispatch, getState) => {
  setTimeout(()=>{
    dispatch({
        type: GET_DATA,
        obj: obj
    });
  },1000);
};
使用axios和async/await

axios 是一個(gè)基于Promise 用于瀏覽器和 nodejs 的 HTTP 客戶端:

從瀏覽器中創(chuàng)建 XMLHttpRequest

從 node.js 發(fā)出 http 請求

支持 Promise API

自動轉(zhuǎn)換JSON數(shù)據(jù)

1.安裝axios:

npm install axios --save

2.在action中使用axios:

import axios from "axios";
export const getData = (obj) => (dispatch, getState) => {
    axios.get("/json/comments.json").then((resp)=>{
        dispatch({
            type: GET_DATA,
            obj: resp
        });
    });
};

async/await:

Javascript的回調(diào)地獄,相信很多人都知道,尤其是在node端,近些年比較流行的是Promise的解決方案,但是隨著 Node 7 的發(fā)布,編程終級解決方案的 async/await應(yīng)聲而出。

function resolveAfter2Seconds() {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve("resolved");
    }, 2000);
  });
}

async function asyncCall() {
  var result = await resolveAfter2Seconds();
}

asyncCall();

async/await的用途是簡化使用 promises 異步調(diào)用的操作,并對一組 Promises執(zhí)行某些操作。await前提是方法返回的是一個(gè)Promise對象,正如Promises類似于結(jié)構(gòu)化回調(diào),async/await類似于組合生成器和 promises。

1.async/await需要安裝babel-plugin-transform-async-to-generator。

npm install babel-plugin-transform-async-to-generator --save

2.在.babelrc中增加配置:

"plugins": [
    "transform-async-to-generator"
]

這樣做僅僅是將async轉(zhuǎn)換generator,如果你當(dāng)前的瀏覽器不支持generator,你將會收到一個(gè)Uncaught ReferenceError: regeneratorRuntime is not defined的錯(cuò)誤,你需要:

3.安裝babel-plugin-transform-runtime:

npm install babel-plugin-transform-async-to-generator --save

4.修改.babelrc中的配置(可以去掉之前配置的transform-async-to-generator):

"plugins": [
    "transform-runtime"
]

5.如果不想引入所有的polyfill(參考上面對babel的解釋),可以增加配置:

"plugins": [
    "transform-runtime",
        {
            "polyfill": false,

            "regenerator": true,
        }
]

6.結(jié)合axios使用

import axios from "axios";
export const getData = (obj) => async (dispatch, getState) => {
    let resp = axios.get("/json/comments.json");
    dispatch({
        type: GET_DATA,
        obj: resp
    });
};

Code Splitting

1.對于webpack1,2之前,你可以使用require.ensure來控制一個(gè)組件的懶加載:

require.ensure([], _require => {
    let Component = _require("./Component.jsx");
},"lazyname")

2.在webpack4中,官方已經(jīng)不再推薦使用require.ensure來使用懶加載功能Dynamic Imports,取而代之的是ES6的import()方法:

import(
  /* webpackChunkName: "my-chunk-name" */
  /* webpackMode: "lazy" */
  "module"
);

不小小看注釋里的代碼,webpack在打包時(shí)會動態(tài)識別這里的代碼來做相關(guān)的配置,例如chunk name等等。

3.Prefetching/Preloading modules:

webpack 4.6.0+支持了Prefetching/Preloading的寫法:

import(/* webpackPreload: true */ "ChartingLibrary");

3.結(jié)合React-Router使用:

react-loadable對上述的功能做了封裝,豐富了一些功能,結(jié)合React-Router起來使用更加方便。

npm install react-loadable --save

在react-router里使用:

function Loading() {
  return 
Loading...
; } let Div2 = Loadable({ loader: () => import("./Div2"), loading: Loading, });
使用CommonsChunkPlugin

CommonsChunkPlugin 插件,是一個(gè)可選的用于建立一個(gè)獨(dú)立文件(又稱作 chunk)的功能,這個(gè)文件包括多個(gè)入口 chunk 的公共模塊。通過將公共模塊拆出來,最終合成的文件能夠在最開始的時(shí)候加載一次,便存起來到緩存中供后續(xù)使用。

1.在webpack4之前的用法:

new webpack.optimize.CommonsChunkPlugin({
    name: "common",
    chunks: ["page1","page2"],
    minChunks: 3
})

name: string: 提出出的名稱

chunks: string[]: webpack會從傳入的chunk里面提取公共代碼,默認(rèn)從所有entry里提取

minChunks: number|infinity|function(module,count)->boolean: 如果傳入數(shù)字或infinity(默認(rèn)值為3),就是告訴webpack,只有當(dāng)模塊重復(fù)的次數(shù)大于等于該數(shù)字時(shí),這個(gè)模塊才會被提取出來。當(dāng)傳入為函數(shù)時(shí),所有符合條件的chunk中的模塊都會被傳入該函數(shù)做計(jì)算,返回true的模塊會被提取到目標(biāo)chunk。

更多的參數(shù)配置,可以參考這里

2.在webpack4之后的用法:

module.exports = {
  //...
  optimization: {
    splitChunks: {
      chunks: "async",
      minSize: 30000,
      minChunks: 1,
      maxAsyncRequests: 5,
      maxInitialRequests: 3,
      automaticNameDelimiter: "~",
      name: true,
      cacheGroups: {
        vendors: {
          test: /[/]node_modules[/]/,
          priority: -10
        },
        default: {
          minChunks: 2,
          priority: -20,
          reuseExistingChunk: true
        }
      }
    }
  }
};

splitChunks: 配置一個(gè)分離chunk(代替老版本的CommonsChunkPlugin)

cacheGroups: 自定義配置主要使用它來決定生成的文件:

test: 限制范圍

name: 生成文件名

priority: 優(yōu)先級

minSize: number: 最小尺寸必須大于此值,默認(rèn)30000B

minChunks: 其他entry引用次數(shù)大于此值,默認(rèn)1

maxInitialRequests: entry文件請求的chunks不應(yīng)該超過此值(請求過多,耗時(shí))

maxAsyncRequests: 異步請求的chunks不應(yīng)該超過此值

automaticNameDelimiter: 自動命名連接符

chunks: 值為”initial”, “async”(默認(rèn)) 或 “all”:

initial: 入口chunk,對于異步導(dǎo)入的文件不處理

async: 異步chunk,只對異步導(dǎo)入的文件處理

all: 全部chunk

你的點(diǎn)贊是我持續(xù)分享好東西的動力,歡迎點(diǎn)贊!

交流

干貨系列文章匯總?cè)缦?,覺得不錯(cuò)點(diǎn)個(gè)Star,歡迎 加群 互相學(xué)習(xí)。

https://github.com/qq44924588...

我是小智,公眾號「大遷世界」作者,對前端技術(shù)保持學(xué)習(xí)愛好者。我會經(jīng)常分享自己所學(xué)所看的干貨,在進(jìn)階的路上,共勉!

關(guān)注公眾號,后臺回復(fù)福利,即可看到福利,你懂的。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/108654.html

相關(guān)文章

  • 前方來報(bào),八月最新資訊--關(guān)于vue2&3的最佳文章推薦

    摘要:哪吒別人的看法都是狗屁,你是誰只有你自己說了才算,這是爹教我的道理。哪吒去他個(gè)鳥命我命由我,不由天是魔是仙,我自己決定哪吒白白搭上一條人命,你傻不傻敖丙不傻誰和你做朋友太乙真人人是否能夠改變命運(yùn),我不曉得。我只曉得,不認(rèn)命是哪吒的命。 showImg(https://segmentfault.com/img/bVbwiGL?w=900&h=378); 出處 查看github最新的Vue...

    izhuhaodev 評論0 收藏0
  • webpack 基礎(chǔ)與項(xiàng)目優(yōu)化實(shí)踐總結(jié)

    摘要:前言本文基于,主要涉及基本概念基本配置和實(shí)際項(xiàng)目打包優(yōu)化。關(guān)于概念方面參考官網(wǎng),常用配置來自于網(wǎng)絡(luò)資源,在文末有相關(guān)參考鏈接,實(shí)踐部分基于自己的項(xiàng)目進(jìn)行優(yōu)化配置。同一文件中,修改某個(gè)影響其他。 前言:本文基于weboack4.x,主要涉及webpack4 基本概念、基本配置和實(shí)際項(xiàng)目打包優(yōu)化。關(guān)于概念方面參考官網(wǎng),常用配置來自于網(wǎng)絡(luò)資源,在文末有相關(guān)參考鏈接,實(shí)踐部分基于自己的項(xiàng)目進(jìn)行...

    Scorpion 評論0 收藏0
  • 關(guān)于Vue2一些值得推薦的文章 -- 五、六月份

    摘要:五六月份推薦集合查看最新的請點(diǎn)擊集前端最近很火的框架資源定時(shí)更新,歡迎一下。蘇幕遮燎沈香宋周邦彥燎沈香,消溽暑。鳥雀呼晴,侵曉窺檐語。葉上初陽乾宿雨,水面清圓,一一風(fēng)荷舉。家住吳門,久作長安旅。五月漁郎相憶否。小楫輕舟,夢入芙蓉浦。 五、六月份推薦集合 查看github最新的Vue weekly;請::點(diǎn)擊::集web前端最近很火的vue2框架資源;定時(shí)更新,歡迎 Star 一下。 蘇...

    sutaking 評論0 收藏0
  • 關(guān)于Vue2一些值得推薦的文章 -- 五、六月份

    摘要:五六月份推薦集合查看最新的請點(diǎn)擊集前端最近很火的框架資源定時(shí)更新,歡迎一下。蘇幕遮燎沈香宋周邦彥燎沈香,消溽暑。鳥雀呼晴,侵曉窺檐語。葉上初陽乾宿雨,水面清圓,一一風(fēng)荷舉。家住吳門,久作長安旅。五月漁郎相憶否。小楫輕舟,夢入芙蓉浦。 五、六月份推薦集合 查看github最新的Vue weekly;請::點(diǎn)擊::集web前端最近很火的vue2框架資源;定時(shí)更新,歡迎 Star 一下。 蘇...

    khs1994 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<