【腾讯Bugly干货分享】基于 Webpack & Vue & Vue-Router 的 SPA 初体验

本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57d13a57132ff21c38110186

导语

最近这几年的前端圈子,由于戏台一般精彩纷呈,从 MVC 到 MVVM,你刚唱罢我登场。 backbone,angularjs 已成昨日黄花,reactjs 如日中天,同时另一更轻量的 vue 发展势头更猛,尤其是即将 release 的2.0版本,号称兼具了 angularjs 和 reactjs 的两者优点。不过现在的官方版本还是1.0 ,下面就是基于1.0版本的初体验。

1. 为什么要 SPA?

SPA: 就是俗称的单页应用(Single Page Web Application)。

在移动端,特别是 hybrid 方式的H5应用中,性能问题一直是痛点。 使用 SPA,没有页面切换,就没有白屏阻塞,可以大大提高 H5 的性能,达到接近原生的流畅体验。

2. 为什么选择 vue?

在选择 vue 之前,使用 reactjs 也做过一个小 Demo,虽然两者都是面向组件的开发思路,但是 reactjs 的全家桶方式,实在太过强势,而自己定义的 JSX 规范,揉和在 JS 的组件框架里,导致如果后期发生页面改版工作,工作量将会巨大。

vue 相对来说,就轻量的多,他的view层,还是原来的 dom 结构,除了一些自定义的 vue 指令作为自定义标签以外,只要学会写组件就可以了,学习成本也比较低。

3. 环境配置

初始化工程,需要 node 环境使用 npm 安装相应的依赖包。

先创建一个测试目录,在里面依次输入以下命令。

 //初始化package.json
 npm init

 //安装vue的依赖
 npm install vue --save
 npm install vue-router --save

 //安装webpack的开发依赖
 npm install webpack --save-dev

 //安装babel的ES6 Loader 的开发依赖
 npm install babel --save-dev
 npm install babel-core --save-dev
 npm install babel-loader --save-dev
 npm install babel-preset-es2015 --save-dev

 //安装html loacer 的开发依赖
 npm install html-loader --save-dev

4. 目录结构

src 为开发目录,其中 components 为组件子目录,templates 为模板子目录。

dist 为构建出的文件目录。

index.html 为入口文件。

package.json 为项目描述文件,是刚才 npm init 所建立。

webpack.config.js 是 webpack 的构建配置文件

5. Webpack 配置

下面是 webpack 的配置文件,如何使用 webpack,请移步 webpack 的官网。

 var webpack= require("webpack");

 module.exports={
     entry:{
         bundle:[ "./src/app.js"]
     },
     output:{
         path:__dirname,
         publicPath:"/",
         filename:"dist/[name].js"
     },
     module:{
         loaders:[
             {test: /\.html$/, loaders: [‘html‘]},
             {test: /(\.js)$/, loader:["babel"] ,exclude:/node_modules/,
              query:{
                      presets:["es2015"]
              }
             }
         ]
     },
     resolve:{
     },
     plugins:[
          /*
          new webpack.optimize.UglifyJsPlugin({
             compress: {
                 warnings: false
             }
         })
                */
     ]
 }

6. 入口文件

index.html

 <!doctype html>
 <html lang="en">
 <head>
     <meta charset="UTF-8">
     <title>Vue Router Demo</title>
 </head>
 <body>
    <div id="app">
       <router-view></router-view>
     </div>
     <script src="dist/bundle.js"></script>
 </body>
 </html>

其中 id 为 app 的 div 是页面容器,其中的 router-view 会由 vue-router 去渲染组件,讲结果挂载到这个 div 上。

