前言  

本文的发表源于知乎的一篇文章。文章链接如下:如果你想靠前端技术还房贷,你不能连这个都不会

前提

1. 本文是一个vue的小组件功能,你需要有vue的经验,并且了解vue的组件。

2. 本文会贴出一个盗版的jQuery.queue函数,这个函数是之前读jq源码自己写的,就是偷得jq,比不上jq强大,但是基本功能还是有的。

3. 本文不适合新手。

切入正文

上面是问题,我们来解读一下需求:

1. 首先要有一个模块,这个模块包括一个input,两个button

2. 模块实现了点击 A,发送urlA请求,并将input值改为请求返回的值,点击B,雷同

3. 用户依次点击A、B,input首先需要改为urlA请求返回的值,再改为urlB返回的值。也就类似于同步请求。

代码如下:

<div id="app">
<my-component></my-component>
</div>
<template id="tpl">
<div>
<input type="text" v-model="inputData" readonly="readonly">
<p> <button @click="request1">按钮A</button>
<button @click="request2">按钮B</button>
</p>
</div>
</template> <script src="https://unpkg.com/vue/dist/vue.js"></script>
// resource.min.js 自己下载的
<script type="text/javascript" src="./resource.min.js"></script> <script>
Vue.use(VueResource)
new Vue({
el: '#app',
components: {
'my-component': {
template: '#tpl',
data: function () {
return {
inputData: '默认的',
ajax2: null
}
},
methods: {
request1: function () {
var url = '我的测试地址:睡眠2秒再返回值'
this.$http.get(url).then(function(res) {
this.inputData = res.data
this.ajax2()
})
},
request2: function () {
this.ajax2 = function () {
var url = '我的测试地址:睡眠1秒返回值'
this.$http.get(url).then(function(res) {
this.inputData = res.data
})
}
},
},
}
}
})
</script>

我定义了一个vue实例,在#app元素内有效。定义个组件,给这个组件一个inputData属性存储input框中的数据,定义一个ajax2属性存储点击B按钮时的发起请求的函数。我们在点击A后返回值后,调用ajax2这个方法,这样就实现了上面的需求,当然仅仅是实现了上面的需求而已,并且代码看上去很难看,因为我们为了实现这个功能不得不在模型上加了个ajax2这个很鸡肋的中间量,为了让代码更好看一些,功能更强一些,不防试一试用队列来解决。

队列的作用

我们不仅要实现上面这个简单的例子,我们需要实现的效果更强壮:

1. 连续交替点击A、B按钮,将返回值有顺序的显示到input上面

2. 每一次点击都会产生一个ajax请求,但是不会立刻发起,会根据点击顺序依次请求。

3. 利用一个队列对象来实现,使代码变得更简洁更美观。

代码如下:

  <div id="app">
<my-component :q="q"></my-component>
</div>
<template id="tpl">
<div>
<input type="text" v-model="inputData" readonly="readonly">
<p>
<button @click="request('测试地址1:睡眠2秒')">按钮A</button>
<button @click="request('测试地址2:睡眠1秒')">按钮B</button>
</p>
</div>
</template>
<script src="https://unpkg.com/vue/dist/vue.js"></script>
<script type="text/javascript" src="./vue-resource.min.js"></script>
<script type="text/javascript" src="http://git.oschina.net/xuazheng/myJquery/raw/master/queue/queue.js?dir=0&filepath=queue%2Fqueue.js&oid=b23c3bf7212ff41aad350bdb505a1afc59929ce6&sha=d0298a8907c9ed1cf25c176807fadbcd14c3e571"></script>
<script type="text/javascript">
Vue.use(VueResource)
new Vue({
el: '#app',
data: {
q: Queue()
},
components: {
'my-component': {
template: '#tpl',
data: function () {
return {
inputData: '默认的'
}
},
methods: {
request: function (url) {
this.q.queue('fx', function (next){
this.$http.get(url).then(function(res) {
this.inputData = res.data
next()
})
}.bind(this))
}
},
props: ['q']
},
}
})
</script>

