国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

webpack 單頁面應(yīng)用實戰(zhàn)

anonymoussf / 2947人閱讀

摘要:關(guān)于這個單頁面應(yīng)用大家可以直接去我的上查看,我將結(jié)合這個項目去介紹。

這篇文章將介紹如何利用 webpack 進(jìn)行單頁面應(yīng)用的開發(fā),算是我在實際開發(fā)中的一些心得和體會,在這里給大家做一個分享。webpack 的介紹這里就不多說了,可以直接去官網(wǎng)查看。 關(guān)于這個單頁面應(yīng)用大家可以直接去我的github上查看https://github.com/huangshuwei/webpackForSPA,我將結(jié)合這個項目去介紹。如果大家覺得這篇文章有不妥的地方,還請指出。

這篇文章的目的是解決我們在開發(fā)中會遇到的問題,不是一篇基礎(chǔ)教程,還請諒解。

項目目錄

我將根據(jù)這個目錄結(jié)構(gòu)進(jìn)行講解

dist:發(fā)布的文件目錄,即webpack編譯輸出的目錄

libs:放置公共的文件,如js、css、img、font等

mockServer:模擬后端服務(wù),即用webpack開發(fā)時模擬調(diào)用的后端服務(wù)(用nodejs服務(wù)模擬)

node_modules:項目依賴的包

src:資源文件,里面包含css、font、html、img、js

package.json:項目配置

webpack.config.js:webpack的配置文件

項目的使用

建議先運(yùn)行一下這個項目,有一個大致的了解,再往下閱讀。使用說明:

首先克隆一份到你的本地
$ git clone https://github.com/huangshuwei/webpackForSPA.git

然后 cd 到 ‘webpackForSPA’目錄下
$ cd webpackForSPA

接著你可以運(yùn)行不同的命令查看結(jié)果

發(fā)布模式:
$ npm run build

開發(fā)模式:
$ npm run dev

熱更新模式
$ npm run dev-hrm

如果使用了熱更新模式,并且想要結(jié)合后端服務(wù)形式運(yùn)行,那么cd 到‘mockServer’目錄下,并執(zhí)行node 服務(wù):
$ cd mockServer

$ node server.js

區(qū)分開發(fā)、熱更新、發(fā)布模式

一般開發(fā)時和發(fā)布時是不同的,比如開發(fā)時文件的訪問目錄包含‘dist’目錄,但是發(fā)布上線時,一般會把‘dist’文件夾去掉。
當(dāng)然還有其他的一些細(xì)節(jié)不同。

開發(fā)模式:

能看到webpack編譯輸出的文件

js、css、html文件不需要壓縮

可以正確的運(yùn)行編譯輸出后的文件

這種模式一般只是用來看webpack編譯輸出后的文件是否正確

熱更新模式:

看不到webpack編譯輸出的文件

js、css、html文件不需要壓縮

更改完文件后無需重新編譯并自動刷新瀏覽器

可以結(jié)合后端服務(wù)開發(fā),避過瀏覽器同源策略,如結(jié)合java、.net服務(wù)等

發(fā)布模式:

能看到webpack編譯輸出的文件

js、css、html文件壓縮

文件的層級目錄不需要包含‘dist’目錄

我區(qū)分開發(fā)、熱更新、發(fā)布模式是通過配置‘package.json’文件的運(yùn)行命令,有些人是通過創(chuàng)建多個不同的webpack的配置文件來達(dá)到想要的效果。

像這個項目就是使用了多個webpack的配置文件。

配置命令

這是在 package.json 文件中配置的

// package.json 文件
...
"scripts": {
    "build": "webpack  --profile --progress --colors --display-error-details",
    "dev": "webpack  --display-modules --profile --progress --colors --display-error-details",
    "dev-hrm": "webpack-dev-server --config"
  },
...

color 輸出結(jié)果帶彩色,比如:會用紅色顯示耗時較長的步驟

profile 輸出性能數(shù)據(jù),可以看到每一步的耗時

progress 輸出當(dāng)前編譯的進(jìn)度,以百分比的形式呈現(xiàn)

