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

資訊專欄INFORMATION COLUMN

如何打造一個令人愉悅的前端開發(fā)環(huán)境(四)

StonePanda / 2024人閱讀

摘要:在前后端分離的前端項(xiàng)目開發(fā)中經(jīng)常用到。是的一個中間件。即是一個重要的功能。配置先來在配置文件中引入添加一個和通信的客戶端添加應(yīng)用入口文件在插件中引入在我們的開發(fā)環(huán)境中是這樣配置的。

原文鏈接
此文是我同事寫的,搭建Express結(jié)合Webpack。以下是正文,后面我會附上我的解讀

Express 結(jié)合 Webpack 實(shí)現(xiàn)HMR

本篇文件主要講結(jié)合 Webpack 和 Express 實(shí)現(xiàn)前后端熱更新開發(fā),如果你還不太了解webpack推薦閱讀
webpack 官網(wǎng)文檔

What 什么是 webpack dev server

Webpack dev server 是一個輕量的node.js express服務(wù)器,實(shí)現(xiàn)了 webpack 編譯代碼實(shí)時輸出更新。在前后端分離的前端項(xiàng)目開發(fā)中經(jīng)常用到。不過這篇文章應(yīng)該不會講到它。

webpack dev middleware

Webpack dev middleware 是 WebPack 的一個中間件。它用于在 Express 中分發(fā)需要通過 WebPack 編譯的文件。多帶帶使用它就可以完成代碼的熱重載(hot reloading)功能。

特性:

不會在硬盤中寫入文件,完全基于內(nèi)存實(shí)現(xiàn)。

如果使用 watch 模式監(jiān)聽代碼修改,Webpack 會自動編譯,如果在 Webpack 編譯過程中請求文件,Webpack dev middleware 會延遲請求,直到編譯完成之后再開始發(fā)送編譯完成的文件。

webpack hot middleware

Webpack hot middleware 它通過訂閱 Webpack 的編譯更新,之后通過執(zhí)行 webpack 的 HMR api 將這些代碼模塊的更新推送給瀏覽器端。

HMR

HMR 即 Hot Module Replacement 是 Webpack 一個重要的功能。它可以使我們不用通過手動地刷新瀏覽器頁面實(shí)現(xiàn)將我們的更新代碼實(shí)時應(yīng)用到當(dāng)前頁面中。

HMR 的實(shí)現(xiàn)原理是在我們的開發(fā)中的應(yīng)用代碼中加入了 HMR Runtime,它是 HMR 的客戶端(瀏覽器端 client)用于和開發(fā)服務(wù)器通信,接收更新的模塊。服務(wù)端工作就是前面提到的 Webpack hot middleware 的,它會在代碼更新編譯完成之后通過以 json 格式輸出給HMR Runtime 就會更具 json 中描述來動態(tài)更新相應(yīng)的代碼。

How webpack 配置

先來在webpack配置文件中引入

var webpack = require("webpack");
var HotMiddleWareConfig = "webpack-hot-middleware/client?path=/__webpack_hmr&timeout=20000"

module.exports = {
    context: __dirname,
    entry: [
        // 添加一個和HotMiddleWare通信的客戶端             
        HotMiddleWareConfig,
        // 添加web應(yīng)用入口文件 
        "./client.js"
    ],
    output: {
        path: __dirname,
        publicPath: "/",
        filename: "bundle.js"
    },
    devtool: "#source-map",
    plugins: [
        new webpack.optimize.OccurenceOrderPlugin(),
        // 在 webpack 插件中引入 webpack.HotModuleReplacementPlugin 
        new webpack.HotModuleReplacementPlugin(),
        new webpack.NoErrorsPlugin()
    ],
};

webpack-hot-middleware example webpack.config.js

在我們的開發(fā)環(huán)境中是這樣配置的。getEntries 是自動根據(jù)我們規(guī)則獲取到入口文件并加上 webpack hot middle 配置。

var webpack = require("webpack");
var path = require("path")
var merge = require("webpack-merge")
var baseConfig = require("./webpack.base")
var getEntries = require("./getEntries")

var publicPath = "http://0.0.0.0:7799/dist/";
var hotMiddlewareScript = "webpack-hot-middleware/client?reload=true";

var assetsInsert = require("./assetsInsert")

module.exports = merge(baseConfig, {
  entry: getEntries(hotMiddlewareScript),
  devtool: "#eval-source-map",
  output: {
    filename: "./[name].[hash].js",
    path: path.resolve("./public/dist"),
    publicPath: publicPath
  },
  plugins: [
    new webpack.DefinePlugin({
      "process.env": {
        NODE_ENV: ""development""
      }
    }),
    new webpack.optimize.OccurenceOrderPlugin(),
    new webpack.HotModuleReplacementPlugin(),
    new webpack.NoErrorsPlugin(),
    new assetsInsert()
  ]
})
Express 中的配置

在 Express 的配置主要就4個步驟:

引入 webpack 的配置文件和 生成 webpack 的編譯器

將編譯器連接至 webpack dev middleware

將編譯器連接至 webpack hot middleware

