VUE做前端SEO不好优化,VUE前端开发遇到的问题

服务端渲染对于刚接触vue的新手来说,并不是那么友好,虽然已有官方SSR中文文档。但是对于一个已经开发完毕的vue项目去接SSR无论是从工作量还是技术角度来说,都是一种挑战。

2、预渲染方式

在构建时(buildtime)简单地生成针对特定路由的静态HTML文件。优点是设置预渲染更简单,并可以将您的前端作为一个完全静态的站点。如果您使用webpack,您可以使用prerender-spa-plugin轻松地添加预渲染。它已经被Vue应用程序广泛测试。

VUE前端开发遇到的问题

这次给大家带来Vue在前端开发中需要注意什么,Vue在前端开发的注意事项有哪些,下面就是实战案例,一起来看一下。

基于Vue官方风格指南整理

一、强制1. 组件名为多个单词

组件名应该始终是多个单词的,根组件 App 除外。

正例:

export default {name: TodoItem,// ...

}

反例:

export default {name: Todo,// ...

}2. 组件数据

组件的 data 必须是一个函数。

当在组件中使用 data 属性的时候 (除了 new Vue 外的任何地方),它的值必须是返回一个对象的函数。

正例:

// In a .vue file

export default {data () {return {foo: bar}}

}

// 在一个 Vue 的根实例上直接使用对象是可以的,

// 因为只存在一个这样的实例。

new Vue({data: {foo: bar}

})反例:

export default {data: {foo: bar}

}3. Prop定义

Prop 定义应该尽量详细。

在你提交的代码中,prop 的定义应该尽量详细,至少需要指定其类型。

正例:

props: {status: String

}

// 更好的做法!

props: {status: {type: String,required: true,validator: function (value) {return [syncing,synced,version-conflict,error].indexOf(value) !== -1}}

}反例:

// 这样做只有开发原型系统时可以接受

props: [status]4. 为v-for设置键值

总是用 key 配合 v-for。

在组件上_总是_必须用 key 配合 v-for,以便维护内部组件及其子树的状态。甚至在元素上维护可预测的行为,比如动画中的对象固化 (object constancy),也是一种好的做法。

正例:

{{ }}

反例:

{{ }}

v-if 和 v-for 用在一起

永远不要把 v-if 和 v-for 同时用在同一个元素上。

一般我们在两种常见的情况下会倾向于这样做:

为了过滤一个列表中的项目 (比如 v-for="user in users" v-if="")。在这种情形下,请将 users 替换为一个计算属性 (比如 activeUsers),让其返回过滤后的列表。

为了避免渲染本应该被隐藏的列表 (比如 v-for="user in users" v-if="shouldShowUsers")。这种情形下,请将 v-if 移动至容器元素上 (比如 ul, ol)。

正例:

{{ }}

反例:

{{ }}

6. 为组件样式设置作用域

对于应用来说,顶级 App 组件和布局组件中的样式可以是全局的,但是其它所有组件都应该是有作用域的。

这条规则只和单文件组件有关。你不一定要使用 scoped 特性。设置作用域也可以通过 CSS Modules,那是一个基于 class 的类似 BEM 的策略,当然你也可以使用其它的库或约定。

不管怎样,对于组件库,我们应该更倾向于选用基于 class 的策略而不是 scoped 特性。

这让覆写内部样式更容易:使用了常人可理解的 class 名称且没有太高的选择器优先级,而且不太会导致冲突。

正例:

{border: none;border-radius: 2px;

}

.c-Button--close {background-color: red;

}

反例:

{background-color: red;

} {border: none;border-radius: 2px;

}

.button-close {background-color: red;

}

二、强烈推荐(增强可读性)1. 组件文件

只要有能够拼接文件的构建系统,就把每个组件单独分成文件。

当你需要编辑一个组件或查阅一个组件的用法时,可以更快速的找到它。

正例:

components/

|-

|- :

V

(TodoList, {// ...

})

(TodoItem, {// ...

})2. 单文件组件文件的大小写

单文件组件的文件名应该要么始终是单词大写开头 (PascalCase)

正例:

components/

|- :

components/

|-

|- . 基础组件名

