zepto源码分析------ajax篇

上级研究了zepto的事件,也说了一下zepto的主要模块构成。恩,今天来说说的zepto的ajax是如何实现。

好的,和上次一样,先上代码再说。

  $.ajax = function(options) {
    // 覆盖配置
    var settings = $.extend({}, options || {}),
      // 没有找到在哪里
      deferred = $.Deferred && $.Deferred(),
      urlAnchor
    // 完成完整对配置覆盖
    for (key in $.ajaxSettings)
      if (settings[key] === undefined) settings[key] = $.ajaxSettings[key]

    // 这里要涉及到三个函数 ajaxStart>triggerGlobal>triggerAndReturn(触发ajax)
    ajaxStart(settings)

    // 是跨域
    if (!settings.crossDomain) {
      urlAnchor = document.createElement(‘a‘)
      urlAnchor.href = settings.url
      urlAnchor.href = urlAnchor.href
      // 验证是否真的跨域了----确保对参数的修正
      settings.crossDomain = (originAnchor.protocol + ‘//‘ + originAnchor.host) !== (urlAnchor.protocol + ‘//‘ + urlAnchor.host)
    }

    if (!settings.url) settings.url = window.location.toString()
    // 序列化数据
    serializeData(settings)

    var dataType = settings.dataType,
    // 解析地址如果存在回调那么hasplaceholder = true
      hasPlaceholder = /\?.+=\?/.test(settings.url)

    if (hasPlaceholder) dataType = ‘jsonp‘

    if (settings.cache === false || (
        (!options || options.cache !== true) &&
        (‘script‘ == dataType || ‘jsonp‘ == dataType)
      ))
      // 增加一个选择器(不是很明白)
      settings.url = appendQuery(settings.url, ‘_=‘ + Date.now())

    if (‘jsonp‘ == dataType) {
      if (!hasPlaceholder)
        settings.url = appendQuery(settings.url,
          settings.jsonp ? (settings.jsonp + ‘=?‘) : settings.jsonp === false ? ‘‘ : ‘callback=?‘)
      // 如果是跨域则用jsonp跨域
      return $.ajaxJSONP(settings, deferred)
    }

    var mime = settings.accepts[dataType],
      headers = {},
      setHeader = function(name, value) {
        headers[name.toLowerCase()] = [name, value]
      },
      protocol = /^([\w-]+:)\/\//.test(settings.url) ? RegExp.$1 : window.location.protocol,
      xhr = settings.xhr(),
      nativeSetHeader = xhr.setRequestHeader,
      abortTimeout

    if (deferred) deferred.promise(xhr)

    // 如果是跨域请求
    if (!settings.crossDomain) setHeader(‘X-Requested-With‘, ‘XMLHttpRequest‘)
    setHeader(‘Accept‘, mime || ‘*/*‘)
    if (mime = settings.mimeType || mime) {
      if (mime.indexOf(‘,‘) > -1) mime = mime.split(‘,‘, 2)[0]
      xhr.overrideMimeType && xhr.overrideMimeType(mime)
    }
    if (settings.contentType || (settings.contentType !== false && settings.data && settings.type.toUpperCase() != ‘GET‘))
      setHeader(‘Content-Type‘, settings.contentType || ‘application/x-www-form-urlencoded‘)

    if (settings.headers)
      for (name in settings.headers) setHeader(name, settings.headers[name])
    xhr.setRequestHeader = setHeader

    // 监听请求状态
    xhr.onreadystatechange = function() {
      if (xhr.readyState == 4) {
        xhr.onreadystatechange = empty
        clearTimeout(abortTimeout)
        var result, error = false
        if ((xhr.status >= 200 && xhr.status < 300) || xhr.status == 304 || (xhr.status == 0 && protocol == ‘file:‘)) {
          dataType = dataType || mimeToDataType(settings.mimeType || xhr.getResponseHeader(‘content-type‘))
          result = xhr.responseText

          try {
            // http://perfectionkills.com/global-eval-what-are-the-options/
            if (dataType == ‘script‘)(1, eval)(result)
            else if (dataType == ‘xml‘) result = xhr.responseXML
            else if (dataType == ‘json‘) result = blankRE.test(result) ? null : $.parseJSON(result)
          } catch (e) {
            error = e
          }

          if (error) ajaxError(error, ‘parsererror‘, xhr, settings, deferred)
          else ajaxSuccess(result, xhr, settings, deferred)
        } else {
          ajaxError(xhr.statusText || null, xhr.status ? ‘error‘ : ‘abort‘, xhr, settings, deferred)
        }
      }
    }

    if (ajaxBeforeSend(xhr, settings) === false) {
      xhr.abort()
      ajaxError(null, ‘abort‘, xhr, settings, deferred)
      return xhr
    }

    if (settings.xhrFields)
      for (name in settings.xhrFields) xhr[name] = settings.xhrFields[name]

    // 设置异步和同步
    var async = ‘async‘ in settings ? settings.async : true
    xhr.open(settings.type, settings.url, async, settings.username, settings.password)

    for (name in headers) nativeSetHeader.apply(xhr, headers[name])

    if (settings.timeout > 0) abortTimeout = setTimeout(function() {
      xhr.onreadystatechange = empty
      xhr.abort()
      ajaxError(null, ‘timeout‘, xhr, settings, deferred)
    }, settings.timeout)

    // avoid sending empty string (#319)
    xhr.send(settings.data ? settings.data : null)
    return xhr
  }

好了,这是ajax的核心部分,主要的注释我读写在了上面。其实我们都知道一个事情。无论是post还是get一定会有open(xxx)这个方法,那么我们很明显在上面的源码中只看了一句。那问题来了。作者写这么多干什么?我们再从上至下分析一下。ajax这个函数里面的功能实现过程setting(配置覆盖原本在 $.ajaxSettings里已经写好了一些配置,通过扩展的方式把用户的配置扩展进来)----处理是不是跨域了----请求开始---状态监听----回调处理-----数据解析。好了,大致过程就是这样了。剩下的一些函数自己去摸索就行了。

我们在开发页面的时候不但会使用ajax请求 有些时候我们遇到跨域这种东西。好的,我们来说说它是怎么实现跨域的。还是上代码,不过这次比较简单,嘿嘿。

 $.ajaxJSONP = function(options, deferred) {
    if (!(‘type‘ in options)) return $.ajax(options)

    var _callbackName = options.jsonpCallback,
      callbackName = ($.isFunction(_callbackName) ?
        _callbackName() : _callbackName) || (‘jsonp‘ + (++jsonpID)),
      script = document.createElement(‘script‘),
      originalCallback = window[callbackName],
      responseData,
      abort = function(errorType) {
        $(script).triggerHandler(‘error‘, errorType || ‘abort‘)
      },
      xhr = {
        abort: abort
      },
      abortTimeout

    if (deferred) deferred.promise(xhr)

    $(script).on(‘load error‘, function(e, errorType) {
      clearTimeout(abortTimeout)
      $(script).off().remove()

      if (e.type == ‘error‘ || !responseData) {
        ajaxError(null, errorType || ‘error‘, xhr, options, deferred)
      } else {
        ajaxSuccess(responseData[0], xhr, options, deferred)
      }

      window[callbackName] = originalCallback
      if (responseData && $.isFunction(originalCallback))
        originalCallback(responseData[0])

      originalCallback = responseData = undefined
    })

    if (ajaxBeforeSend(xhr, options) === false) {
      abort(‘abort‘)
      return xhr
    }

    window[callbackName] = function() {
      responseData = arguments
    }

    script.src = options.url.replace(/\?(.+)=\?/, ‘?$1=‘ + callbackName)
    document.head.appendChild(script)

    if (options.timeout > 0) abortTimeout = setTimeout(function() {
      abort(‘timeout‘)
    }, options.timeout)

    return xhr
  }

一看,我连注释都没写。很明显这个就很简单了。是吧。我们说说实现原理。

一般来说跨域会受到同源策略的限制也就是我们所说的跨域了。那么我们有什么办法可以去除破这个限制。好的,平时我们经常会这样<img src="xxx"/>或者<script src=""><script/>我们会发现,即使这些图片是百度的我们也可以弄过来,(但是现在百度好像也不行了,有些图片,呵呵).我们就可以联想到.json是吧,不是和js差不多嘛。那么我们直接引用一个js然后把内容搞出来不就可以了。

大致方法是这样的。引用一个js,然后加入到head中,这里就需要在js的路径中写入回调函数名称。然后数据就像传回来。但是这里有个问题。我们看见了里面有这么一句话window[callback]。也就是说,回调函数必须挂载到Window对象下,否则不行。

一会继续表单。。。嘿嘿。

时间: 2024-11-05 12:25:57

zepto源码分析------ajax篇的相关文章

zepto源码分析------事件篇

说说zepto.js的源码.今天先分析事件,我比较懒,前面的基础函数就不分析了.直接按模块写就可以了. 在分析之前先简单的说一下zepto的大致结构 var Zepto = (function{})() // 事件模块 (function($){})(zepto) //ajax模块 (function($){})(zepto) // form事件 (function($){})(zepto) 大致结构就是这样了;其实里面的函数也挺简单的,我们都知道zepto这个框架是专为移动端开发,可谓是短小精

zepto源码分析系列

如果你也开发移动端web,如果你也用zepto,应该值得你看看.有问题请留言. Zepto源码分析-架构 Zepto源码分析-zepto(DOM)模块 Zepto源码分析-callbacks模块 Zepto源码分析-event模块 Zepto源码分析-ajax模块 Zepto源码分析-form模块 Zepto源码分析-deferred模块 Zepto源码分析-动画(fx fx_method)模块 内容一定要200字一定要200字内容一定要200字一定要200字内容一定要200字一定要200字内容

Android事件传递机制详解及最新源码分析——ViewGroup篇

在上一篇<Android事件传递机制详解及最新源码分析--View篇>中,详细讲解了View事件的传递机制,没掌握或者掌握不扎实的小伙伴,强烈建议先阅读上一篇. 好了,废话还是少说,直奔主题,开始本篇的ViewGroup事件传递机制探索之旅. 依然从简单的Demo例子现象开始分析 新建安卓工程,首先自定义一个Button以及一个RelativeLayout,很简单,只是重写了主要与事件传递机制相关的方法,代码如下: 自定义WLButton类: 1 public class WLButton e

Zepto源码分析之二~三个API

由于时间关系:本次只对这三个API($.camelCase.$.contains.$.each)方法进行分析 第一个方法变量转驼峰:$.camelCase('hello-world-welcome'); 源码: var camelize; /** * 字符串替换 * 使用replace第二个参数带回调 */ camelize = function(str) { return str.replace(/-+(.)?/g, function(match, chr) { return chr ? ch

JUC源码分析-集合篇(三)ConcurrentLinkedQueue

JUC源码分析-集合篇(三)ConcurrentLinkedQueue 在并发编程中,有时候需要使用线程安全的队列.如果要实现一个线程安全的队列有两种方式:一种是使用阻塞算法,另一种是使用非阻塞算法.使用阻塞算法的队列可以用一个锁(入队和出队用同一把锁)或两个锁(入队和出队用不同的锁)等方式来实现.非阻塞的实现方 式则可以使用循环 CAS 的方式来实现.本节让我们一起来研究一下 Doug Lea 是如何使用非阻塞的方式来实现线程安全队列 ConcurrentLinkedQueue 的,相信从大师

JUC源码分析-集合篇(五)BlockingQueue 阻塞式队列实现原理

JUC源码分析-集合篇(五)BlockingQueue 阻塞式队列实现原理 以 LinkedBlockingQueue 分析 BlockingQueue 阻塞式队列的实现原理. 1. 数据结构 LinkedBlockingQueue 和 ConcurrentLinkedQueue 一样都是由 head 节点和 last 节点组成,每个节点(Node)由节点元素(item)和指向下一个节点(next)的引用组成,节点与节点之间就是通过这个 next 关联起来,从而组成一张链表结构的队列.默认情况下

JUC源码分析-集合篇(七)PriorityBlockingQueue

JUC源码分析-集合篇(七)PriorityBlockingQueue PriorityBlockingQueue 是带优先级的无界阻塞队列,每次出队都返回优先级最高的元素,是二叉树最小堆的实现. PriorityBlockingQueue 数据结构和 PriorityQueue 一致,而线程安全性使用的是 ReentrantLock. 1. 基本属性 // 最大可分配队列容量 Integer.MAX_VALUE - 8,减 8 是因为有的 VM 实现在数组头有些内容 private stati

zepto源码分析-代码结构【转载】

本来想学习一下jQuery的源码,但由于jQuery的源码有10000多行,设计相当复杂,所以决定从zepto开始,分析一个成熟的框架的代码结构及执行步骤. 网上也有很多zepto的源码分析,有的给源码添加注释,有的谈与jQuery的不同,但是都没有系统的讲解zepto框架的代码结构及初始化Zepto对象的过程. 准备 默认你已经对面向对象有一定的了解,本文是边实践边写的,虽有些乱,但好处是为大家提供了分析的思路. 英文文档. 中文文档 注意在文中$变量表示一个函数对象,而$()表示执行函数,他

Spring 事务源码分析——Hibernate篇

在Spring与Hibernate整合的时候,可以利用Spring的事务管理机制,为我们管理事务的开启.提交.回滚等操作.这样的方式极大的减少了我们的代码量,让我们只专注于业务代码的编写.在使用Hibernate的时候,每一个操作都要经历事务开启与提交这样的操作,他们在业务代码的周围,这样来看是不是就想到了AOP编程,把这部分代码抽取出来.没错,Spring正是这样做的,Spring的事务管理就是基于AOP的. 1 Spring的事务隔离与传播 Srping的事务定义了五个隔离等级(isolat