zoukankan      html  css  js  c++  java
  • Zepto源码分析-ajax模块

    源码注释

    //     Zepto.js
    //     (c) 2010-2015 Thomas Fuchs
    //     Zepto.js may be freely distributed under the MIT license.
    
    ;(function($){
      var jsonpID = 0,
          document = window.document,
          key,
          name,
          rscript = /<script[^<]*(?:(?!</script>)<[^<]*)*</script>/gi,
          scriptTypeRE = /^(?:text|application)/javascript/i,
          xmlTypeRE = /^(?:text|application)/xml/i,
          jsonType = 'application/json',       //各种响应的contentType类型
          htmlType = 'text/html',
          blankRE = /^s*$/,
          originAnchor = document.createElement('a')
    
      originAnchor.href = window.location.href
    
      // trigger a custom event and return false if it was cancelled
        /**
         *
         * 触发自定义事件
         * @param context
         * @param eventName
         * @param data
         * @returns {boolean}
         */
      function triggerAndReturn(context, eventName, data) {
        var event = $.Event(eventName)   //创建Event对象
        $(context).trigger(event, data)   //触发
        return !event.isDefaultPrevented()   //TODO:返回event.isDefaultPrevented?
      }
    
      // trigger an Ajax "global" event
    
        /**
         * 触发ajax的全局事件
         * @param settings
         * @param context
         * @param eventName
         * @param data
         * @returns {*}
         */
      function triggerGlobal(settings, context, eventName, data) {
          //settings.global   是否触发ajax的全局事件
        if (settings.global) return triggerAndReturn(context || document, eventName, data)
      }
    
      // Number of active Ajax requests
        //统计ajax request数量,用于相关全局事件 ajaxStart ajaxStop的计数
      $.active = 0
    
    
        /**
         *触发全局 ‘ajaxStart’事件
         * @param settings
         */
      function ajaxStart(settings) {
            //settings.global 传递进来的是否触发全局事件参数
            //$.active++ === 0      $.active = 0,此判断才会true
        if (settings.global && $.active++ === 0) triggerGlobal(settings, null, 'ajaxStart')
      }
        /**
         * 尝试抛出所有请求停止事件,写法和ajaxStart相同
         * @param settings
         */
      function ajaxStop(settings) {
        if (settings.global && !(--$.active)) triggerGlobal(settings, null, 'ajaxStop')
      }
    
      // triggers an extra global event "ajaxBeforeSend" that's like "ajaxSend" but cancelable
        //触发全局ajaxBeforeSend事件,如果返回false,则取消此次请求
    
        /**
         * 请求前置器,  beforeSend ,返回false,取消此次请求
         * 或抛出 ajaxBeforeSend 全局事件
           抛出ajaxSend事件
        *
         * @param xhr
         * @param settings
         * @returns {boolean}
         */
      function ajaxBeforeSend(xhr, settings) {
        var context = settings.context
        if (settings.beforeSend.call(context, xhr, settings) === false ||
            triggerGlobal(settings, context, 'ajaxBeforeSend', [xhr, settings]) === false)
          return false
    
        triggerGlobal(settings, context, 'ajaxSend', [xhr, settings])
      }
    
        /**
         * 请求成功调用函数
         * @param data
         * @param xhr
         * @param settings
         * @param deferred
         */
      function ajaxSuccess(data, xhr, settings, deferred) {
        var context = settings.context, status = 'success'
    
        //调用success函数
        settings.success.call(context, data, status, xhr)
    
        //调用所有成功回调函数
        if (deferred) deferred.resolveWith(context, [data, status, xhr])
    
         //抛出全局事件  'ajaxSuccess'
        triggerGlobal(settings, context, 'ajaxSuccess', [xhr, settings, data])
    
    
        //请求完成
        ajaxComplete(status, xhr, settings)
      }
      // type: "timeout", "error", "abort", "parsererror"
        /**
         * 请求失败调用函数
         * @param error
         * @param type
         * @param xhr
         * @param settings
         * @param deferred
         */
      function ajaxError(error, type, xhr, settings, deferred) {
        var context = settings.context
        settings.error.call(context, xhr, type, error)
        if (deferred) deferred.rejectWith(context, [xhr, type, error])
        triggerGlobal(settings, context, 'ajaxError', [xhr, settings, error || type])
        ajaxComplete(type, xhr, settings)
      }
      // status: "success", "notmodified", "error", "timeout", "abort", "parsererror"
        /**
         * 请求完成调用函数
         * @param status
         * @param xhr
         * @param settings
         */
      function ajaxComplete(status, xhr, settings) {
        var context = settings.context
    
            //执行complete
        settings.complete.call(context, xhr, status)
        triggerGlobal(settings, context, 'ajaxComplete', [xhr, settings])
    
            //尝试抛出所有请求停止事件
        ajaxStop(settings)
      }
    
      // Empty function, used as default callback
      function empty() {}
    
    
        /**
         * jsonp请求
         * @param options
         * @param deferred
         * @returns {*}
         */
      $.ajaxJSONP = function(options, deferred){
          //未设置type,就走     ajax     让参数初始化.
          //如直接调用ajaxJSONP,type未设置
           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,
    
          //中断请求,抛出error事件
        //这里不一定能中断script的加载,但在下面阻止回调函数的执行
         abort = function(errorType) {
            $(script).triggerHandler('error', errorType || 'abort')
          },
          xhr = { abort: abort }, abortTimeout
    
          //xhr为只读deferred
        if (deferred) deferred.promise(xhr)
    
          //监听加载完,加载出错事件
        $(script).on('load error', function(e, errorType){
            //清除超时设置timeout
          clearTimeout(abortTimeout)
    
            //删除加载用的script。因为已加载完了
          $(script).off().remove()
    
            //错误调用error
          if (e.type == 'error' || !responseData) {
              ajaxError(null, errorType || 'error', xhr, options, deferred)
          } else {
              //成功调用success
            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)
    
          //超时处理,通过setTimeout延时处理
        if (options.timeout > 0) abortTimeout = setTimeout(function(){
          abort('timeout')
        }, options.timeout)
    
        return xhr
      }
    
    
      /**
       * ajax 必传的默认值
       */
      $.ajaxSettings = {
        // Default type of request
        type: 'GET',
        // Callback that is executed before request
        beforeSend: empty,
        // Callback that is executed if the request succeeds
        success: empty,
        // Callback that is executed the the server drops error
        error: empty,
        // Callback that is executed on request complete (both: error and success)
        complete: empty,
        // The context for the callbacks
        context: null,
        // Whether to trigger "global" Ajax events
        global: true,
        // Transport
        xhr: function () {
          return new window.XMLHttpRequest()
        },
        // MIME types mapping
        // IIS returns Javascript as "application/x-javascript"
          //媒体数据源,简写对应实际写法
        accepts: {
          script: 'text/javascript, application/javascript, application/x-javascript',
          json:   jsonType,
          xml:    'application/xml, text/xml',
          html:   htmlType,
          text:   'text/plain'
        },
        // Whether the request is to another domain
        crossDomain: false,
        // Default timeout
        timeout: 0,
        // Whether data should be serialized to string
        processData: true,
        // Whether the browser should be allowed to cache GET responses
        cache: true
    
      }
    
    
        /**
         * 根据响应回来的媒体类型,转换成易读的类型  html,json,scirpt,xml,text等
         * @param mime
         * @returns {*|string|string}
         */
      function mimeToDataType(mime) {
        if (mime) mime = mime.split(';', 2)[0]
        return mime && ( mime == htmlType ? 'html' :
          mime == jsonType ? 'json' :
          scriptTypeRE.test(mime) ? 'script' :
          xmlTypeRE.test(mime) && 'xml' ) || 'text'
      }
    
        /**
         * 将查询参数追加到URL后面
         * @param url
         * @param query  查询参数
         * @returns {*}
         */
      function appendQuery(url, query) {
        if (query == '') return url
    
            //replace(/[&?]{1,2}/, '?') 匹配到的第一个[&?]{1,2} 替换成?
        return (url + '&' + query).replace(/[&?]{1,2}/, '?')
      }
    
      // serialize payload and append it to the URL for GET requests
    
      /**
       * 序列化
       * 针对options.data 转换成 a=b&c=1
       */
      function serializeData(options) {
          //options.processData: 对于非get请求,是否将请求参数options.data转换为字符串
        if (options.processData && options.data && $.type(options.data) != "string")
    
          //将data数据序列化为字符串,	 转换成 a=b&c=1
          // options.traditional
          options.data = $.param(options.data, options.traditional)
    
    //      get请求,将序列化的数据追加到url后面
        if (options.data && (!options.type || options.type.toUpperCase() == 'GET'))
          options.url = appendQuery(options.url, options.data), options.data = undefined
      }
    
        /**
         * ajax 请求
         */
      $.ajax = function(options){
        var settings = $.extend({}, options || {}), //创建新的options对象,不影响options的值
            deferred = $.Deferred && $.Deferred(),   //设置异步队列
            urlAnchor, hashIndex
    
          //未传 $.ajaxSettings里的值,复制$.ajaxSettings的值
        for (key in $.ajaxSettings) if (settings[key] === undefined) settings[key] = $.ajaxSettings[key]
    
         //触发全局事件 'ajaxStart'
        ajaxStart(settings)
    
          //是否设置了跨域,未设置,需通过ip  协议 端口一致来判断跨域
        if (!settings.crossDomain) {
          urlAnchor = document.createElement('a')
            //如果没有设置请求地址,则取当前页面地址
          urlAnchor.href = settings.url
          // cleans up URL for .href (IE only), see https://github.com/madrobby/zepto/pull/1049
          urlAnchor.href = urlAnchor.href
    
            //通过ip  协议 端口来判断跨域  location.host = host:port
          settings.crossDomain = (originAnchor.protocol + '//' + originAnchor.host) !== (urlAnchor.protocol + '//' + urlAnchor.host)
        }
    
        //未设置url,取当前地址栏
        if (!settings.url) settings.url = window.location.toString()
    
          //如果有hash,截掉hash,因为hash     ajax不会传递到后台
          if ((hashIndex = settings.url.indexOf('#')) > -1) settings.url = settings.url.slice(0, hashIndex)
    
    
          //将data进行转换
        serializeData(settings)
    
          //TODO: /?.+=?/.test(settings.url)      有xxx.html?a=1?=cccc类似形式,为jsonp
          var dataType = settings.dataType, hasPlaceholder = /?.+=?/.test(settings.url)
        if (hasPlaceholder) dataType = 'jsonp'
    
          //不设置缓存,加时间戳 '_=' + Date.now()
        if (settings.cache === false || (
             (!options || options.cache !== true) &&
             ('script' == dataType || 'jsonp' == dataType)
            ))
          settings.url = appendQuery(settings.url, '_=' + Date.now())
    
          //如果是jsonp,调用$.ajaxJSONP,不走XHR,走script
        if ('jsonp' == dataType) {
          if (!hasPlaceholder)
            settings.url = appendQuery(settings.url,
              settings.jsonp ? (settings.jsonp + '=?') : settings.jsonp === false ? '' : 'callback=?')
          return $.ajaxJSONP(settings, deferred)
        }
    
        var mime = settings.accepts[dataType], //媒体类型
            headers = { },
            setHeader = function(name, value) { headers[name.toLowerCase()] = [name, value] }, //设置请求头的方法
           //如果URL没协议,读取本地URL的协议
            protocol = /^([w-]+:)///.test(settings.url) ? RegExp.$1 : window.location.protocol,
            xhr = settings.xhr(),
            nativeSetHeader = xhr.setRequestHeader,
            abortTimeout
    
          //将xhr设为只读Deferred对象,不能更改状态
        if (deferred) deferred.promise(xhr)
    
          //如果没有跨域
          // x-requested-with  XMLHttpRequest  //表明是AJax异步
          //x-requested-with  null//表明同步,浏览器工具栏未显示,在后台request可以获取到
        if (!settings.crossDomain) setHeader('X-Requested-With', 'XMLHttpRequest')
        setHeader('Accept', mime || '*/*')  //默认接受任何类型
        if (mime = settings.mimeType || mime) {
            //媒体数据源里对应多个,如 script: 'text/javascript, application/javascript, application/x-javascript',
            //设置为最新的写法, text/javascript等都是老浏览废弃的写法
          if (mime.indexOf(',') > -1) mime = mime.split(',', 2)[0]
    
            //对Mozilla的修正
            // 来自服务器的响应没有 XML mime-type 头部(header),则一些版本的 Mozilla浏览器不能正常运行。
            // 对于这种情况,xhr.overrideMimeType(mime); 语句将覆盖发送给服务器的头部,强制mime 作为 mime-type。*/
          xhr.overrideMimeType && xhr.overrideMimeType(mime)
        }
          //如果不是Get请求,设置Content-Type
          //Content-Type: 内容类型  指定响应的 HTTP内容类型。决定浏览器将以什么形式、什么编码读取这个文件.  如果未指定 ContentType,默认为TEXT/HTML。
          /**
          application/x-www-form-urlencoded:是一种编码格式,窗体数据被编码为名称/值对,是标准的编码格式。
    当action为get时候,浏览器用x-www-form-urlencoded的编码方式把form数据转换成一个字串(name1=value1&name2=value2...),然后把这个字串append到url后面,用?分割,加载这个新的url。 当action为post时候,浏览器把form数据封装到http body中,然后发送到server
          **/
          /**
           * 如果有 type=file的话,需要设为multipart/form-data了。浏览器会把整个表单以控件为单位分割,并为每个部分加上 Content-Disposition(form-data或者file),Content-Type(默认为text/plain),name(控件 name)等信息,并加上分割符(boundary)。
           */
    
           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(){
            /**
             * 0:请求未初始化(还没有调用 open())。
                 1:请求已经建立,但是还没有发送(还没有调用 send())。
                 2:请求已发送,正在处理中(通常现在可以从响应中获取内容头)。
                 3:请求在处理中;通常响应中已有部分数据可用了,但是服务器还没有完成响应的生成。
                 4:响应已完成;您可以获取并使用服务器的响应了。
             */
          if (xhr.readyState == 4) {
            xhr.onreadystatechange = empty
            clearTimeout(abortTimeout)   //清除超时
            var result, error = false
    
    
              //根据状态来判断请求是否成功
              //>=200 && < 300 表示成功
             //304 文件未修改 成功
              //xhr.status == 0 && protocol == 'file:'  未请求,打开的本地文件,非localhost  ip形式
            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/
                  //(1,eval)(result)  (1,eval)这是一个典型的逗号操作符,返回最右边的值
                  // (1,eval)  eval 的区别是:前者是一个值,不可以再覆盖。后者是变量,如var a = 1; (1,a) = 1;    会报错;
                  // (1,eval)(result)  eval(result) 的区别是:前者变成值后,只能读取window域下的变量。而后者,遵循作用域链,从局部变量上溯到window域
                  //显然 (1,eval)(result)  避免了作用域链的上溯操作,性能稍好
                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 }
    
                //解析出错,抛出 'parsererror'事件
              if (error) ajaxError(error, 'parsererror', xhr, settings, deferred)
    
              //执行success
              else ajaxSuccess(result, xhr, settings, deferred)
            } else {
                //如果请求出错
                // xhr.status = 0 / null   执行abort,  其他 执行error
              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
        }
    
          // xhrFields 设置  如设置跨域凭证 withCredentials
        if (settings.xhrFields) for (name in settings.xhrFields) xhr[name] = settings.xhrFields[name]
    
          //'async' in settings  ajaxSetting未设置过async为false,设置过,包括null,都为true
        var async = 'async' in settings ? settings.async : true
          //准备xhr请求
        xhr.open(settings.type, settings.url, async, settings.username, settings.password)
    
          //设置请求头
        for (name in headers) nativeSetHeader.apply(xhr, headers[name])
    
    
          //超时处理:设置了settings.timeout,超时后调用xhr.abort()中断请求
        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)
          // 一般来说 post是settings.data  get为null  因为get的查询参数和url一起
        xhr.send(settings.data ? settings.data : null)
        return xhr
      }
    
      // handle optional data/success arguments
        /**
         * 参数转换成ajax格式
         * @param url
         * @param data
         * @param success
         * @param dataType
         * @returns {{url: *, data: *, success: *, dataType: *}}
         */
      function parseArguments(url, data, success, dataType) {
        if ($.isFunction(data)) dataType = success, success = data, data = undefined     //如果data是function,则认为它是请求成功后的回调
        if (!$.isFunction(success)) dataType = success, success = undefined
        return {
          url: url
        , data: data      //如果data不是function实例
        , success: success
        , dataType: dataType
        }
      }
    
        /**
         * 便捷方法 get请求
         * @returns {*}
         */
      $.get = function(/* url, data, success, dataType */){
        return $.ajax(parseArguments.apply(null, arguments))
      }
    
        /**
         * 便捷方法 post请求
         * @returns {*}
         */
      $.post = function(/* url, data, success, dataType */){
        var options = parseArguments.apply(null, arguments)
        options.type = 'POST'
        return $.ajax(options)
      }
    
        /**
         *  便捷方法 响应数据类型为JSON
         * content-type: 'application/json'
         * @returns {*}
         */
      $.getJSON = function(/* url, data, success */){
        var options = parseArguments.apply(null, arguments)
        options.dataType = 'json'
        return $.ajax(options)
      }
    
    
        /**
         * 载入远程 HTML 文件代码并插入至 DOM 中
         * @param url    HTML 网页网址    可以指定选择符,来筛选载入的 HTML 文档,DOM 中将仅插入筛选出的 HTML 代码。语法形如 "url #some > selector"。
         * @param data    发送至服务器的 key/value 数据
         * @param success 载入成功时回调函数
         * @returns {*}
         */
      $.fn.load = function(url, data, success){
        if (!this.length) return this
    
        var self = this, parts = url.split(/s/), selector,
            options = parseArguments(url, data, success),
            callback = options.success
    
          //parts.length > 1 代表url后面有选择符selector
        if (parts.length > 1) options.url = parts[0], selector = parts[1]
    
    
        options.success = function(response){
           // response.replace(rscript, "") 过滤出script标签
            //$('<div>').html(response.replace(rscript, ""))  innerHTML方式转换成DOM
          self.html(selector ?
            $('<div>').html(response.replace(rscript, "")).find(selector)
            : response)
    
            //执行回调
          callback && callback.apply(self, arguments)
        }
        $.ajax(options)
        return this
      }
    
        //URI编码方法。原生的escape/unescape已被废弃。使用encodeURIComponent/decodeURIComponent
      var escape = encodeURIComponent
    
        /**
         *  序列化
         * @param params      结果数组
         * @param obj      数组会按照name/value对进行序列化,普通对象按照key/value对进行序列化
         * @param traditional    是否使用传统的方式浅层序列化
         * @param scope     和traditional=true一起使用。递归时,标记原始key。仅本身递归使用参数
         */
      function serialize(params, obj, traditional, scope){
        var type, array = $.isArray(obj), hash = $.isPlainObject(obj)
    
        $.each(obj, function(key, value) {
          type = $.type(value)
    
            //如果是递归,scope有原始key值。key值修正
          if (scope) key = traditional ? scope :
            scope + '[' + (hash || type == 'object' || type == 'array' ? key : '') + ']'
          // handle data in serializeArray() format
            //当处理的数据为[{},{},{}]这种情况的时候,一般指的是序列化表单后的结果
    
            //如果是表单
          if (!scope && array) params.add(value.name, value.value)
    
          // recurse into nested objects
          //value是数组或对象,traditional为false,需要深层序列化,继续递归序列化。
          //在这里标记scope值
          else if (type == "array" || (!traditional && type == "object"))
            serialize(params, value, traditional, key)
    
          //默认 obj属性赋值到params
          else params.add(key, value)
        })
      }
    
    
        /**
         * 将表单元素数组或者对象序列化
         * @param obj          数组会按照name/value对进行序列化,普通对象按照key/value对进行序列化
         * @param traditional 是否使用传统的方式浅层序列化
         * @returns {string}
         */
      $.param = function(obj, traditional){
        var params = []
    
        //URI编码后添加到数组里
        params.add = function(key, value) {
          if ($.isFunction(value)) value = value()
          if (value == null) value = ""
    
            //encodeURIComponent       编码
            this.push(escape(key) + '=' + escape(value))
        }
        serialize(params, obj, traditional)
        return params.join('&').replace(/%20/g, '+')
      }
    })(Zepto)
    

      

    方法图

  • 相关阅读:
    FactoryBean的作用
    ztree点击文字勾选checkbox,radio实现方法
    js判断字符长度 汉字算两个字符
    双系统引导修复
    thinkpadT440p
    分布式服务器集群
    Eclipse插件安装方式
    用SourceTree轻巧Git项目图解
    廖雪峰git使用完整教程
    hessian
  • 原文地址:https://www.cnblogs.com/mominger/p/4398982.html
Copyright © 2011-2022 走看看