应用特定样式和约定的基础组件 (也就是展示类的、无逻辑的或无状态的组件) 应该全部以一个特定的前缀开头,比如 Base、App 或 V。

正例:

components/

|-

|-

|- :

components/

|-

|-

|- . 单例组件名

只应该拥有单个活跃实例的组件应该以 The 前缀命名,以示其唯一性。

这不意味着组件只可用于一个单页面,而是每个页面只使用一次。这些组件永远不接受任何 prop,因为它们是为你的应用定制的,而不是它们在你的应用中的上下文。如果你发现有必要添加 prop,那就表明这实际上是一个可复用的组件,只是目前在每个页面里只使用一次。

正例:

components/

|-

|- :

components/

|-

|- . 紧密耦合的组件名

和父组件紧密耦合的子组件应该以父组件名作为前缀命名。

如果一个组件只在某个父组件的场景下有意义,这层关系应该体现在其名字上。因为编辑器通常会按字母顺序组织文件,所以这样做可以把相关联的文件排在一起。

正例:

components/

|-

|-

|-

components/

|-

|- :

components/

|-

|- . 组件名中的单词顺序

组件名应该以高级别的 (通常是一般化描述的) 单词开头,以描述性的修饰词结尾。

正例:

components/

|-

|-

|-

|-

|-

|- :

components/

|-

|-

|-

|-

|-

|- . 模板中的组件名大小写

总是 PascalCase 的

正例:反例:8. 完整单词的组件名

组件名应该倾向于完整单词而不是缩写。

正例:

components/

|-

|- :

components/

|-

|- . 多个特性的元素

多个特性的元素应该分多行撰写,每个特性一行。

正例:反例:10. 模板中简单的表达式

组件模板应该只包含简单的表达式,复杂的表达式则应该重构为计算属性或方法。

复杂表达式会让你的模板变得不那么声明式。我们应该尽量描述应该出现的是什么,而非如何计算那个值。而且计算属性和方法使得代码可以重用。

正例:{{ normalizedFullName }}

// 复杂表达式已经移入一个计算属性

computed: {normalizedFullName: function () {return ( ).map(function (word) {return word[0].toUpperCase() (1)}).join( )}

}反例:

{{( ).map(function (word) {return word[0].toUpperCase() (1)}).join( )

}}11. 简单的计算属性

正例:

computed: {basePrice: function () {return / (1 - )},discount: function () {return * ( || 0)},finalPrice: function () {return - }

}反例:

computed: {price: function () {var basePrice = / (1 - )return (basePrice -basePrice * ( || 0))}

}12. 带引号的特性值

非空 HTML 特性值应该始终带引号 (单引号或双引号,选你 JS 里不用的那个)。

在 HTML 中不带空格的特性值是可以没有引号的,但这样做常常导致带空格的特征值被回避,导致其可读性变差。

正例:

反例:

13. 指令缩写

都用指令缩写 (用 : 表示 v-bind: 和用 @ 表示 v-on:)

正例:

反例:

三、推荐1. 单文件组件的顶级元素的顺序

单文件组件应该总是让、 和 标签的顺序保持一致。且 要放在因为另外两个标签至少要有一个。

正例:.../* ... */四、谨慎使用 (有潜在危险的模式)1. 没有在 v-if/v-if-else/v-else 中使用 key

如果一组 v-if v-else 的元素类型相同,最好使用 key (比如两个 元素)。

正例:

错误:{{ error }}{{ results }}反例:错误:{{ error }}{{ results }}

2. scoped 中的元素选择器

元素选择器应该避免在 scoped 中出现。

在 scoped 样式中,类选择器比元素选择器更好,因为大量使用元素选择器是很慢的。

正例:

{background-color: red;

}

反例:

Xbutton {background-color: red;

}

3. 隐性的父子组件通信

应该优先通过 prop 和事件进行父子组件之间的通信,而不是 this.$parent 或改变 prop。

正例:

(TodoItem, {props: {todo: {type: Object,required: true}},template: ``

})反例:

(TodoItem, {props: {todo: {type: Object,required: true}},methods: {removeTodo () {var vm = thisvm.$ = vm.$(function (todo) {return !== })}},template: `{{ }}X`

})4. 非 Flux 的全局状态管理