display-modules 默認(rèn)情況下 node_modules 下的模塊會被隱藏,加上這個參數(shù)可以顯示這些被隱藏的模塊

display-error-details 輸出詳細(xì)的錯誤信息

webpack-dev-server 將會開啟熱更新

更多請參考官網(wǎng) cli

配置好了package.json文件,我們就可以這樣運(yùn)行

// 開發(fā)模式
npm run dev

// 熱更新模式
npm run dev-hrm

// 發(fā)布模式
npm run build
配置變量標(biāo)識

配置完了命令,當(dāng)我們運(yùn)行不同的命令時,我們可以通過‘process.env.npm_lifecycle_event’去獲取當(dāng)前運(yùn)行的命令,根據(jù)不同的命令,我們可以按照自己的需要做相應(yīng)的處理。比如開發(fā)模式時,允許開啟調(diào)試,靜態(tài)資源不要壓縮;發(fā)布模式時,不允許調(diào)試,靜態(tài)資源要壓縮。具體如下:

// webpack.config.js

// 獲取當(dāng)前運(yùn)行的模式
var currentTarget = process.env.npm_lifecycle_event;

var debug,          // 是否是調(diào)試
    devServer,      // 是否是熱更新模式
    minimize;       // 是否需要壓縮

if (currentTarget == "build") { // 發(fā)布模式

    debug = false, devServer = false, minimize = true;
    
} else if (currentTarget == "dev") { // 開發(fā)模式

    debug = true, devServer = false, minimize = false;
    
} else if (currentTarget == "dev-hrm") { // 熱更新模式

    debug = true, devServer = true, minimize = false;
}
基礎(chǔ)配置 配置路徑

為了方便我們頻繁使用路徑,如下配置

// webpack.config.js
var PATHS = {
    // 發(fā)布目錄
    publicPath: debug ? "/webpackForSPA/dist/" : "/webpackForSPA/",

    // 公共資源目錄
    libsPath: path.resolve(process.cwd(), "./libs"),
    
    // src 資源目錄
    srcPath: path.resolve(process.cwd(), "src"),
}
配置別名

webpack的別名的目的就是簡化我們的操作,引用資源時直接使用別名即可(和 seajs 里的別名用法一樣)。配置如下:

// webpack.config.js
...
resolve:{
     alias: {
        // js
        jquery: path.join(PATHS.libsPath, "js/jquery/jquery"),
        underscore: path.join(PATHS.libsPath, "js/underscore/underscore.js"),

        // css
        bootstrapcss: path.join(PATHS.libsPath, "css/bootstrap/bootstrap-3.3.5.css"),
        indexcss: path.join(PATHS.srcPath, "css/index.css"),
    }
}
...
配置webpack編譯入口
// webpack.config.js
...
entry:{
    // 入口 js
    index: "./src/js/index.js",
    // 公共js包含的文件
    common: [
        path.join(PATHS.libsPath, "js/jquery/jquery.js"),
        path.join(PATHS.libsPath, "js/underscore/underscore.js")
    ],
}
...
配置webpack編譯輸出
// webpack.config.js
...
output:{
    // 輸出目錄
    path: path.join(__dirname, "dist"),

    // 發(fā)布后,資源的引用目錄
    publicPath: PATHS.publicPath,

    // 文件名稱
    filename: "js/[name].js",

    // 按需加載模塊時輸出的文件名稱
    chunkFilename: "js/[name].js"
}
...
提取css到多帶帶的文件

當(dāng)我們在js文件中通過require("")引用js時,webpack 默認(rèn)會將css文件與當(dāng)前js文件打包一起,但是這種方式會阻塞頁面的加載,因為css的執(zhí)行要等待js文件加載進(jìn)來。所以我們會把css從js文件中提取出來,放到一個多帶帶的css文件中。這時我們要使用webpack的插件:extract-text-webpack-plugin,配置如下:

引入插件

// webpack.config.js
var ExtractTextPlugin = require("extract-text-webpack-plugin");

配置 loader

// webpack.config.js
...
loaders: [
    {
        test: /.css$/,
        loader: ExtractTextPlugin.extract("style-loader", "css-loader!postcss-loader")
    },
    ...
]
...