引入了我的queue,在vue实例上,创建一个q属性存储queue对象并传递给子组件。在子组件中,我们每次点击一个按钮,都会将一个ajax请求的函数添加到fx队列中,在jquery中,fx类型的队列存储着动画函数,可以处理异步函数队列的有序执行。不传递这个值会默认fx类型,所以也可以直接在queue方法中传递一个方法就行。

队列的代码如下:

;function Queue() {

  // 数据缓存对象
var cache = {}; var queueList = {
// type默认是fx,是动画队列
queue: function(type,data) {
var args = arguments;
//没有参数直接返回
if(!args.length){
return;
} var q = null;
// 只有一个参数并且是个函数
if(args.length == && typeof type === 'function') {
data = type;
type = 'fx';
} q = cache[type] || []; // 添加缓存
if( data instanceof Array) {
q = data;
}else {
q.push(data)
}
cache[type] = q; //如果是动画队列并且没有开始的动画,执行第一个动画函数
if(type == 'fx' && q.toString().indexOf('inprogress') === -) {
queueList.dequeue()
} return q; },
dequeue: function(type) {
var fn, queue;
type = type || 'fx';
queue = cache[type];
if(queue.length == ) {
return;
} fn = queue.shift(); if( fn === 'inprogress' ) {
fn = queue.shift();
}
if( fn ) {
if(type === 'fx') {
queue.unshift('inprogress');
}
fn.call(null,function() {
queueList.dequeue(type);
})
}
},
// 延迟使用setTimeout来实现
delay: function(type,timeout) {
if(!type) {
return;
}
if(arguments.length == ) {
timeout = type;
type = 'fx';
} if(typeof timeout == 'number') {
var q = cache[type];
if(!q) {
q = cache[type] = [_delay];
}else {
q.push(_delay)
} }
function _delay() {
setTimeout(queueList.dequeue, timeout);
} return this; },
get: function(type) {
type = type || 'fx';
return cache[type];
}
} return queueList;
}

这个就不解释了,比起jquery源码,这个代码就显得很简单,jquery中做了大量的处理,然而博主并没有那么厉害,只能简单的写了这点。有js基础的应该看得懂,如果想学jq源码,推荐 艾伦 Aaron 的博客园

感谢阅读

一篇知乎的故事 - javascript技术贴的更多相关文章

  1. FineUI(专业版)实现百变通知框(无JavaScript代码)!

    博客园已经越来越不公正了,居然说我这篇文章没有实质的内容!! 我其实真的想问哪些通篇几十个字,没任何代码和技术分享,嚷嚷着送书的文章的就能雄霸博客园首页几天,我这篇文章偏偏就为管理员所容不下. 其实我 ...

  2. 能说明你的Javascript技术很烂的五个原因

    Javascript在互联网上名声很臭,但你又很难再找到一个像它这样如此动态.如此被广泛使用.如此根植于我们的生活中的另外一种语言.它的低学习门槛让很多人都称它为学前脚本语言,它另外一个让人嘲笑的东西 ...

  3. 细数Javascript技术栈中的四种依赖注入

    作为面向对象编程中实现控制反转(Inversion of Control,下文称IoC)最常见的技术手段之一,依赖注入(Dependency Injection,下文称DI)可谓在OOP编程中大行其道 ...

  4. 第61节:Java中的DOM和Javascript技术

    Java中的DOM和Javascript技术 DOM是一门技术,是文档对象模型.所需的文档只有标记型文档,如我们所学的html文档(文档中的所有标签都封装成为对象了) DOM: 为Document O ...

  5. 通过 profiling 定位 golang 性能问题 - 内存篇 原创 张威虎 滴滴技术 2019-08-02

    通过 profiling 定位 golang 性能问题 - 内存篇 原创 张威虎 滴滴技术 2019-08-02

  6. 奇艺iOS移动端网络优化实践 | 请求成功率优化篇 原创 Charles 爱奇艺技术

    奇艺iOS移动端网络优化实践 | 请求成功率优化篇 原创 Charles 爱奇艺技术

  7. 2000条你应知的WPF小姿势 基础篇<28-33 WPF启动故事>

    在正文开始之前需要介绍一个人:Sean Sexton. 来自明尼苏达双城的软件工程师.最为出色的是他维护了两个博客:2,000Things You Should Know About C# 和 2,0 ...

  8. 【JavaScript】我的JavaScript技术总结第一篇——编程细节

    遍历数组 for (var i=0, l=arr.length; i<l; i++) 这样写的一个好处就是让每次循环少一步获取数组对象长度的操作,数组长度越长,价值越明显. 判断变量的真假 if ...

  9. 一篇JavaScript技术栈带你了解继承和原型链

    作者 | Jeskson 来源 | 达达前端小酒馆 1 在学习JavaScript中,我们知道它是一种灵活的语言,具有面向对象,函数式风格的编程模式,面向对象具有两点要记住,三大特性,六大原则. 那么 ...