应该优先通过 Vuex 管理全局状态,而不是通过 this.$root 或一个全局事件总线。

正例:

// store/modules/

export default {state: {list: []},mutations: {REMOVE_TODO (state, todoId) { = (todo => !== todoId)}},actions: {rem

VUE前端开发常见问题

文章中给你列举了部分的面试题,这些都是公司面试常遇到的,还有需要的还可以自己去查阅一下资料

1、active-class是哪个组件的属性?嵌套路由怎么定义?

答:vue-router模块的router-link组件。

2、怎么定义vue-router的动态路由?怎么获取传过来的动态参数?

答:,对path属性加上/:id。

3、vue-router有哪几种导航钩子?

答:三种,一种是全局导航钩子:(to,from,next),作用:跳转前进行判断拦截。第二种:组件内的钩子;第三种:单独路由独享组件

4、scss是什么?安装使用的步骤是?有哪几大特性?

答:预处理css,把css当前函数编写,定义变量,嵌套。 先装css-loader、node-loader、sass-loader等加载器模块,:extenstion,再加多一个模块:module里面test、loader

、scss是什么??有哪几大特性?

答:css的预编译。

使用步骤:

第一步:用npm 下三个loader(sass-loader、css-loader、node-sass)

第二步:,

第三步:还是在同一个文件,配置一个module属性

第四步:然后在组件的style标签加上lang属性 ,例如:lang=”scss”

有哪几大特性:

1、可以用变量,例如($变量名称=值);

2、可以用混合器,例如()

3、可以嵌套

5、mint-ui是什么?怎么使用?说出至少三个组件使用方法?

答:基于vue的前端组件库。npm安装,然后import样式和js,(mintUi)全局引入。在单个组件局部引入:import {Toast} from ‘mint-ui’。组件一:Toast(‘登录成功’);组件二:mint-header;组件三:mint-swiper

6、v-model是什么?怎么使用? vue中标签怎么绑定事件?

答:可以实现双向绑定,指令(v-class、v-for、v-if、v-show、v-on)。vue的model层的data属性。绑定事件:

7、axios是什么?怎么使用?描述使用它实现登录功能的流程?

答:请求后台资源的模块。npm install axios -S装好,然后发送的是跨域,需在配置文件中config/。后台如果是Tp5则定义一个资源路由。js中使用import进来,。,

8、axios tp5进阶中,(‘api/user’)是进行的什么操作?(‘api/user/8′)呢?

答:跨域,添加用户操作,更新操作。

9、什么是RESTful API?怎么使用?

答:是一个api的标准,无状态请求。请求的路由地址是固定的,如果是tp5则先路由配置中把资源路由配置好。标准有:.post .put .delete

10、vuex是什么?怎么使用?哪种功能场景使用它?

答:vue框架中状态管理。,注入。新建了一个目录store,….. export 。场景有:单页应用中,组件之间的状态。音乐播放、登录状态、加入购物车

11、mvvm框架是什么?它和其它框架(jquery)的区别是什么?哪些场景适合?

答:一个model view viewModel框架,数据模型model,viewModel连接两个

区别:vue数据驱动,通过数据来显示视图层而不是节点操作。

场景:数据操作比较多的场景,更加便捷

12、自定义指令(v-check、v-focus)的方法有哪些?它有哪些钩子函数?还有哪些钩子函数参数?

答:全局定义指令:在vue对象的directive方法里面有两个参数,一个是指令名称,另外一个是函数。组件内定义指令:directives

钩子函数:bind(绑定事件触发)、inserted(节点插入的时候触发)、update(组件内相关更新)

钩子函数参数:el、binding

13、说出至少4种vue当中的指令和它的用法?

答:v-if:判断是否隐藏;v-for:数据循环出来;v-bind:class:绑定一个属性;v-model:实现双向绑定

14、vue-router是什么?它有哪些组件?

答:vue用来写路由一个插件。router-link、router-view

15、导航钩子有哪些?它们有哪些参数?

答:导航钩子有:a/全局钩子和组件内独享的钩子。b/beforeRouteEnter、afterEnter、beforeRouterUpdate、beforeRouteLeave

参数:有to(去的那个路由)、from(离开的路由)、next(一定要用这个函数才能去到下一个路由,如果不用就拦截)最常用就这几种

16、Vue的双向数据绑定原理是什么?

答: 是采用数据劫持结合发布者-订阅者模式的方式,()来劫持各个属性的setter,getter,在数据变动时发布消息给订阅者,触发相应的监听回调。

具体步骤:

第一步:需要observe的数据对象进行递归遍历,包括子属性对象的属性,都加上 setter和getter

这样的话,给这个对象的某个值赋值,就会触发setter,那么就能监听到了数据变化

第二步:compile解析模板指令,将模板中的变量替换成数据,然后初始化渲染页面视图,并将每个指令对应的节点绑定更新函数,添加监听数据的订阅者,一旦数据有变动,收到通知,更新视图

第三步:Watcher订阅者是Observer和Compile之间通信的桥梁,主要做的事情是:

1、在自身实例化时往属性订阅器(dep)里面添加自己

2、自身必须有一个update()方法

3、()通知时,能调用自身的update()方法,并触发Compile中绑定的回调,则功成身退。

第四步:MVVM作为数据绑定的入口,整合Observer、Compile和Watcher三者,通过Observer来监听自己的model数据变化,通过Compile来解析编译模板指令,最终利用Watcher搭起Observer和Compile之间的通信桥梁,达到数据变化 -> 视图更新;视图交互变化(input) -> 数据model变更的双向绑定效果。

ps:16题答案同样适合”vue data是怎么实现的?”此面试题。

17、请详细说下你对vue生命周期的理解?

答:总共分为8个阶段创建前/后,载入前/后,更新前/后,销毁前/后。

创建前/后: 在beforeCreated阶段,vue实例的挂载元素$el和数据对象data都为undefined,还未初始化。在created阶段,vue实例的数据对象data有了,$el还没有。

载入前/后:在beforeMount阶段,vue实例的$el和data都初始化了,但还是挂载之前为虚拟的dom节点,。在mounted阶段,vue实例挂载完成,。

更新前/后:当data变化时,会触发beforeUpdate和updated方法。

销毁前/后:在执行destroy方法后,对data的改变不会再触发周期函数,说明此时vue实例已经解除了事件监听以及和dom的绑定,但是dom结构依然存在

18、请说下封装 vue 组件的过程?

答:组件可以提升整个项目的开发效率。能够把页面抽象成多个相对独立的模块,解决了我们传统项目开发:效率低、难维护、复用性等问题。

,。子组件需要数据,可以在props中接受定义。而子组件修改好数据后,想把数据传递给父组件。可以采用emit方法。

19、你是怎么认识vuex的?

答:vuex可以理解为一种开发模式或框架。比如PHP有thinkphp,java有spring等。

通过状态(数据源)集中管理驱动组件的变化(好比spring的IOC容器对bean进行集中管理)。

应用级的状态集中放在store中; 改变状态的方式是提交mutations,这是个同步的事物; 异步逻辑应该封装在action中。

20、vue-loader是什么?使用它的用途有哪些?

答:,跟template/js/style转换成js模块。

用途:js可以写es6、style样式可以scss或less、template可以加jade等

21、?

答:assets文件夹是放静态资源;components是放组件;router是定义路由相关的配置;view视图;;

22、?有遇到过哪些问题吗?

答:第一步:在components目录新建你的组件文件(),script一定要export default {

第二步:在需要用的页面(组件)中导入:import smithButton from ‘../components/’

第三步:注入到vue的子组件的components属性上面,components:{smithButton}

第四步:在template视图view中使用,

问题有:smithButton命名,使用的时候则smith-button。

23、?

答:简而言之,就是先转化成AST树,再得到的render函数返回VNode(Vue的虚拟DOM节点)

详情步骤:

通过compile编译器把template编译成AST语法树(abstract syntax tree 即 源代码的抽象语法结构的树状表现形式),compile是createCompiler的返回值,createCompiler是用以创建编译器的。另外compile还负责合并option。

AST会经过generate(将AST语法树转化成render funtion字符串的过程)得到render函数,render的返回值是VNode,VNode是Vue的虚拟DOM节点,里面有(标签名、子节点、文本等等)