配置 plugins

// webpack.config.js
...
plugins:[
    new ExtractTextPlugin("css/[name].css", {allChunks: true}),
    ...
]
...
公共js打包

項目中,我們通常會有公共的js,比如 jquery、bootstrap、underscore 等,那么這時候我們需要將這些公共的js多帶帶打包。這時我們需要用webpack自帶的插件:

// webpack.config.js
...
plugins:[
    // 會把 ‘entry’ 定義的 common 對應(yīng)的兩個js 打包為 ‘common.js’
    new webpack.optimize.CommonsChunkPlugin("common", "js/[name].js", Infinity),
]
...
資源添加版本號

項目上線后,資源的版本號十分重要。資源沒有版本號,即使重新發(fā)布,客戶端瀏覽器可能會把老的資源緩存下來,導(dǎo)致無法下載最新的資源。webpack 支持給資源添加版本號,不僅僅是js、css,甚至font、img都可以添加版本號。我們可以通過webpack中的‘chunkhash’來解決。

首先要了解下webpack 中 [hash]、[chunkhash]、[chunkhash:8]的區(qū)別。

[hash]:webpack編譯會產(chǎn)生一個hash值

[chunkhash]:每個模塊的hash值

[chunkhash:8]:取[chunkhash]的前8位

推薦發(fā)布模式使用版本號,其他模式無需使用,熱更新模式不支持‘chunkhash’,但是支持‘hash’

資源加版本號,那么我們的輸出的部分都要做改動,并且要區(qū)分當(dāng)前的命令模式,如下:

// webpack.config.js
...
output:{
    // 輸出目錄
    path: path.join(__dirname, "dist"),

    // 發(fā)布后,資源的引用目錄
    publicPath: PATHS.publicPath,

    // 文件名稱
    filename: devServer ? "js/[name].js" : "js/[name]-[chunkhash:8].js",

    // 按需加載模塊時輸出的文件名稱
    chunkFilename: devServer ? "js/[name].js" : "js/[name]-[chunkhash:8].js"
}
...

輸出公共js的地方也要改動:

// webpack.config.js
...
plugins:[
    // 會把 ‘entry’ 定義的 common 對應(yīng)的兩個js 打包為 ‘common.js’
    new webpack.optimize.CommonsChunkPlugin("common", "" + (devServer ? "js/[name].js" : "js/[name]-[chunkhash:8].js"), Infinity),
]
...
頁面自動引入含有版本號的文件

有個版本號后,我們考慮如何通過html引用這些含有版本號的js、css、font、img。webpack每次編譯后的資源 chunkhash 會隨著內(nèi)容的變化而變化,所以我們不可能每次都手動的更改html這些資源的引用路徑。這時我們要用到webpack的插件:html-webpack-plugin。這個插件的目的是生成html,也可以根據(jù)模板生成html,當(dāng)然還有其他的功能,具體看插件介紹。下面是的配置:

引入插件

// webpack.config.js
var HtmlWebpackPlugin = require("html-webpack-plugin");

配置 plugins,生成需要的html

// webpack.config.js
...
plugins:[
    new HtmlWebpackPlugin({
        filename: "index.html",
        template: __dirname + "/src/index.html",
        inject: "true"
    }),
    new HtmlWebpackPlugin({
        filename: "html/hrm.html",
        template: __dirname + "/src/html/hrm.html",
        inject: false,
    }),
    new HtmlWebpackPlugin({
        filename: "html/home.html",
        template: __dirname + "/src/html/home.html",
        inject: false,
    }),
]
...

我們前面說過,webpack 默認(rèn)只識別 js 文件,所以對于html也要使用對應(yīng)的loader:

// webpack.config.js
...
loaders:[
     {test: /.html$/,loader: "html"},
]
...
引用圖片和字體

引用圖片和字體,需要對應(yīng)的loader,并且可以設(shè)置這些資源大小的臨界值,當(dāng)小于臨界值的時候,字體或者圖片文件會以base64的形式在html引用,否則則是以資源路徑的形式引用。如下:

// webpack.config.js