定義 express 配置

var http = require("http");

var express = require("express");

var app = express();

app.use(require("morgan")("short"));

// ************************************
// This is the real meat of the example
// ************************************
(function() {
  // Step 1: 引入 webpack 的配置文件和 生成 webpack 的編譯器
  var webpack = require("webpack");
  var webpackConfig = require(process.env.WEBPACK_CONFIG ? process.env.WEBPACK_CONFIG : "./webpack.config");
  var compiler = webpack(webpackConfig);
  // Step 2: 將編譯器掛載給 webpack dev middleware
  app.use(require("webpack-dev-middleware")(compiler, {
    noInfo: true, publicPath: webpackConfig.output.publicPath
  }));

  // Step 3: 將編譯器掛載給 webpack hot middleware
  app.use(require("webpack-hot-middleware")(compiler, {
    log: console.log, path: "/__webpack_hmr", heartbeat: 10 * 1000
  }));
})();

// 定義 express 配置

app.get("/", function(req, res) {
  res.sendFile(__dirname + "/index.html");
});
app.get("/multientry", function(req, res) {
  res.sendFile(__dirname + "/index-multientry.html");
});

if (require.main === module) {
  var server = http.createServer(app);
  server.listen(process.env.PORT || 1616, function() {
    console.log("Listening on %j", server.address());
  });
}

webpack-hot-middleware example server.js

區(qū)分開發(fā)和生產(chǎn)環(huán)境

要注意的是一定要在定義 express router 前定義 webpack 相關(guān)的中間件。還有一點(diǎn)是這里server.js 只是開發(fā)環(huán)境中使用,在生成環(huán)境中我們就不需要再用到它們了。所以在我們實(shí)際的使用中需要通過定義環(huán)境變量來區(qū)分開發(fā)和生產(chǎn)環(huán)境

var NODE_ENV = process.env.NODE_ENV || "production";
var isDev = NODE_ENV === "development";

if (isDev) {
    var webpack = require("webpack"),
        webpackDevMiddleware = require("webpack-dev-middleware"),
        webpackHotMiddleware = require("webpack-hot-middleware"),
        webpackDevConfig = require("./build/webpack.config.js");

    var compiler = webpack(webpackDevConfig);

    app.use(webpackDevMiddleware(compiler, {
        publicPath: webpackDevConfig.output.publicPath,
        noInfo: true,
        stats: {
            colors: true
        }
    }));

    app.use(webpackHotMiddleware(compiler));

    routerConfig(app, {
        dirPath: __dirname + "/server/routes/",
        map: {
            "index": "/",
            "api": "/api/*",
            "proxy": "/proxy/*"
        }
    });

    var reload = require("reload");
    var http = require("http");

    var server = http.createServer(app);
    reload(server, app);

    app.use(express.static(path.join(__dirname, "public")));

    server.listen(port, function() {
        console.log("App (dev) is now running on port " + port + "!");
    });
} else {
    routerConfig(app, {
        dirPath: __dirname + "/server/routes/",
        map: {
            "index": "/",
            "api": "/api/*",
            "proxy": "/proxy/*"
        }
    });
    app.use(express.static(path.join(__dirname, "public")));

    app.listen(port, function() {
        console.log("App (dev) is now running on port " + port + "!");
    });
}
supervisor

以上在前端我們實(shí)現(xiàn)了前端文件的熱更新,但是我們在修改服務(wù)端文件的時候,并不會使Node自動重啟,所以我們使用 supervisor 來作為監(jiān)聽文件修改事件來自動重啟 Node服務(wù)。
supervisor 需要 全局安裝

  npm install supervisor -g

安裝完成之后我們就可以在命令行中使用
我們在 package.json 的 scripts 中寫好常用的命令,之后只用 npm run xxx 即可使用

"scripts": {
    "dev": "export NODE_ENV=development && supervisor -w server,app.js app",
    "build": "node build/build.js",
    "start": "node app"
  },

node-supervisor

supervisor [options] 

supervisor -w server,app.js app

-w 就是一個 options 配置項(xiàng),它用于監(jiān)聽指定目錄或者文件的變更,可以使用,
分隔,監(jiān)聽多個目錄或者文件,這就是監(jiān)聽了 server 目錄和根目錄的 app.js 到變更之后就會重啟我們的 Express 入口文件 app。

我自己的解讀 首先解釋下他原文代碼幾個難以解釋的點(diǎn):

getEntries 這個是我們自己的載入入口文件的統(tǒng)一方法,具體內(nèi)容其實(shí)我前面的文章提到過,就是規(guī)定好了的,文件夾目錄下的main.js就是我們的入口文件,其他全部忽略,原因也說過,這里再說一次,規(guī)定死了,簡單,方便,利于合作。

var publicPath = "http://0.0.0.0:7799/dist/"; 這里publicPath和大家平時配置Webpack的publicPath 不太一樣的原因,是需要Express能夠認(rèn)出絕對地址來,因?yàn)槟沩?xiàng)目是Express大于Webpack的。

