vue构建单页面应用实战


Posted in Javascript onApril 10, 2017

1. 为什么要 SPA?

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

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

2. 为什么选择 vue?

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

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

 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 的构建配置文件

vue构建单页面应用实战 

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 构建命令

vue构建单页面应用实战 

浏览器中访问:

 vue构建单页面应用实战

查看 bundle 源码:

vue构建单页面应用实战 

发现 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 文件。

vue构建单页面应用实战 

浏览器里试试看:

index 页

 vue构建单页面应用实战

点击 List Page 跳转到 list 页

 vue构建单页面应用实战

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 页面的执行顺序如下:

 vue构建单页面应用实战

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

 vue构建单页面应用实战

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

vue构建单页面应用实战 

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>

浏览器里试试看:

 vue构建单页面应用实战

子组件成功被调用了

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 就是传参。

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

vue构建单页面应用实战

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

 vue构建单页面应用实战

传参逻辑成功。

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')
 }

}

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

初始状态:

 vue构建单页面应用实战

点了 tab1,tab2:

vue构建单页面应用实战 

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')
   }
  }
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
图片格式的JavaScript和CSS速查手册
Aug 20 Javascript
javascript 解析后的xml对象的读取方法细解
Jul 25 Javascript
javascript 导出数据到Excel(处理table中的元素)
Dec 18 Javascript
JavaScript 对象的属性和方法4种不同的类型
Mar 19 Javascript
浅析js中2个等号与3个等号的区别
Aug 06 Javascript
jQuery实现自定义下拉列表
Jan 05 Javascript
ECMAScript 5严格模式(Strict Mode)介绍
Mar 02 Javascript
JS实时弹出新消息提示框并有提示音响起的实现代码
Apr 20 Javascript
bootstrap模态框关闭后清除模态框的数据方法
Aug 10 Javascript
angular学习之动态创建表单的方法
Dec 07 Javascript
vue中axios防止多次触发终止多次请求的示例代码(防抖)
Feb 16 Javascript
JavaScript缓动动画函数的封装方法
Nov 25 Javascript
bootstrap suggest下拉框使用详解
Apr 10 #Javascript
JavaScript中使用webuploader实现上传视频功能(demo)
Apr 10 #Javascript
关于vue.js过渡css类名的理解(推荐)
Apr 10 #Javascript
vue.js单页面应用实例的简单实现
Apr 10 #Javascript
javascript内存分配原理实例分析
Apr 10 #Javascript
移动端触屏幻灯片图片切换插件idangerous swiper.js
Apr 10 #Javascript
Angular中ng-bind和ng-model的区别实例详解
Apr 10 #Javascript
You might like
PDO版本问题 Invalid parameter number: no parameters were bound
2013/01/06 PHP
浅谈使用 PHP 进行手机 APP 开发(API 接口开发)
2014/08/11 PHP
PHP中使用register_shutdown_function函数截获fatal error示例
2015/04/21 PHP
onsubmit阻止form表单提交与onclick的相关操作
2010/09/03 Javascript
node.js中的path.extname方法使用说明
2014/12/09 Javascript
js实现鼠标触发图片抖动效果的方法
2015/02/27 Javascript
jquery二级目录选中当前页的css样式
2016/12/08 Javascript
jQuery源码分析之init的详细介绍
2017/02/13 Javascript
打造通用的匀速运动框架(实例讲解)
2017/10/17 Javascript
node.js基础知识小结
2018/02/26 Javascript
微信小程序实现展示评分结果功能
2019/02/15 Javascript
Node.js Stream ondata触发时机与顺序的探索
2019/03/08 Javascript
javascript开发实现贪吃蛇游戏
2020/07/31 Javascript
浅谈JavaScript中的“!!”作用
2020/08/03 Javascript
[01:02:17]2014 DOTA2华西杯精英邀请赛 5 24 DK VS VG
2014/05/26 DOTA
python复制文件代码实现
2013/12/23 Python
python实现根据窗口标题调用窗口的方法
2015/03/13 Python
python中偏函数partial用法实例分析
2015/07/08 Python
在Python的Django框架中使用通用视图的方法
2015/07/21 Python
Python2包含中文报错的解决方法
2018/07/09 Python
Python3.5多进程原理与用法实例分析
2019/04/05 Python
pandas如何处理缺失值
2019/07/31 Python
python正则表达式匹配IP代码实例
2019/12/28 Python
python读取dicom图像示例(SimpleITK和dicom包实现)
2020/01/16 Python
浅谈python中频繁的print到底能浪费多长时间
2020/02/21 Python
python通过对字典的排序,对json字段进行排序的实例
2020/02/27 Python
Python批量删除mysql中千万级大量数据的脚本分享
2020/12/03 Python
新奇的小玩意:IWOOT
2016/07/21 全球购物
加拿大时尚床上用品零售商:QE Home | Quilts Etc
2018/01/22 全球购物
HashMap和Hashtable的区别
2013/05/18 面试题
幼儿师范毕业生自荐信
2013/11/09 职场文书
事业单位公务员的职业生涯规划
2014/01/15 职场文书
法制宣传月活动方案
2014/05/11 职场文书
群众路线表态发言材料
2014/10/17 职场文书
python文件名批量重命名脚本实例代码
2021/04/22 Python
德劲DE1105机评
2022/04/05 无线电