// 圖片 loader
{
    test: /.(png|gif|jpe?g)$/,
    loader: "url-loader",
    query: {
        /*
         *  limit=10000 : 10kb
         *  圖片大小小于10kb 采用內(nèi)聯(lián)的形式,否則輸出圖片
         * */
        limit: 10000,
        name: "/img/[name]-[hash:8].[ext]"
    }
},

// 字體loader
{
    test: /.(eot|woff|woff2|ttf|svg)$/,
    loader: "url-loader",
    query: {
        limit: 5000,
        name: "/font/[name]-[hash:8].[ext]"
    }
},
資源文件的壓縮

js、css、html的壓縮是少不了的,webpack 自帶了壓縮插件,如果某些對象名稱不想被壓縮,可以排除不想要壓縮的對象名稱。配置如下:

// webpack.config.js
...
plugins:[
    new webpack.optimize.UglifyJsPlugin({ 
            mangle: { // 排除不想要壓縮的對象名稱
                except: ["$super", "$", "exports", "require", "module", "_"]
            },
            compress: {
                warnings: false
            },
            output: {
                comments: false,
            }
        })
]
...
使用jquery、underscore

通過webpack編譯輸出后的項目中,雖然頁面已經(jīng)引用了jquery、underscore,但是還是無法直接使用‘$’、‘_’對象,我們可以這樣:

var $ = require("jquery");
var _ =  require("underscore");

但是這樣實在不方便,如果我們就是要使用‘$’、‘_’對象直接操作,webpack 內(nèi)置的插件可以幫我們解決。具體如下:

// webpack.config.js
new webpack.ProvidePlugin({
        $: "jquery",
        jQuery: "jquery",
        "window.jQuery": "jquery",
        "_": "underscore",
    }),
代碼分割,按需加載

在單頁面應(yīng)用中,當(dāng)我們加載其他的模板文件時,想要引用這個模板文件對應(yīng)的js。如果我們通過這種方式require(),那么webpack會將這個模板文件對應(yīng)的js也會和當(dāng)前js打包成一個js。如果項目比較大,那么js文件也將越來越大。我們希望的是加載模板文件的時候動態(tài)的引用這個模板文件對應(yīng)的js。那么我們可以通過 require.ensure()的方式。

比如現(xiàn)在有兩個導(dǎo)航菜單:

我們給這兩個菜單綁定點擊事件,當(dāng)點擊‘home’時引用對應(yīng)的‘home.js’;當(dāng)點擊‘HRM’時引用對應(yīng)的‘hrm.js’,那么大致可以這樣:

function loadJs(jsPath) {
    var currentMod;
    if (jsPath === "./home") {
        require.ensure([], function (require) {
            currentMod = require("./home");
        }, "home");
    }
    else if (jsPath === "./hrm") {
        require.ensure([], function (require) {
            currentMod = require("./hrm");
        }, "hrm");
    }
}
全局環(huán)境變量

有時我們只有在開發(fā)過程中,才想輸出log日志??梢杂靡韵聎ebpack內(nèi)置的插件解決:

// webpack.config.js
...
plugins:[
      new webpack.DefinePlugin({
        // 全局debug標(biāo)識
        __DEV__: debug,
    }),

]
...

這時代碼中就可以這么寫了:

if (__DEV__) {
    console.log("debug 模式");
}
清空發(fā)布目錄

發(fā)布前清空發(fā)布目錄是有必要的,我們可以通過‘clean-webpack-plugin’插件解決:

引入插件:

// webpack.config.js
var CleanWebpackPlugin = require("clean-webpack-plugin");

配置plugins:

// webpack.config.js
...
plugins:[
    new CleanWebpackPlugin(["dist"], {
        root: "", // An absolute path for the root  of webpack.config.js
        verbose: true,// Write logs to console.
        dry: false // Do not delete anything, good for testing.
    }),
]
...
熱更新結(jié)合后端服務(wù) 熱更新

熱更新可以在你代碼改變的時候即時編譯輸出,不用每次都要從都重新編譯一遍,并且除了第一次編譯比較慢,后面的編譯都是增量編譯,速度很快。有了這個功能,我們就不需要,每次都從頭編譯一次了。配置如下:

// webpack.config.js
...
plugins: [
        // Enable multi-pass compilation for enhanced performance
        // in larger projects. Good default.
        new webpack.HotModuleReplacementPlugin({
            multiStep: true
        }),
],
devServer: {
        // Enable history API fallback so HTML5 History API based
        // routing works. This is a good default that will come
        // in handy in more complicated setups.
        historyApiFallback: true,

        // Unlike the cli flag, this doesn"t set
        // HotModuleReplacementPlugin!
        hot: true,
        inline: true,

        // Display only errors to reduce the amount of output.
        stats: "errors-only",

        host: "localhost", // Defaults to `localhost`   process.env.HOST
        port: "8080",  // Defaults to 8080   process.env.PORT
}
...

這時我們只要打開瀏覽器,輸入:localhost:8080/ 就能看到結(jié)果,并且在你修改某些源文件后,瀏覽器會自動刷新,就能看到webpack 即時編譯輸出的結(jié)果,而不需要重新編譯。

結(jié)合后端服務(wù)

我們在使用webpack開發(fā)時難免要結(jié)合后端服務(wù)開發(fā),比如我們用webstorm 編譯器開發(fā)項目,需要調(diào)用java的服務(wù),由于有同源策略問題,這時我們會收到相關(guān)報錯信息。這時我們可以通過代理的方式繞過同源策略。
這里我用nodejs 模擬一個后端服務(wù),如下:

// ~/mockServer/server.js

var http = require("http");

var content = "▍if you see that,It means you have get the correct data by backend server(mock data by nodejs server)!";

var srv = http.createServer(function (req, res) {
    res.writeHead(200, {"Content-Type": "application/text"});
    res.end(content);
});

srv.listen(8888, function() {
    console.log("listening on localhost:8888");
});

接下來我們需要這樣配置去調(diào)用這個nodejs 的服務(wù)。
首先將熱更新配置的代碼修改為:

// webpack.config.js
...
plugins: [
        // Enable multi-pass compilation for enhanced performance
        // in larger projects. Good default.
        new webpack.HotModuleReplacementPlugin({
            multiStep: true
        }),
],
devServer: {
        // Enable history API fallback so HTML5 History API based
        // routing works. This is a good default that will come
        // in handy in more complicated setups.
        historyApiFallback: true,

        // Unlike the cli flag, this doesn"t set
        // HotModuleReplacementPlugin!
        hot: true,
        inline: true,

        // Display only errors to reduce the amount of output.
        stats: "errors-only",

        host: "localhost", // Defaults to `localhost`   process.env.HOST
        port: "8080",  // Defaults to 8080   process.env.PORT
        proxy: {
                "/devApi/*": {
                    target: "http://localhost:8888/",
                    secure: true,
                    /*
                     * rewrite 的方式擴(kuò)展性更強(qiáng),不限制服務(wù)的名稱
                     * */
                    rewrite: function (req) {
                        req.url = req.url.replace(/^/devApi/, "");
                    }
                }
        }
}
...

然后配置一個全局的環(huán)境變量,通過DefinePlugin