app.js

 var Vue = require(‘vue‘);
 var VueRouter = require(‘vue-router‘);

 Vue.use(VueRouter);
 Vue.config.debug = true;
 Vue.config.delimiters = [‘${‘, ‘}‘]; // 把默认的{{ }} 改成ES6的模板字符串 ${ }
 Vue.config.devtools = true;

 var App = Vue.extend({});
 var router = new VueRouter({});

 router.map(require(‘./routes‘));
 router.start(App, ‘#app‘);
 router.go({"path":"/"});

这是 vue 路由的配置。 其中由于习惯问题,我把 vue 默认的{{ }} 改成了的 ${ } ,总感觉这样看模板,才顺眼一些。

routes.js

 module.exports = {
   ‘/‘: {
     component: require(‘./components/index‘)
   },
    ‘/list‘: {
     component: require(‘./components/list‘)
   },
   ‘*‘: {
     component: require(‘./components/notFound‘)
   }
 }

7. 第一个组件

components/index.js

 module.exports = {
   template: require(‘../templates/index.html‘),

   ready: function () {
   }
 };

templates/index.html

 <h1>Index</h1>
 <hr/>
 <p>Hello World Index!</p>

执行 webpack 构建命令

浏览器中访问:

查看 bundle 源码:

发现 template 模板文件,已经被 webpack 打成字符串了。这其中,其实是 webpack 的 html-loader 起的作用

8. 组件之间跳转

修改刚才的 index 组件,增加一个跳转链接,不用 href 了,要用 vue 的指令 v-link。

 <h1>Index</h1>
 <hr/>
 <p>Hello World Index!</p>
 <p><a v-link="{path:‘/list‘}" >List Page</a></p>

添加 list 组件

components/list.js

 module.exports = {
   template: require(‘../templates/list.html‘),

   data:function(){
       return {items:[{"id":1,"name":"hello11"},{"id":2,"name":"hello22"}]};
   },
   ready: function () {
   }
 };

templates/list.html

 <h1>List</h1>
 <hr/>

 <p>Hello List Page!</p>
 <ul>
     <li v-for="(index,item) in items">
          ${item.id} : ${item.name}
     </li>
 </ul>

v-for 也是 vue 的默认指令,是用来循环数据列表的。

现在开始执行 webpack —watch 命令进行监听,这样就不用每次敲 webpack 命令了。只要开发者每次修改 js 点了保存,webpack 都会自动构建最新的 bundle 文件。

浏览器里试试看:

index 页

点击 List Page 跳转到 list 页

Bingo! 单页面两个组件之间跳转切换成功!

9. 组件生命周期

修改 componets/list.js

 module.exports = {
   template: require(‘../templates/list.html‘),

   data:function(){
       return {items:[{"id":1,"name":"hello11"},{"id":2,"name":"hello22"}]};
   },

   //在实例开始初始化时同步调用。此时数据观测、事件和 watcher 都尚未初始化
   init:function(){
       console.log("init..");
   },

   //在实例创建之后同步调用。此时实例已经结束解析选项,这意味着已建立:数据绑定,计算属性,方法,watcher/事件回调。但是还没有开始 DOM 编译,$el 还不存在。
   created:function(){
       console.log("created..");
   },

   //在编译开始前调用。
   beforeCompile:function(){
        console.log("beforeCompile..");
   },

   //在编译结束后调用。此时所有的指令已生效,因而数据的变化将触发 DOM 更新。但是不担保 $el 已插入文档。
   compiled:function(){
        console.log("compiled..");
   },

    //在编译结束和 $el 第一次插入文档之后调用,如在第一次 attached 钩子之后调用。注意必须是由 Vue 插入(如 vm.$appendTo() 等方法或指令更新)才触发 ready 钩子。
   ready: function () {
     console.log("ready..");

   },

   //在 vm.$el 插入 DOM 时调用。必须是由指令或实例方法(如 $appendTo())插入,直接操作 vm.$el 不会 触发这个钩子。
   attached:function(){
        console.log("attached..");
   },

   //在 vm.$el 从 DOM 中删除时调用。必须是由指令或实例方法删除,直接操作 vm.$el 不会 触发这个钩子。
   detached:function(){
        console.log("detached..");
   },

   //在开始销毁实例时调用。此时实例仍然有功能。
   beforeDestroy:function(){
        console.log("beforeDestroy..");
   },

   //在实例被销毁之后调用。此时所有的绑定和实例的指令已经解绑,所有的子实例也已经被销毁。如果有离开过渡,destroyed 钩子在过渡完成之后调用。
   destroyed:function(){
        console.log("destroyed..");
   }

 };

在浏览器里执行了看看:

首次进入 List 页面的执行顺序如下:

此时点一下浏览器的后退,List Component 会被销毁,执行顺序如下:

这是官方的生命周期的图:

10. 父组件与子组件

在很多情况下,组件是有父子关系的,比如 list 列表组件有个子组件 item

components/item.js

 module.exports = {
   template: require(‘../templates/item.html‘),

   props:["id","name"],

   ready: function () {

   }
 };

templates/item.html

 <p>我是subitem: ${id} - ${name}</p>

修改 list 组件,添加 item 的引用

components/list.js

 //引用item组件
 import item from "./item";

 module.exports = {
   template: require(‘../templates/list.html‘),

   data:function(){
       return {items:[{"id":1,"name":"hello11"},{"id":2,"name":"hello22"}]};
   },

  //定义item组件为子组件
   components:{
      "item":item
   },

   ready: function () {
   }

 };

templates/list.html

 <h1>List</h1>
 <hr/>
 <p>Hello List Page!</p>
 <ul>
     <li v-for="(index,item) in items">
         <!--使用item子组件,同时把id,name使用props传值给item子组件-->
         <item v-bind:id="item.id" v-bind:name="item.name"></item>
     </li>
 </ul>

浏览器里试试看:

子组件成功被调用了

11. 组件跳转传参

组件之间的跳转传参,也是一种非常常见的情况。下面为列表页,增加跳转到详情页的跳转,并传参 id 给详情页

修改路由 routes.js

 module.exports = {

   ‘/‘: {
     component: require(‘./components/index‘)
   },
    ‘/list‘: {
     component: require(‘./components/list‘)
   },

   //增加详情页的跳转路由,并在路径上加上id传参,具名为name:show
    ‘/show/:id‘: {
       name:"show",
       component: require(‘./components/show‘)
     },
   ‘*‘: {
     component: require(‘./components/notFound‘)
   }
 }

添加组件 show

components/show.js

 module.exports = {
   template: require(‘../templates/show.html‘),

   data:function(){
       return {};
   },

   created:function(){
        //获取params的参数ID
        var id=this.$route.params.id;

        //根据获取的参数ID,返回不同的data对象(真实业务中,这里应该是Ajax获取数据)
        if (id==1){
             this.$data={"id":id,"name":"hello111","age":24};
        }else{
         this.$data={"id":id,"name":"hello222","age":28};
        }
   },

   ready: function () {
       console.log(this.$data);
   }

 };

templates/show.html

 <h1>Show</h1>
 <hr/>

 <p>Hello show page!</p>

 <p>id:${id}</p>
 <p>name:${name}</p>
 <p>age:${age}</p>

修改 templates/item.html

 <p>我是subitem: <a v-link="{name:‘show‘,params: { ‘id‘: id } }"> ${id} : ${name}</a> </p>

这里 name:’show’ 表示具名路由路径,params 就是传参。

继续浏览器里点到详情页试试:

点击“hello11”,跳转到详情页:

传参逻辑成功。

12. 嵌套路由

仅有路由跳转是远远不够的,很多情况下,我们还有同一个页面上,多标签页的切换,在 vue 中,用嵌套路由,也可以非常方便的实现。

添加两个小组件

components/tab1.js

 module.exports = {
   template: "<p>Tab1 content</p>"
 };

components/tab2.js

 module.exports = {
   template: "<p>Tab2 content</p>"
 };

修改 components/index.js 组件,挂载这两个子组件

 import tab1 from "./tab1";
 import tab2 from "./tab2";

 module.exports = {
   template: require(‘../templates/index.html‘),

   components:{
      "tab1":tab1,
      "tab2":tab2
   },

   ready: function () {

   }
 };

在路由里加上子路由

 module.exports = {

   ‘/‘: {
     component: require(‘./components/index‘),

     //子路由
     subRoutes:{
       "/tab1":{
           component:require(‘./components/tab1‘)
       },
       "/tab2":{
           component:require(‘./components/tab2‘)
       }
     }
   },

    ‘/list‘: {
     component: require(‘./components/list‘)
   },

    ‘/show/:id‘: {
       name:"show",
       component: require(‘./components/show‘)
     },

   ‘*‘: {
     component: require(‘./components/notFound‘)
   }

 }

好了,在浏览器里试一下:

初始状态:

点了 tab1,tab2:

Tab 切换没问题,可是,初始状态显示是空的,能不能默认显示 Tab1 Content 呢?很简单,调整下路由就可以了:

 module.exports = {

   ‘/‘: {
     component: require(‘./components/index‘),

     //子路由
     subRoutes:{
      //默认显示Tab1
       "/":{
           component:require(‘./components/tab1‘)
       },
       "/tab1":{
           component:require(‘./components/tab1‘)
       },
       "/tab2":{
           component:require(‘./components/tab2‘)
       }
     }
   }
 }

13. 业界 vue 使用案例

小米移动官网:http://m.mi.com/1/#/index

饿了吗招聘:https://jobs-mobile.ele.me/#!/



更多精彩内容欢迎关注bugly的微信公众账号:

腾讯 Bugly是一款专为移动开发者打造的质量监控工具,帮助开发者快速,便捷的定位线上应用崩溃的情况以及解决方案。智能合并功能帮助开发同学把每天上报的数千条 Crash 根据根因合并分类,每日日报会列出影响用户数最多的崩溃,精准定位功能帮助开发同学定位到出问题的代码行,实时上报可以在发布后快速的了解应用的质量情况,适配最新的 iOS, Android 官方操作系统,鹅厂的工程师都在使用,快来加入我们吧!

时间: 2024-10-07 06:04:47

【腾讯Bugly干货分享】基于 Webpack & Vue & Vue-Router 的 SPA 初体验的相关文章

【腾讯Bugly干货分享】彻底弄懂 Http 缓存机制 - 基于缓存策略三要素分解法

本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/qOMO0LIdA47j3RjhbCWUEQ 作者:李志刚 导语 Http 缓存机制作为 web 性能优化的重要手段,对从事 Web 开发的小伙伴们来说是必须要掌握的知识,但最近我遇到了几个缓存头设置相关的题目,发现有好几道题答错了,有的甚至在知道了正确答案后依然不明白其原因,可谓相当的郁闷呢!!为了确认下是否只是自己理解不深,我特意请教了其他几位小伙

【腾讯Bugly干货分享】深入源码探索 ReactNative 通信机制

Bugly 技术干货系列内容主要涉及移动开发方向,是由 Bugly 邀请腾讯内部各位技术大咖,通过日常工作经验的总结以及感悟撰写而成,内容均属原创,转载请标明出处. 本文从源码角度剖析 RNA 中 Java <> Js 的通信机制(基于最新的 RNA Release 20). 对于传统 Java<>Js 通信而言,Js 调用 Java 通不外乎 Jsbridge.onprompt.log 及 addjavascriptinterface 四种方式,在 Java 调用 Js 只有 l

【腾讯Bugly干货分享】JSPatch 成长之路

本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/579efa7083355a9a57a1ac5b Dev Club 是一个交流移动开发技术,结交朋友,扩展人脉的社群,成员都是经过审核的移动开发工程师.每周都会举行嘉宾分享,话题讨论等活动. 本期,我们邀请了腾讯WXG iOS开发工程师——bang 陈振焯,为大家分享<JSPatch成长之路>. 如何加入 Dev Club? 移动端开发经验 >= 2 年,微信扫描下方群管

【腾讯Bugly干货分享】经典随机Crash之一:线程安全

本文作者:鲁可--腾讯SNG专项测试组 测试工程师 背景 Android QQ 在2016下半年连着好几个版本二灰 Crash 率都很高,如果说有新需求,一灰的 Crash 率高,还能找点理由,可是开发童鞋解过一灰的 Crash 单后,为啥二灰还有这么高的 Crash 率,我们还有覆盖全 SNG.不少外 BG 明星产品的终端稳定性测试工具 NewMonkey 随身版(NewMonkey系腾讯内部研发的测试工具,外部app有兴趣请点击这里填问卷调查申请使用)每天都在跑,更何况大多 Top Cras

【腾讯Bugly干货分享】从0到1打造直播 App

本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/5811d42e7fd6ec467453bf58 作者:李智文 概要 分享内容: 互联网内容载体变迁历程,文字--图片/声音--视频--VR/AR----..从直播1.0秀场时代(YY),2.0游戏直播(斗鱼.虎牙.熊猫)到如今全民直播3.0泛生活娱乐时代(映客.花椒),国外直播app(Meerkat .Periscope),随着VA/AR/MR提出的沉浸式视听体验,直播4.0时

【腾讯Bugly干货分享】美团大众点评 Hybrid 化建设

本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:http://mp.weixin.qq.com/s/rNGD6SotKoO8frmxIU8-xw 本期 T 沙龙探讨了移动端热更新相关的话题.由于沙龙时间的限制,本期我们选取了美团的 Hybrid 化建设.去哪儿的跨平台 ListView 性能优化.微博 Android 端热更新踩过的坑话题.还期待热更新.热修复哪些话题?欢迎留言给我们.也欢迎报名参加 T 沙龙分享自己开发中的心得. Hybrid 是移动

【腾讯Bugly干货分享】TRIM:提升磁盘性能,缓解Android卡顿

Bugly 技术干货系列内容主要涉及移动开发方向,是由 Bugly 邀请腾讯内部各位技术大咖,通过日常工作经验的总结以及感悟撰写而成,内容均属原创,转载请标明出处.在业内,Android 手机一直有着“越用越慢”的口碑.根据第三方的调研数据显示,有77%的 Android 手机用户承认自己曾遭遇过手机变慢的影响.他们不明白为什么购买之初“如丝般顺滑”的 Android 手机,在使用不到一年之后都会“卡顿”得让人抓狂!根据我们初步的测试数据,手机长期所使用产生的磁盘碎片可以使得磁盘的写入效率下降为

【腾讯Bugly干货分享】微信读书iOS性能优化

本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/578c93ca9644bd524bfcabe8 "8小时内拼工作,8小时外拼成长"这是大家共同的理想.除了每天忙于工作外,我们都希望能更多地区吸收领域内的新知识与新技能,从而走向人生巅峰. Dev Club 是一个交流移动开发技术,结交朋友,扩展人脉的社群,成员都是经过审核的移动开发工程师.每周都会举行嘉宾分享,话题讨论等活动. 上一期我们邀请了腾讯SNG工程师&qu

【腾讯bugly干货分享】Android自绘动画实现与优化实战——以Tencent OS录音机波形动

前言 本文为腾讯bugly的原创内容,非经过本文作者同意禁止转载,原文地址为:http://bugly.qq.com/bbs/forum.php?mod=viewthread&tid=1180 我们所熟知的,Android 的图形绘制主要是基于 View 这个类实现. 每个 View 的绘制都需要经过 onMeasure.onLayout.onDraw 三步曲,分别对应到测量大小.布局.绘制. Android 系统为了简化线程开发,降低应用开发的难度,将这三个过程都放在应用的主线程(UI 线程)