jquery.ajax

  1. 这所有的最终都是通过jQuery.ajax()来完成的。
  2. ajax : function(s) {
  3. //两次继承s,以便在测试中能检测
  4. s = jQuery.extend(true, s, jQuery.extend(true, {},
  5. jQuery.ajaxSettings, s)); ①
  6. var jsonp, jsre = /=\?(&|$)/g, status, data,
  7. type = s.type .toUpperCase();
  8. // 如果不是字符集串就转换在查询字符集串
  9. if (s.data && s.processData && typeof s.data != "string")
  10. s.data = jQuery.param(s.data);
  11. // 构建jsonp请求字符集串。jsonp是跨域请求,要加上callback=?后面将会加函数名
  12. if (s.dataType == "jsonp") { ②
  13. if (type == "GET") {//使get的url包含 callback=?后面将会进行加函数名
  14. if (!s.url.match(jsre))
  15. s.url += (s.url.match(/\?/) ? "&" : "?")
  16. + (s.jsonp || "callback") + "=?";
  17. } // 构建新的s.data,使其包含callback=function name
  18. elseif (!s.data || !s.data.match(jsre))
  19. s.data = (s.data ? s.data + "&" : "") + (s.jsonp||"callback")+ "=?";
  20. s.dataType = "json";
  21. }
  22. //判断是否为jsonp,如果是,进行处理。
  23. if (s.dataType == "json"
  24. && (s.data && s.data.match(jsre) || s.url.match(jsre))) {③
  25. jsonp = "jsonp" + jsc++;
  26. / /为请求字符集串的callback=加上生成回调函数名
  27. if (s.data)s.data = (s.data + "").replace(jsre, "=" + jsonp + "$1");
  28. s.url = s.url.replace(jsre, "=" + jsonp + "$1");
  29. // 我们需要保证jsonp 类型响应能正确地执行
  30. //jsonp的类型必须为script。这样才能执行服务器返回的
  31. //代码。这里就是调用这个回调函数。
  32. s.dataType = "script";
  33. //window下注册一个jsonp回调函数有,让ajax请求返回的代码调用执行它,
  34. //在服务器端我们生成的代码 如callbackname(data);形式传入data.
  35. window[jsonp] = function(tmp) {
  36. data = tmp;success();complete();
  37. // 垃圾回收,释放联变量,删除jsonp的对象,除去head中加的script元素
  38. window[jsonp] = undefined;
  39. try { delete window[jsonp];
  40. } catch (e) { }
  41. if (head) head.removeChild(script);
  42. };
  43. }
  44. if (s.dataType == "script" && s.cache == null) s.cache = false;
  45. // 加上时间戳,可见加cache:false就会加上时间戳
  46. if (s.cache === false && type == "GET") {
  47. var ts = now();
  48. var ret = s.url.replace(/(\?|&)_=.*?(&|$)/, "$1_=" + ts + "$2");
  49. // 没有代替,就追加在url的尾部
  50. s.url = ret+ ((ret == s.url) ? (s.url.match(/\?/) ? "&" : "?") + "_="
  51. + ts : "");
  52. }
  53. // data有效,追加到get类型的url上去
  54. if (s.data && type == "GET") {
  55. s.url += (s.url.match(/\?/) ? "&" : "?") + s.data;
  56. // 防止IE会重复发送get和post data
  57. s.data = null;
  58. }
  59. // 监听一个新的请求
  60. if (s.global && !jQuery.active++) jQuery.event.trigger("ajaxStart");④
  61. // 监听一个绝对的url,和保存domain
  62. var parts = /^(\w+:)?\/\/([^\/?#]+)/.exec(s.url);
  63. // 如果我们正在请求一个远程文档和正在load json或script通过get类型
  64. //location是window的属性,通过和地址栏中的地址比较判断是不是跨域。
  65. if (s.dataType == "script" && type == "GET"&& parts && (parts[1] &&
  66. parts[1] != location.protocol || parts[2] != location.host)) {⑤
  67. // 在head中加上<script src=""></script>
  68. var head = document.getElementsByTagName("head")[0];
  69. var script = document.createElement("script");
  70. script.src = s.url;
  71. if (s.scriptCharset) script.charset = s.scriptCharset;
  72. //如果datatype不是jsonp,但是url却是跨域的。采用scriptr的
  73. //onload或onreadystatechange事件来触发回调函数。
  74. if (!jsonp) {
  75. var done = false;
  76. // 对所有浏览器都加上处理器
  77. script.onload = script.onreadystatechange = function() {
  78. if (!done&& (!this.readyState || this.readyState == "loaded"
  79. || this.readyState == "complete")) {
  80. done = true; success();
  81. complete();head.removeChild(script);
  82. }
  83. };
  84. }
  85. head.appendChild(script);
  86. // 已经使用了script 元素注射来处理所有的事情
  87. return undefined;
  88. }
  89. var requestDone = false;
  90. // 创建request,IE7不能通过XMLHttpRequest来完成,只能通过ActiveXObject
  91. var xhr = window.ActiveXObject ⑥
  92. ? new ActiveXObject("Microsoft.XMLHTTP"): new XMLHttpRequest();
  93. // 创建一个请求的连接,在opera中如果用户名为null会弹出login窗口中。
  94. if (s.username)xhr.open(type, s.url, s.async, s.username, s.password);
  95. else xhr.open(type, s.url, s.async);
  96. // try/catch是为防止FF3在跨域请求时报错
  97. try {// 设定Content-Type ⑦
  98. if (s.data)
  99. xhr.setRequestHeader("Content-Type", s.contentType);
  100. // 设定If-Modified-Since
  101. if (s.ifModified)
  102. xhr.setRequestHeader("If-Modified-Since",
  103. jQuery.lastModified[s.url]|| "Thu, 01 Jan 1970 00:00:00 GMT");
  104. // 这里是为了让服务器能判断这个请求是XMLHttpRequest
  105. xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest");
  106. // 设定 Accepts header 。指能接收的content-type,在服务器端设定
  107. xhr.setRequestHeader("Accept", s.dataType && s.accepts[s.dataType]
  108. ? s.accepts[s.dataType] + ", */*": s.accepts._default);
  109. } catch (e) {}
  110. //拦截方法,我们可以在send之前进行拦截。返回false就不send
  111. if (s.beforeSend && s.beforeSend(xhr, s) === false) { ⑧
  112. // 清除active 请求计数
  113. s.global && jQuery.active--;
  114. xhr.abort();
  115. returnfalse;
  116. }
  117. // 触发全局的ajaxSend事件
  118. if (s.global) jQuery.event.trigger("ajaxSend", [xhr, s]);
  119. // 等待response返回,主要是为后面setInterval用。
  120. var onreadystatechange = function(isTimeout) { ⑨
  121. // 接收成功或请求超时
  122. if (!requestDone && xhr&& (xhr.readyState == 4 ||isTimeout == "timeout")) { requestDone = true;
  123. //清除定时器
  124. if (ival) {clearInterval(ival); ival = null; }
  125. // 分析status:tiemout-->error-->notmodified-->success
  126. status = isTimeout == "timeout" ? "timeout" : !jQuery
  127. ttpSuccess(xhr) ? "error" : s.ifModified&& jQuery.
  128. httpNotModified(xhr, s.url) ? "notmodified": "success";
  129. //如果success且返回了数据,那么分析这些数据
  130. if (status == "success") {
  131. try { data = jQuery.httpData(xhr, s.dataType, s);
  132. } catch (e) { status = "parsererror"; }
  133. }
  134. // 分析数据成功之后,进行last-modified和success的处理。
  135. if (status == "success") {
  136. var modRes;
  137. try {modRes = xhr.getResponseHeader("Last-Modified");
  138. } catch (e) { //FF中如果head取不到,会抛出异常}
  139. //保存last-mordified的标识。
  140. if (s.ifModified && modRes)jQuery.lastModified[s.url] = modRes;
  141. // JSONP 有自己的callback
  142. if (!jsonp) success();
  143. } else // 失败时的处理
  144. jQuery.handleError(s, xhr, status);
  145. // 无论如何都进行cpmplate.timeout和接收成功
  146. complete();
  147. if (s.async) xhr = null; // 防内存泄漏
  148. }
  149. };
  150. if (s.async) {
  151. // 这里是采用poll的方式,不是push的方式
  152. //这里为什么不采用onreadystatechange?
  153. var ival = setInterval(onreadystatechange, 13);
  154. //如果过了timeout还没有请求到,会中断请求的。
  155. if (s.timeout > 0)
  156. setTimeout(function() {
  157. if (xhr) { xhr.abort();
  158. if (!requestDone) onreadystatechange("timeout"); }
  159. }, s.timeout);
  160. }
  161. // 发送
  162. try {xhr.send(s.data); catch(e){jQuery.handleError(s,xhr,null,e);} ⑩
  163. // firefox 1.5 doesn't fire statechange for sync requests
  164. if (!s.async) onreadystatechange();
  165. function success() {
  166. // 调用构建请求对象时指定的success回调。
  167. if (s.success) s.success(data, status);
  168. // 执行全局的回调
  169. if (s.global) jQuery.event.trigger("ajaxSuccess", [xhr, s]);
  170. }
  171. function complete() {
  172. // 本地的回调
  173. if (s.complete) s.complete(xhr, status);
  174. // 执行全局的回调
  175. if (s.global) jQuery.event.trigger("ajaxComplete", [xhr, s]);
  176. // 全局的ajax计数器
  177. if (s.global && !--jQuery.active)jQuery.event.trigger("ajaxStop");
  178. }
  179. // return XMLHttpRequest便进行about()或其它操作.
  180. return xhr;
  181. },
  182. Jquery.ajax是大包大揽的非常复杂的一个方法。它并没有像其它的lib一样,把每个小部分都分开来。它是整个都整在一个函数中。看起来很多,实际上上也没有脱离前面所说的ajax的请求的五步。它的很大一部分代码在处理跨域请求的处理上。下面就分别就ajax的代码进行分析。
  183. ajaxSettings
  184. 在①处通过继承的方式把传入参数s和默认的jQuery.ajaxSettings都clone到s变量中。S的同名属性会覆盖jQuery.ajaxSettings的同名属性。这里两次继承s,以便在测试中能检测。
  185. //默认的ajax的请求参数
  186. ajaxSettings : {
  187. url : location.href,//默认是地址栏中url
  188. global : true,//默认支持全局的ajax事件
  189. type : "GET",
  190. timeout : 0,
  191. contentType : "application/x-www-form-urlencoded",
  192. processData : true,
  193. async : true,
  194. data : null,
  195. username : null,
  196. password : null,
  197. accepts : {
  198. xml : "application/xml, text/xml",
  199. html : "text/html",
  200. script : "text/javascript, application/javascript",
  201. json : "application/json, text/javascript",
  202. text : "text/plain",
  203. _default : "*/*"
  204. }
  205. 这是默认的ajax的设定,我们要在参数s设定同名的属性来覆盖这些属性。但是我们不能覆盖accepts。这个会在后面的代码用到。我们可以通过设定s.dataType等于accepts中的某一个属性key指定请求的data类型,如xml,html,script,json,text。dataType还支持默认的_default和跨域的jsonp。不过其最终会解析成script。
  206. scriptTag
  207. ②~⑥是处理跨域请求的部分。对于dataType为jsonp的类型,给其请求的字符串(可能是s.data)加上callback=callbackfn的key/value串,然后在window下注册一个callbackfn的函数。这个函数的形式如callbackfn(data){ data = tmp;success();complete();}。它代理了通过ajax(s)的传入s参数中success();complete()的功能。它就是调用这个函数,实际上是调用success();complete()的函数。
  208. 那么怎么调用呢?ajax不支持跨域。在⑤处,我们可以看到这里是采用scriptTag的方式来完成。先在页面的<head>中添加一个<script src=url />的标签。因为在<head>中。浏览器会自动载入并运行请求返回的script。如果是jsonp的形式,服务器端还要动态生成的content-type为script的代码:callbackfn(data);只有这样才会调用在window中注册的函数callbackfn。同时传入所需要的参数。
  209. 如dataType == "script"形式的跨域,那只能是通过script.onload 或 script.onreadystatechange事件来触发回调。这里我们可以通过服务器返回的script代码:var data=xxx。来传递参数给s.success();s.complete()。Jquery这里采用是全局变量data来进行操作的。
  210. Ajax Event
  211. ④是采用了jQuery.event.trigger("ajaxStart");来触发全局的ajaxStart事件。这也是说只要注册了这个事件的元素,在任何的ajax的请求时ajaxStart都会执行元素注册的事件处理函数。这和Ext的事件有点相似。但是它不是全局的。
  212. jQuery.each("ajaxStart,ajaxStop,ajaxComplete,ajaxError,ajaxSuccess,ajaxSend".split(","), function(i, o) {
  213. jQuery.fn[o] = function(f) {// f:function
  214. returnthis.bind(o, f);
  215. };
  216. 上面的代码是为jquery对象注册了ajaxStart,ajaxStop,ajaxComplete,ajaxError,ajaxSuccess,ajaxSend这几种ajax的事件方法,在jquery.ajax中不同的时刻都会触发这些事件。当然我们也可以采用s.global=false来设定不触发这些事件。
  217. 因为这是全局的,个人认为其设计的目的就是为了在这些时候能以某种形式来告诉用户ajax的进行的状态。如在ajaxstart的时候,我们可能通过一个topest的div层(加上遮罩的效果)的元素注册一个ajaxstart事件的处理方法。该方法就是显示这个层和显示“你的数据正在提交。。。”这个的提示。这是这6种事件的最佳用法了。
  218. 如果进行私有处理,那么要在事件的处理函数中进行判断。因为每个事件处理函数的第二参数是jquery.ajax(s)的s参数。我们可以在这个参数中做私有的标识,如eventType:xxx。每类不同的请求有不同的eventType值。在事件处理函数再根据这个eventType==xxx进行判断,从而进行私有的处理。如果有大量的这样的私有处理也是会影响ajax的效率的。
  219. setRequestHeader
  220. ⑥处是创建一个xhr对象并通过open来创建一个连接(socket)。
  221. ⑦处是设定请求的头部(setRequestHeader)。如果data的存在的话,那就得设定Content-Type,便于服务器按一定的规则来解码。可以看出post的方式通过data传递数据要安全一点。
  222. 那么服务器如果区别这个请求是ajax呢?因为同步和异步ajax的请求的头文件是一样的。我们如果通过X-Requested-With"="XMLHttpRequest”来标识这个请求是ajax的请求。如果服务器硬是要区分的话,就可以通过获取该头部来判断。
  223. 在头部的定义中,还可能通过Accept来指定接受的数据的类型,如application/xml, text/xml", "text/html", "text/javascript, 等等。
  224. 头部还有一个If-Modified-Since的属性用来提高效率的。它和”Last-Modified配合起来使用。在浏览器第一次请求某一个URL时,服务器端的返回状态会是200,内容是你请求的资源,同时有一个Last-Modified的属性标记此文件在服务期端最后被修改的时间,格式类似这样:Last-Modified: Fri, 12 May 2006 18:53:33 GMT
  225. 客户端第二次请求此URL时,根据 HTTP 协议的规定,浏览器会向服务器传送 If-Modified-Since 报头,询问该时间之后文件是否有被修改过: If-Modified-Since: Fri, 12 May 2006 18:53:33 GMT 如果服务器端的资源没有变化,则自动返回 HTTP 304 (Not Changed.)状态码,内容为空,这样就节省了传输数据量。
  226. 当服务器端代码发生改变或者重启服务器时,则重新发出资源,返回和第一次请求时类似。从而保证不向客户端重复发出资源,也保证当服务器有变化时,客户端能够得到最新的资源。
  227. 拦截处理
  228. ⑧处是一个send之前的拦截处理,可以通过s. beforeSend(xhr, s)函数的形式传入拦截函数。保证在发送之前确保满足某些条件。在取得返回数据的时候,也可以通过s.dataFilter(data, type);形式来拦截处理data。不过这里主要的作用对data进一步的筛选。
  229. onreadystatechange
  230. ⑨处是onreadystatechange的回调处理。这里采用是poll的形式进行处理。它把返回的状态分成status:tiemout-->error-->notmodified-->success—>parsererror这几种。如果status == "success"那么分析这些数据之后再进行last-modified相关的处理。为了不取回没有修改过数据。
  231. 分析数据的代码如下:
  232. //处理请求返回的数据
  233. httpData : function(xhr, type, s) {
  234. var ct = xhr.getResponseHeader("content-type"),
  235. xml = type == "xml" || !type && ct && ct.indexOf("xml") >= 0,
  236. data = xml? xhr.responseXML : xhr.responseText;
  237. if (xml && data.documentElement.tagName == "parsererror")
  238. throw "parsererror";
  239. //允许一个pre-filtering函数清洁repsonse
  240. if (s && s.dataFilter)
  241. data = s.dataFilter(data, type);
  242. //script时,就运行
  243. if (type == "script") jQuery.globalEval(data);
  244. //json,生成json对象。
  245. if (type == "json") data = eval("(" + data + ")");
  246. return data;
  247. },
  248. 如果返回的content-type是xml,html,text等都返回。对script执行jQuery.globalEval来执行它。对于Json类型,通过eval来生成返回的json对象。
  249. // 在全局的范围eval 代码,也就是在<head></head>中
  250. globalEval : function(data) {
  251. data = jQuery.trim(data);
  252. if (data) {
  253. // Inspired by code by Andrea Giammarchi
  254. // http://webreflection.blogspot.com/2007/08/
  255. //global-scope-evaluation-and-dom.html
  256. var head = document.getElementsByTagName("head")[0]
  257. || document.documentElement,
  258. script = document.createElement("script");
  259. script.type = "text/javascript";
  260. if (jQuery.browser.msie) script.text = data;
  261. else script.appendChild(document.createTextNode(data));
  262. // Use insertBefore instead of appendChild to circumvent an IE6
  263. // bug. This arises when a base node is used (#2709).
  264. head.insertBefore(script, head.firstChild);
  265. head.removeChild(script);
  266. } },