vue异步组件 懒加载& webpack按需加载——性能优化

利用webpack对代码进行分割是懒加载的前提,懒加载就是异步调用组件,需要时候才下载(按需加载)。

为什么需要懒加载?
在单页应用中,如果没有应用懒加载,运用webpack打包后的文件将会异常的大,造成进入首页时,需要加载的内容过多,延时过长,不利于用户体验,而运用懒加载则可以将页面进行划分,需要的时候加载页面,可以有效的分担首页所承担的加载压力,减少首页加载用时。

vue开发过程中,我们会做出特别多特别多的组件,包括login,header,footer,main等等。
这样使整个网站看起来就十分的庞大,当我们在打开网页的时候,突然一下子把这些所有的组件加载上来,这么多的请求全部同时开始请求,势必会造成网页打开很慢,使客户得到的是非常差劲的体验。

因此,vue为我们专门设立了异步组件,通过异步组件,我们可以得到两点好处:
1、 用不到的组件不会加载,因此网页打开速度会很快,当你用到这个组件的时候,才会通过异步请求进行加载;
2、 缓存组件,通过异步加载的组件会缓存起来,当你下一次再用到这个组件时,丝毫不会有任何的疑迟,组件很快会从缓存中加载出来。

异步组件=原理同webpack的按需加载
好处:
1)按需加载,可以节省首次加载时间,提高速度,性能优化
2)第一次加载完成会缓存

异步组件的描述:

Vue允许将组件定义为一个异步解析(加载)组件定义的工厂函数,即Vue只在实际需要渲染组件时,才会触发调用工厂函数,并且将结果缓存起来,用于将来再次渲染。

!!!ps:要使用异步组件,一个比较推荐的方式就是配合webpack代码分离功能。

例子:(全局|局部)
法一:结合webpack

//1)全局:
 Vue.component('component-name',function(resolve){
 //require 语法告诉 webpack自动将编译后的代码分割成不同的块
 //这些块将通过 Ajax 请求自动下载
   require(['./my-async-componnet'],resolve)
 })
 //注册全局组件名,但只有一个名字,没有实体,相当于空的
//当需要这个组件时,调用上面的工厂函数,触发webpack的异步加载模块方法
//然后异步请求一个模块,请求成功后,这个模块内容即为组件实体部分,并对应地方渲染,加载内容也缓存下来。

//2)局部
 new Vue({
   components: {
        'component-name':function(resolve) {
           require(['./my-component'], resolve)
        }
   }
 })

法二:通过webpack2+es2015返回一个promise

//1)全局:
Vue.component('component-name',
 ()=> import('./my-async-componnet')
 )
//2) 局部:
 new Vue({
   components: {
       'component-name': () =>  import('./my-async-componnet')
   }
 })

高级异步组件(即处理加载状态)
工厂对象可以返回一个对象,对象里面的一些配置参数

    const asyncComponent = () => ({
    // 需要加载的组件 (应该是一个 `Promise` 对象)
    component: import('./my-async-componnet'),
    //异步加载时使用的组件(加载中的效果)
    loading: loadingComponent,
    //加载失败时使用的组件
    error: ErrorComponent,
    //展示加载中组件的延时时间,默认200毫秒
    delay: 200,
    //超时时间,超过该时间显示加载失败的组件
    timeout: 3000
    })

单文件组件+vue-router案例:
2个组件:first、second代表2个页面, 路由文件异步组件方式:

     //传统写法:import First from '@/components/First'
    方法一:
            const first = () => import("../components/first.vue")
            const second = () => import("../components/second.vue")
    方法二:
            const first = resolve => require.ensure([], () => resolve( require(../components/first.vue)),  'chunkname1')
            const second = resolve => require.ensure([], () => resolve( require(../components/second.vue)),  'chunkname2')
      //其余配置如以往一样
      const routes = [
          {
              path: '/',
              component: first
           },
            {
              path: '/',
              component: second
           }
      ]

webpack中利用require.ensure()实现按需加载

require.ensure()(webpack异步加载|代码分割)

  • 解释1:把js模块独立导出一个个js文件,然后使用这个模块时,webpack会构造这个script dom元素,加入到document.head中,由浏览器自动发起异步请求这个js文件,再写个回调,去定义得到这个js文件后的业务逻辑。
  • 解释2:webpack 在编译时,会静态地解析代码中的 require.ensure(),同时将模块添加到一个分开的 chunk 当中。这个新的 chunk 会被 webpack 通过 jsonp 来按需加载。

require.ensure这个函数是一个代码分离的分割线,表示 回调里面的require
是我们想要进行分割出去的,即require(’./baidumap.js’),把baidumap.js分割出去,形成一个webpack打包的单独js文件。当然ensure里面也是可以写一些同步的require的,如:

    var sync = require('syncdemo.js')   //下面ensure里面也用到
    btn.click(function() {
        require.ensure([], function() {
              var map = require('./map.js') //map.js放在我们当前目录下
              //这个不会独立出去,因为它已经加载到模块缓存中了
               var sync = require('syncdemo.js')  
    })
  })

===》ensure会把没有使用过的require资源进行独立分成成一个js文件

ensure语法及参数解释

require.ensure(dependencies: string[], callback: function(require),chunkName:string)

  • 第一个参数:默认[],放置当前这个模块所依赖的其他模块。比如:假设A 和 B都是异步的,B中需要A,则B下载之前,先要下载A :require.ensure([‘A.js’], function...),!!!!需要注意:webpack会把参数里面的依赖异步模块和当前的需要分离出去的异步模块给一起打包成同一个js文件,这里可能会出现一个重复打包的问题, 假设A 和 B都是异步的, ensure A 中依赖B,ensure B中 依赖A,那么会生成两个文件,都包含A和B模块。
  • 第二个参数:当所有依赖都加载完成后,webpack会执行这个回调函数。require对象的一个实现会作为一个参数传递给这个回调函数。因此,我们可以进一步require()依赖和其他模块提供下一步的执行。
  • 第三个参数,chunkname打包后代码块的名字:chunk提供给这个特定的require.ensure()的chunk的名称。通过提供 require.ensure() 不同执行点相同的名称,我们可以保证所有的依赖都会一起放进相同的 文件束(bundle)。

使用 vue-cli构建的项目,在 默认情况下 ,执行 npm run build 会将所有的js代码打包为一个整体。如果想要让build之后的代码更便于识别,可配置webpack代码,单独打包自己定义的名字(需要配置chunkFileName和publicPath)

  module.exports = {
        entry:  './src/main.js',
        output: {
              path: path.resolve(__dirname, './dist'),
              publicPath: '/dist/', //是按需加载单独打包出来的chunk是以publicPath为基准来存放的
              filename: 'build.js',
              chunkFilename: 'js/[name]-[chunkhash:8].js' //最终生成的路径和名字
          }
    }
打包结果.png
目录结果.png

参考地址:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342