Vue中的渲染方式总结可分四种:

  • 原有模板语法,挂载渲染
  • 使用render属性,createElement函数直接渲染
  • 使用render属性,配合组件的template属性,createElement函数渲染
  • 使用render属性,配合单文件组件,createElement函数渲染

方式一

原有模板语法,挂载渲染:就是对使用Vue标签语法的hmtl进行渲染。

实例代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app"><input v-model="message"><p>Message is: {{ message }}</p>
</div>
</body>
<script src="js/vue.js"></script>
<script type="text/javascript">var v = new Vue({el: '#app',data: {message: '这是内容'}});
</script>
</html>

特点

当页面返回时html中的v-model等属性并没有被渲染,保持不变发给客户端,客户端直到加载了Vue,创建了实例之后才会将这些标识渲染出来。

v-if显示后消失或消失后显示造成跳变

这种方式的缺点是明显的:浏览器第一时间无法识别这些标识!这会使得在某些情况下出现奇怪的现象,下面我修改代码以做演示:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app"><template v-if="type === '1'"><div style="width: 400px;height: 300px;background-color: blue;"></div></template><div style="width: 400px;height: 200px;background-color: red;"></div>
</div>
</body>
<script src="js/vue.js"></script>
<script type="text/javascript">var v = new Vue({el: '#app',data: {type: '1'}});
</script>
</html>

可以看到上面的第一个div在type为1时应该显示,但是它在页面初始加载是不会显示的,一个html并没有<template>标签,整个标签会被跳过,那么这个页面展示的结果就会有一个一瞬间的跳变。

这个过程非常短,但是人眼是绝对识别得到的,在谷歌浏览器下使用Ctrl+Shift+R,360浏览器下使用Ctrl+F5强制不使用缓存加载页面我们就能看到这个跳变了!

注意另一种写法也会有跳变情况:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app"><div v-if="type === '1'" style="width: 400px;height: 300px;background-color: blue;"></div><div style="width: 400px;height: 200px;background-color: red;"></div>
</div>
</body>
<script src="js/vue.js"></script>
<script type="text/javascript">var v = new Vue({el: '#app',data: {type: '2'}});
</script>
</html>

解释:div默认不会识别v-if默认显示,但是tpye为2,Vue加载后隐藏了该div,造成跳变。

如何解决跳变问题?

使用下面所有方法都可以解决,思路是用render函数进行渲染,而目标div内不含任何内容,也就是先什么都不显示,然后渲染出来正确的html,自然不会有跳变。

方式二

使用render属性,createElement函数直接渲染:原本无html,通过JavaScript 的完全编程的能力生成页面。

实例代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app">
</div>
</body>
<script src="js/vue.js"></script>
<script type="text/javascript">var v = new Vue({el: '#app',data: {message: '这是内容'},render: function (createElement) {return createElement('p', 'Message is:' + this.message)}});
</script>
</html>

这里目标div内初始时没有任何内容,使用render方法生成内容,这里我没有渲染出v-model的效果,因为实现起来比较复杂,要用到组件,具体参考官网渲染函数 & JSX,Vue 的模板实际是就是被编译成了 render 函数。

这里render不要使用ECMAScript6语法如下:

render: h => h('p', this.message)

因为在ECMAScript6语法中,这里的this会指向[object Window],这个this.message是获取不到值的,而在上面一个的例子中this会指向当前Vue实例。

特点

极少会使用的方式,只适合一些细节渲染,虽完全控制输出,但不够直观,实现复杂,远不及模板标记语法:让Vue去帮我们编译渲染。

方式三

使用render属性,配合组件的template属性,createElement函数渲染。

相比于上一个方式,加入组件,利用template属性,更为直观,同样是原本无html,通过render函数渲染。