随机推荐

  1. USB自定义HID设备实现-STM32

    该文档使用USB固件库,在其基础上进行了自己的定制,完成了一个USB-HID设备,首先是usb_desc.c文件,里面存放了usb各种描述符的存在 #include "usb_desc.h& ...

  2. LPC1768的USB-相关结构体定义

    #ifndef __USB_H__ #define __USB_H__ //usb传输数据的宏定义描述 #include "sys.h" typedef __packed unio ...

  3. iOS技术框架构和更新版本的技术特性

    Core OS层 Sytem 系统层包括内核环境,驱动及操作系统层unix接口.内核以mach为基础,它 负责操作系统的各个方面,包括管理系统的虚拟内存,线程,文件系统,网络以及进程间通讯.这一层包含 ...

  4. CodeForces 754D Fedor and coupons (优先队列)

    题意:给定n个优惠券,每张都有一定的优惠区间,然后要选k张,保证k张共同的优惠区间最大. 析:先把所有的优惠券按左端点排序,然后维护一个容量为k的优先队列,每次更新优先队列中的最小值,和当前的右端点, ...

  5. 100套新鲜免费的PS笔刷下载

    这篇文章所有的Photoshop笔刷都是免费且高质量的.笔刷总类齐全:有飞鸟.冰块.水.树枝.喷墨.科技元素.皮肤纹理.烟火等等!用它们来加速你的工作流程,提升作品档次吧!”一挪妖娆举动,一刷风情万种 ...

  6. ANSI标准

    NSI:美国国家标准学会(AMERICAN NATIONAL STANDARDS INSTITUTE: ANSI)成立于1918年.当时,美国的许多企业和专业技术团体,已开始了标准化工作,但因彼此间没 ...

  7. python3中bytes与string的互相转换

    首先来设置一个原始的字符串, Python 3.2.3 (default, Apr 11 2012, 07:15:24) [MSC v.1500 32 bit (Intel)] on win32 Ty ...

  8. python 模块添加

    python包含子目录中的模块方法比较简单,关键是能够在sys.path里面找到通向模块文件的路径.下面将具体介绍几种常用情况: (1)主程序与模块程序在同一目录下: 如下面程序结构:`-- src  ...

  9. react中常用的一些方法

    React.createClass:创建一个ReactClass(组件类),参数是一个对象且必须带有 render 属性方法,该方法必须返回一个封闭的容器(容器内可以有其它不限结构的容器)或 null ...

  10. HDU-1864-最大报销额

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1864 这题开始题意没搞清楚,就做题了,导致浪费了很多的时间,不应该啊, 注意事项:每张发票上,单项物品 ...