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对象下,否则不行。

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

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。