实例代码:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app">
</div>
</body>
<script src="js/vue.js"></script>
<script type="text/javascript">var MyComponent = {data () {return {message: '这是内容'}},template: `<div id="app"><input v-model="message"><p>Message is: {{ message }}</p></div>`};var v = new Vue({el: '#app',components: {'my-component': MyComponent},render: function (createElement) {return createElement('my-component')}//ECMAScript6: render: h => h('my-component')});
</script>
</html>

这里渲染组件时的渲染的内容会覆盖原<div id="app"></div>,而不是在其之下生成子节点。

特点

动态渲染,并且通过组件实现了模块分离,但是html模板被包在````里,使用不方便,IDE无法高亮代码,不适合大型项目。

下面是官网上对于这种方式的描述:

  • 全局定义 (Global definitions) 强制要求每个 component 中的命名不得重复(上面我用的是局部注册组件,甚至组件和实例不能分文件)
  • 字符串模板 (String templates) 缺乏语法高亮,在 HTML 有多行的时候,需要用到丑陋的
  • 不支持 CSS (No CSS support) 意味着当 HTML 和 JavaScript 组件化时,CSS 明显被遗漏
  • 没有构建步骤 (No build step) 限制只能使用 HTML 和 ES5 JavaScript, 而不能使用预处理器,如 Pug (formerly Jade) 和 Babel

方式四

说了前三种方式,其实只是一步步引出第四种方式:使用render属性,配合单文件组件,createElement函数渲染。

这种方式是绝大部分Vue项目(官方脚手架就是采用该渲染方式)的渲染方式。

在理解这一部分内容之前,你需要有webpack、vue-loader的知识:

1、webpack是核心,将组件文件打包,配合loader解析文件。

2、vue-loader是vue单文件组件的核心,负责解析.vue文件。

webpack以及loader

webpack,一种打包文件工具,配合一些插件可以模块化文件的打包处理等,我也是边学边做,自行百度了解。

loader,webpack非常常用的属性,让你能配合各种loader随意加载各种东西,看下官网给出的解释:

loader 用于对模块的源代码进行转换。loader 可以使你在 import 或"加载"模块时预处理文件。因此,loader 类似于其他构建工具中“任务(task)”,并提供了处理前端构建步骤的强大方法。loader 可以将文件从不同的语言(如 TypeScript)转换为 JavaScript,或将内联图像转换为 data URL。loader 甚至允许你直接在 JavaScript 模块中 import CSS文件!

比如这里我们有.vue模块组件,我们要在另一个文件中import App from './App.vue'该如何做到,我一步步讲。

实现步骤

我是使用命令的方式先打包出一个目标文件,再运行服务器,好像可以写到package.json的'scripts'字段里做一个步骤的简化,我没试,还是最简单的方式吧。

1、我是用命令行打包所以先全局安装webpack,注意webpack4.0+版本后要附带安装一个webpack-cli或是webpack-command其中之一,我是安的webpack-command:

npm install webpack webpack-command -g

2、因为babel-loader、vue-loader等必须要有webpack,好像不在项目中载入webpack也不行,所以进入项目根目录(node_modules文件夹所在目录),npm install webpack --save在项目中加载webpack。

3、接着,再把需要的loader加载一下,首先是vue-loader,是解析.vue文件的核心loader,接着是babel-loader,它是用来修改不兼容的javaScript语法的loader,用来处理.vue文件中的<script></script>,还有处理<style></style>的css-loader、vue-style-loader,还有vue-loader强制附带个vue-template-compiler,我不知道是干嘛的。

加载babel:npm install babel-loader babel-core babel-preset-env --save

加载其他必要的:npm install vue-loader css-loader vue-style-loader vue-template-compiler --save

4、在根目录下新建一个名为dist的文件夹存放生成的文件。

5、在根目录下新建文件App.vue,这就是单文件组件了,而且是根组件,添加代码如下:

<template><div id="app"><input v-model="message"><p>Message is: {{ message }}</p></div>
</template><script>
export default {data () {return {message: 'dsdsadasad'}}
}
</script>

6、在根目录下新建webpack打包入口文件entry-client.js,添加代码如下:

import Vue from 'vue' //注意在静态编译时全都使用import,不要使用require,无法生效
import App from './App' //注意在静态编译时全都使用import,不要使用require,无法生效const app = new Vue({components: {App},render: h => h(App)
});app.$mount('#app');

7、在根目录下新建一个名为build的文件夹,在该文件夹内创建一个文件名为webpack.config.js的文件作为webpack配置文件,代码如下:

const path = require('path')
const VueLoaderPlugin = require('vue-loader/lib/plugin')module.exports = {entry: {'client.bundle': '../entry-client.js',},output: {path: path.resolve(__dirname, '../dist'), //生成文件放到dist目录下filename: '[name].js'},mode: 'development', //开发模式,有错误vue会在控制台提示resolve: {//加了下面这项我们就能将import App from './App.vue' 省略为import App from './App'extensions: ['.js', '.vue', '.json']},//设置loader以及插件,可参考vue-loader官网module: {rules: [{test: /.vue$/,loader: 'vue-loader'},// 它会应用到普通的 `.js` 文件// 以及 `.vue` 文件中的 `<script>` 块{test: /.js$/,loader: 'babel-loader'},// 它会应用到普通的 `.css` 文件// 以及 `.vue` 文件中的 `<style>` 块{test: /.css$/,use: ['vue-style-loader','css-loader']}]},plugins: [// 请确保引入这个插件来保证vue-loader的使用new VueLoaderPlugin()]
};

8、命令行进入build文件夹,输入命令webpack打包文件,然后就可以在dist目录下看到生成的client.bundle.js目标文件了。

9、在根目录下创建测试文件index.html,添加代码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>TestVue</title>
</head>
<body>
<div id="app">
</div>
</body>
<script type="text/javascript" src="dist/client.bundle.js"></script>
</html>

特点

单文件组件,模块化,动态渲染,典型的单页面应用

vue 多页面多模块分模块打包 分插件安装_Vue渲染方式相关推荐

  1. JavaScript高级语法打包 - babel插件安装配置报错!Error: Cannot find module ‘@babel/preset-preset.env‘

    目录 1. 插件安装和配置 2. 运行打包 - 报错信息 3. 解决办法 1. 插件安装和配置 安装babel转换器相关的包: npm i babel-loader @babel/core @babe ...

  2. vue 分模块打包 脚手架_vue-cli分模块独立打包

    基于vue-cli3多模块独立打包 一.目标 我们要实现什么? 所谓分模块打包,也可以说一个项目一个模块,理解: 在src目录下,多个项目共用一些数据方法,但是每个项目有自己独立的入口文件,路由文件, ...

  3. bootstrap-pagination数据全部加载到前端再进行处理_求助!vue单页项目如何改造路由使其能够进行分模块打包?...

    最近遇到一个很难解决的问题,搜索了百度谷歌都没有相关的资料,有可能是我搜索的关键词不对或者是需要我换一种思路去解决,希望各位大佬能帮忙看一下怎么解决.问题目前是这样的,目前有个基于VUE的单页后台项目 ...

  4. 多系统集成:vue大型项目之分模块运行/打包

    本文以vue-cli3+为例,实现多系统集成下的分模块打包.分模块打包方式多种多样,下文可适用于多系统之间互不干扰,主系统可集成各子系统,各子系统又可单独运行的业务场景. 一.目标 我们要实现什么?所 ...

  5. vue 分模块打包 脚手架_Vue面试官最爱的底层源码问题,你可以这样回答!

    最近看到身边很多人都在投简历,有因为企业裁员的,有因为自己想跳槽的,原因不一,但是最终大家都会需要接触到面试这个事情.但是很多人对待面试不够认真,只会等待结果,不去努力.所以这边想整理一些懒人面试技巧 ...

  6. vue 多页面打包配置

    因为我们公司的项目是多页面应用,不同于传统单页面应用,一个包就可以了.我们必须要分模块打包,因此特意研究了下webpack,然后整理了一下多页面打包的逻辑. 1.首先,用vue-cli搭好基本的项目结 ...

  7. 【SSM整合】SSM详细整合-maven分模块架构

    本文从一个简单的三层案例入手,分模块整合SSM框架,从环境搭建到测试,每一步都比较详细,希望能对你有所帮助. 文章难免存在错误,望大佬及时指教. 文章所有代码在文章末尾给出. 0x01.SSM整合说明 ...

  8. 02-Maven高级-分模块开发、依赖传递、聚合、继承(SpringBoot的部分底层原理)、多模块开发(环境切换)、Nexus私服搭建与使用

    文章目录 学习目标 一.分模块开发与设计 1. 分模块开发的意义 问题导入 模块拆分原则 2. 分模块开发(模块拆分) 问题导入 2.1 创建Maven模块 2.2 书写模块代码 2.3 通过mave ...

  9. 基于SpringBoot构建分模块项目

    前言 步骤过于详细,多图慎入!!! 假设一个场景,要开发一个4s店维修部的办公系统,其功能有:前台接待,维修抢单,财务结算,库存管理.于是我们创建一个项目balabalabala写完交工. 一段时间后 ...

最新文章

  1. 基于三维点云数据的主成分分析方法(PCA)的python实现
  2. seaborn可视化散点图并自定义可视化结果图像的大小(Change the Size of a Seaborn Plot)
  3. 算法练习----java字符全排列
  4. Class.getResources()和classLoader.getResources()区别
  5. Linux文件的切分和结合
  6. HTML知识点总结之ul,ol,li标签
  7. 简单配置局域网FTP
  8. Facebook 中国程序员之死
  9. silverlight 跨域socket
  10. Java字节流的使用
  11. mysql 不同服务器数据库表同步_mysql 不同服务器数据库表同步
  12. python cls参数_python cls self 讲解
  13. Mysql-slowlog
  14. 业余草最新热门博客推荐
  15. 基于java的租房系统源代码_基于jsp的租房管理系统-JavaEE实现租房管理系统 - java项目源码...
  16. select默认选中
  17. Android Studio打开之后class显示灰色 代码没有颜色区分 输入没了提示的解决方法
  18. Javaweb安全——JSP Webshell
  19. 图像篡改检测C语言,图像篡改检测和定位(二)
  20. 微软账户服务器连不上开不了机,Win10无法登录微软账户提示“内部服务器错误(500)”怎么解决?...

热门文章

  1. P6810 「MCOI-02」Convex Hull 凸包
  2. CF372D. Choosing Subtree is Fun
  3. P3033 [USACO11NOV]牛的障碍Cow Steeplechase
  4. [CQOI2017] 老C的键盘(树形dp + 组合数)
  5. [HAOI2018] 染色(二项式反演+NTT)
  6. YBTOJ:向量问题(线段树分治、凸包)
  7. P2607-[ZJOI2008]骑士【基环树,树形dp】
  8. 【做题记录】[NOIP2016 普及组] 魔法阵
  9. 【Trie】最大异或对(ybtoj Trie-2)
  10. 照看小猫(nowcoder 217602)