routerConfig 這個方法是我們自己的一個方法,內(nèi)容前面文章提到過,就是我寫的一個用來加載所有路由的方法,免得重復(fù)寫各種引用。npm地址

export NODE_ENV=development 注意這里,windows的環(huán)境可能會失敗,可以替換成 cross-env NODE_ENV=development

其次說下這么做的原因

前面文章也提到了,我司目前前端的整體架構(gòu)是使用Node做中間層,那么問題就是Node渲染層會高于Webpack層,而且很多時候,不一定使用SPA的方式,要兼容這個架構(gòu),所以才需要這些配置。

這個配置也很好解決了我們開發(fā)中的幾個痛點(diǎn),一個Node自動重新,一個文件熱更新,結(jié)合起來,基本不需要自己不停的手動刷新瀏覽器,而且能保存當(dāng)前狀態(tài),這點(diǎn)很關(guān)鍵,能節(jié)省不少時間,提升開發(fā)效率。

當(dāng)然也有痛點(diǎn),例如要多一個模板文件,而且文件目錄要根據(jù)規(guī)范來,要不是不會渲染的。

最后記得這是一個連續(xù)的文章系列,只看這里你不一定能配置成功!?。。?/b>

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

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

相關(guān)文章

  • 如何打造一個令人愉悅前端開發(fā)環(huán)境

    摘要:在前后端分離的前端項(xiàng)目開發(fā)中經(jīng)常用到。是的一個中間件。即是一個重要的功能。配置先來在配置文件中引入添加一個和通信的客戶端添加應(yīng)用入口文件在插件中引入在我們的開發(fā)環(huán)境中是這樣配置的。 原文鏈接此文是我同事寫的,搭建Express結(jié)合Webpack。以下是正文,后面我會附上我的解讀 Express 結(jié)合 Webpack 實(shí)現(xiàn)HMR 本篇文件主要講結(jié)合 Webpack 和 Express 實(shí)...

    animabear 評論0 收藏0
  • 如何打造一個令人愉悅前端開發(fā)環(huán)境(二)

    摘要:解析不同文件使用哪些,這個比較簡單,很多文章都有,就不多說了,注意的是,這里的可以換成你自己的預(yù)編譯器,例如等,或者直接用都行,當(dāng)然還可以用一種通用方法,后面補(bǔ)上。 前情提要 上一篇文章介紹了目前前端比較流行的各種編輯器,以及各種流行的打包方式,最后給了一個Gulp的例子,這個例子還是14年的時候?qū)懙?,還有一些可以優(yōu)化的空間,就不討論了,這篇文章主要講目前火熱的打包構(gòu)建方式--Webp...

    Lowky 評論0 收藏0
  • 如何打造一個令人愉悅前端開發(fā)環(huán)境(二)

    摘要:解析不同文件使用哪些,這個比較簡單,很多文章都有,就不多說了,注意的是,這里的可以換成你自己的預(yù)編譯器,例如等,或者直接用都行,當(dāng)然還可以用一種通用方法,后面補(bǔ)上。 前情提要 上一篇文章介紹了目前前端比較流行的各種編輯器,以及各種流行的打包方式,最后給了一個Gulp的例子,這個例子還是14年的時候?qū)懙?,還有一些可以優(yōu)化的空間,就不討論了,這篇文章主要講目前火熱的打包構(gòu)建方式--Webp...

    Object 評論0 收藏0
  • 如何打造一個令人愉悅前端開發(fā)環(huán)境(三)

    摘要:的最后一個大招就是替換一些傳統(tǒng)的服務(wù)端語言,例如,,等,在業(yè)務(wù)層上面使用來開發(fā)服務(wù)端完全不成問題。更多的的使用細(xì)節(jié)和技巧建議關(guān)注美團(tuán)博客大搜車論壇下一篇我們開啟如何結(jié)合和搭建一個開發(fā)環(huán)境和項(xiàng)目目錄 往期回顧 前面2期都講得是瀏覽器端的東西比較多,包括Webpack,雖然是Node處理的,但是還是瀏覽器端用的多,對于現(xiàn)在的前端開發(fā)來說,不懂一點(diǎn)服務(wù)端的東西,簡直沒辦法活,一般的招聘要求都...

    cgh1999520 評論0 收藏0
  • 如何打造一個令人愉悅前端開發(fā)環(huán)境(一)

    摘要:我覺得這方面的原因是當(dāng)時對和的依賴,導(dǎo)致大家對的興趣不弄,錯過了最佳時機(jī),這個其實(shí)跟百度自己的的技術(shù)棧有很大關(guān)系。這個阮一峰對于前端構(gòu)建的變化吐槽過,說新的構(gòu)建工具就是的構(gòu)建工具。 文章來源 最近幾年,前端發(fā)展越來越迅速,各種萌新加入了前端這個大家庭,大有趕IOS、超Android的趨勢呀!同時,萌新們提出了各種前端工作問題,除了最基礎(chǔ)的html、css、js三板斧之外,最讓人頭疼的應(yīng)...

    KavenFan 評論0 收藏0

發(fā)表評論

0條評論

StonePanda

|高級講師

TA的文章

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