// webpack.config.js
...
plugins: [
 new webpack.DefinePlugin({
        __DEVAPI__: devServer ? "/devApi/" : """",
    }),
]
...

最后在調(diào)用服務(wù)的地方,只需要在調(diào)用地址前添加 __DEVAPI__全局環(huán)境變量即可,如:

$.ajax({
        url: __DEVAPI__ + "http://localhost:8888/",
        data: {},
        type: "get",
        dataType: "text",
        success: function (text) {}
    })

這樣在熱更新的模式下,當(dāng)有__DEVAPI__ 的地方就會自動識別為/devApi/,而這里會通過代理處理幫你重寫掉,繞過同源策略。

自動打開瀏覽器

雖然以上的工作幾乎已經(jīng)滿足我們對webpack的要求了,但是我們還想懶一點,想在熱更新模式下,編譯完成后自動打開瀏覽器。那么我們可以通過這個插件open-browser-webpack-plugin解決:

引用插件

// webpack.config.js
var OpenBrowserPlugin = require("open-browser-webpack-plugin");

配置插件,這個配置要根據(jù)項目的具體情況去配置:

// webpack.config.js
...
plugins: [
 new OpenBrowserPlugin({url: "http://localhost:8080" + PATHS.publicPath + "index.html"})
]
...
總結(jié)

以上就是這篇文章的主要內(nèi)容,希望通過這篇文章能夠給大家?guī)硪恍﹩l(fā)。如果有覺得哪里不對,或者不合理的地方,歡迎指出。其實webpack還有一個關(guān)于版本號的bug,不知道是不是有人解決了,如果有人已經(jīng)解決了,還請分享。

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

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/79887.html

相關(guān)文章

  • Webpack實戰(zhàn) - 使用動態(tài) entry 改善調(diào)試體驗

    摘要:本文相關(guān)代碼已經(jīng)存放在,可自行下載使用多入口復(fù)習(xí)的優(yōu)勢不言而喻,因此在實際應(yīng)用中我們也常常使用它調(diào)試多入口應(yīng)用,所謂多入口是指多個頁面會使用多個入口文件,在官方教程介紹了如何配置這里指定了個入口文件,打包之后分別會在文件夾中生成個打包之后 本文相關(guān)代碼已經(jīng)存放在 dynamic-entry,可自行下載使用 0. 多入口 (復(fù)習(xí)) webpack 的優(yōu)勢不言而喻,因此在實際應(yīng)用中我們也常...

    oogh 評論0 收藏0
  • 基于 Webpack 4 多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn)

    摘要:原作者原鏈接基于多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn)法律聲明警告本作品遵循署名非商業(yè)性使用禁止演繹未本地化版本協(xié)議發(fā)布。這是什么背景現(xiàn)代化的前端項目中很多都使用了客戶端渲染的單頁面應(yīng)用。 原作者:@LinuxerPHL原鏈接:基于 Webpack 4 多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn) 法律聲明 警告:本作品遵循 署名-非商業(yè)性使用-禁止演繹3.0 未本地化版本(CC BY-...

    big_cat 評論0 收藏0
  • 基于 Webpack 4 多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn)

    摘要:原作者原博文地址基于多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn)法律聲明警告本作品遵循署名非商業(yè)性使用禁止演繹未本地化版本協(xié)議發(fā)布。這是什么背景現(xiàn)代化的前端項目中很多都使用了客戶端渲染的單頁面應(yīng)用。 原作者:@LinuxerPHL原博文地址: 基于 Webpack 4 多入口生成模板用于服務(wù)端渲染的方案及實戰(zhàn) 法律聲明 警告:本作品遵循 署名-非商業(yè)性使用-禁止演繹3.0 未本地化版本(...

    Lavender 評論0 收藏0
  • webpack4 配置解析和實戰(zhàn)

    摘要:特性比較熱門的兩大特性,零配置和速度快號稱提速上限一般情況下,相比于低版本,場景下第三方依賴打包速度和場景下本地服務(wù)首次啟動速度都得到顯著提升零配置通過指定當(dāng)前場景為開發(fā)模式還是生產(chǎn)模式,自動設(shè)置好當(dāng)前場景的默認(rèn)配置,用戶即可馬上使用,不需 webpack4特性 webpack4比較熱門的兩大特性,零配置和速度快(號稱提速上限98%) 一般情況下,webpack4相比于低版本,prod...

    王笑朝 評論0 收藏0
  • webpack 使用優(yōu)化(一)

    摘要:原因就是默認(rèn)會把最重要的東西放到公共里,這里面包含啟動應(yīng)用程序的依賴項模塊與模塊的依賴關(guān)系以及文件的版本號等信息。 之前寫了一篇關(guān)于webpack 如何使用的文章:webpack 單頁面應(yīng)用實戰(zhàn),并且寫了一個 單頁面應(yīng)用的小項目 放到了github上。正巧公司前段時間用webpack 做了一個項目,項目不大,是基于單頁面應(yīng)用的。但是上線后才發(fā)現(xiàn)了一些問題,原來還是有一些要優(yōu)化改進(jìn)的地方...

    Caicloud 評論0 收藏0

發(fā)表評論

0條評論

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