jQuery調用WebService返回JSON數據

jQuery調用WebService返回JSON數據

來源:寧波北大青鳥海曙校區

 

  • .NET Framework 3.5的發佈解決了WebService調用中json問題,本文將介紹jQuery調用基於.NET Framework 3.5的WebService返回JSON數據,另外還要介紹一下用jQuery調用WebService的參數設置及設置不當所出現的問題,還有出現問題的原因

jQuery調用WebService網上的介紹也比較多,最近的項目中我也用到不少,一直都很少用.NET Ajax,比較鍾情於jQuery調用請求WebService有幾種方法,這主要說一下POST與GET方法,其實安全方法考慮不建議使用GET方法,下面就說一下用jquery調用WebService的參數設置及設置不當所出現的問題,還有出現問題的原因。我們這裏只討論返回JSON數據的情況,相信大家都比較瞭解JSON格式的數據對於ajax的方便,不瞭解的可以從網上找一下這方面的資料來看一下,這裏就不多說了,或者我以後再寫一篇這方面的文章。

下面是jQuery調用WebService服務器端代碼:

WS1和WS2方法爲POST方法請求的方法,所以設置UseHttpGet 爲false,WS3與WS4爲GET方法請求的方法,設置UseHttpGet 爲true。

  1. using System.Web.Script.Services; 
  2. using System.Web.Services; 
  3. namespace WebService35 
  4.     ///  
  5.     /// WebService1 的摘要說明 
  6.     ///  
  7.     [WebService(Namespace = "http://tempuri.org/")] 
  8.     [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)] 
  9.     [System.ComponentModel.ToolboxItem(false)] 
  10.     // 若要允許使用 ASP.NET AJAX 從腳本中調用此 Web 服務,請取消對下行的註釋。 
  11.     [System.Web.Script.Services.ScriptService] 
  12.     public class WebService1 : System.Web.Services.WebService 
  13.     { 
  14.         [WebMethod] 
  15.         [ScriptMethod(UseHttpGet = false)] 
  16.         public string WS1() 
  17.         { 
  18.             return "POST無參數"; 
  19.         } 
  20.         [WebMethod] 
  21.         [ScriptMethod(UseHttpGet = false)] 
  22.         public string WS2(string s) 
  23.         { 
  24.             return s; 
  25.         } 
  26.         [WebMethod] 
  27.         [ScriptMethod(UseHttpGet = true)] 
  28.         public string WS3() 
  29.         { 
  30.             return "GET無參數"; 
  31.         } 
  32.         [WebMethod] 
  33.         [ScriptMethod(UseHttpGet = true)] 
  34.         public string WS4(string s) 
  35.         { 
  36.             return s; 
  37.         } 
  38.     }    
  39. }
 
  1. function fun1() { 
  2.            $.ajax({ 
  3.                url: "WebService1.asmx/WS1", 
  4.                type: "POST", 
  5.                dataType: "json", 
  6.                contentType: "application/json; charset=utf-8", 
  7.                data: "", 
  8.                success: function(json) { 
  9.                    alert(json.d); 
  10.                }, 
  11.                error: function(x, e) { 
  12.                    alert(x.responseText); 
  13.                }, 
  14.                complete: function(x) { 
  15.                    alert(x.responseText); 
  16.                } 
  17.            }); 
  18.        }
 

上面的JS方法爲用POST方法請求無參數的WebService方法的代碼,不過以上代碼並不能返回正確的JSON格式的數據,而是返回XML格式的數據,回爲要使WebService返回JSON格式的數據,要在Request Headers中設置Content-Type爲application/json,有人要問了,你不設置了contentType爲 “application/json; charset=utf-8”了嗎?沒錯,是設置了,不過在jquery中,如果Content-Length爲0或者沒有設置,它會忽略你設置的 contentType的,我可以看下面的這個圖,這是抓取的Request Headers的數據,可以看到Content-Length爲0,並且沒有Content-Type,所WebService就不知道我們需要JSON 格式的數據,它就返回了默認的XML格式的數據給我們,之所以爲0,是因爲我沒有提交任何數據。

組織結構

這要怎麼辦呢?繼續看下面的JS代碼,因爲我們這裏是調用的一個沒能參數的WebService方法,所以我們可以提交一個空和JSON對象“{}",如下所示,設置data爲{}。

  1. function fun1() { 
  2.             $.ajax({ 
  3.                 url: "WebService1.asmx/WS1", 
  4.                 type: "POST", 
  5.                 dataType: "json", 
  6.                 contentType: "application/json; charset=utf-8", 
  7.                 data: "{}", 
  8.                 success: function(json) { 
  9.                     alert(json.d); 
  10.                 }, 
  11.                 error: function(x, e) { 
  12.                     alert(x.responseText); 
  13.                 }, 
  14.                 complete: function(x) { 
  15.                     alert(x.responseText); 
  16.                 } 
  17.             }); 
  18.         }

現在我再來看下圖,可以看到,Content-Length已經爲2了,並且也有Contetn-Type,還是我們設置的值,這樣就能正確的返回JSON數據給我們使用了。

返回JSON格式

還有一種方法就是:既然jquery不給我們設置Content-Type,我們可以自己設置,如下面的代碼所示,我們在發送數據之前設置一下Content-Type爲“application/json; charset=utf-8”,這樣就可以了。

  1. function fun1() { 
  2.             $.ajax({ 
  3.                 url: "WebService1.asmx/WS1", 
  4.                 type: "POST", 
  5.                 dataType: "json", 
  6.                 data: "", 
  7.                 beforeSend: function(x) { 
  8.                     x.setRequestHeader("Content-Type", "application/json; charset=utf-8"); 
  9.                 }, 
  10.                 success: function(json) { 
  11.                     alert(json.d); 
  12.                 }, 
  13.                 error: function(x, e) { 
  14.                     alert(x.responseText); 
  15.                 }, 
  16.                 complete: function(x) { 
  17.                     alert(x.responseText); 
  18.                 } 
  19.             }); 
  20.         }

下面是我們手工設置了Content-Type之後抓取的Request Headers,可以看到,即使Content-Length爲0,裏面也有了正確的Content-Type了。

設置了Content-Type之後

不過,需要注意的是:如果我們設置了jquery的contentType,又發送了一個空的JSON對象,並且還手工設置了Content-Type,就如下代碼所示:

  1. function fun1() { 
  2.             $.ajax({ 
  3.                 url: "WebService1.asmx/WS1", 
  4.                 type: "POST", 
  5.                 dataType: "json", 
  6.                 contentType: "application/json; charset=utf-8", 
  7.                 data: "{}", 
  8.                 beforeSend: function(x) { 
  9.                     x.setRequestHeader("Content-Type", "application/json; charset=utf-8"); 
  10.                 }, 
  11.                 success: function(json) { 
  12.                     alert(json.d); 
  13.                 }, 
  14.                 error: function(x, e) { 
  15.                     alert(x.responseText); 
  16.                 }, 
  17.                 complete: function(x) { 
  18.                     alert(x.responseText); 
  19.                 } 
  20.             }); 
  21.         }

那麼在IE發送的Requst Headers就如下圖所示,你會看到Content-Type有兩個用逗號隔開的值,這是爲什麼呢?因爲,jquery爲Content-Type設置了一次值,我們手工又設置一次,而在IE是多次設置Content-Type的值它會追加,而不是替換,不過這並不影響WebService正確返回 JSON數據給我們,不過應該避免這種情況出現。

IE發送的Requst Headers

如果說上面那種設置兩次Content-Type的值還能正確的返回JSON數據,那麼下面代碼就不能正確返回JSON數據了。

  1. function fun1() { 
  2.             $.ajax({ 
  3.                 url: "WebService1.asmx/WS1", 
  4.                 type: "POST", 
  5.                 dataType: "json", 
  6.                 data: "{}", 
  7.                 beforeSend: function(x) { 
  8.                     x.setRequestHeader("Content-Type", "application/json; charset=utf-8"); 
  9.                 }, 
  10.                 success: function(json) { 
  11.                     alert(json.d); 
  12.                 }, 
  13.                 error: function(x, e) { 
  14.                     alert(x.responseText); 
  15.                 }, 
  16.                 complete: function(x) { 
  17.                     alert(x.responseText); 
  18.                 } 
  19.             }); 
  20.         }

從下圖可以看到Content-Type也有兩個值,不過這個和上面那個還有點不一樣,這次呢這個值是不一樣的,一個是 application/x-www-form-urlencoded,一個是application/json; charset=utf-8,這種情況就不能正確的返回JSON格式的數據了。這又是爲什麼呢?這是因爲我們沒有爲jquery設置 contentType爲,並且又提交了一個空的JSON對象,可以爲什麼這樣就會使用Content-Type出現這樣的情況的呢?因爲jquery的 ajax用POST方法提交數據的時候,如果沒有設置contentType,並且所發送的數據不爲空,那麼它就會爲ContentType設置一個默認值,也就是application/x-www-form-urlencoded,所以就會出現這種情況了。

所以呢,在用POST方法請求的時候,如果有提交數據,也就是jquery ajax的datar屬性不空的情況下(不爲空的情況:1.發送一個空對象調用無參數的WebService方法;2.請一個有參數的 WebService方法。),一定要設置contentType屬性,並且不能手工設置Content-Type了。

手工設置Content-Type

下面是請求有參數的WebService方法,一些情況在上面也都說過了,這裏就不多說了。

不過有一點要注意,就是用POST方法請求的時候,不用手工去編碼有漢字的參數值,如下面的data: "{s:'POST有參數'}",就不用寫成data: "{s:"+encodeURI('POST有參數')+"}"了。

  1. function fun2() { 
  2.             $.ajax({ 
  3.                 url: "WebService1.asmx/WS2", 
  4.                 contentType: "application/json; charset=utf-8", 
  5.                 type: "POST", 
  6.                 dataType: "json", 
  7.                 data: "{s:'POST有參數'}", 
  8.                 success: function(json) { 
  9.                     alert(json.d); 
  10.                 }, 
  11.                 error: function(x, e) { 
  12.                     alert(x.responseText); ; 
  13.                 }, 
  14.                 complete: function(x) { 
  15.                     alert(x.responseText); 
  16.                 } 
  17.             }); 
  18.         }

以上是我們說的用POST方法請求,下面是用GET方法請求。

下面是一個用GET方法請求一個無參數的WebService方法,不過這是一段錯誤的代碼,錯在哪兒呢,各們童鞋可以自己想一下,我們下面一起說。

  1. function fun3() { 
  2.      $.ajax({ 
  3.          url: "WebService1.asmx/WS3", 
  4.          type: "GET", 
  5.          dataType: "json", 
  6.          data: "", 
  7.          contentType: "application/json; charset=utf-8", 
  8.          success: function(json) { 
  9.              alert(json.d); 
  10.          }, 
  11.          error: function(x, e) { 
  12.              alert(x.responseText); 
  13.          }, 
  14.          complete: function(x) { 
  15.              alert(x.responseText); 
  16.          } 
  17.      }); 
  18. }

下圖是用上面一段代碼請求所抓取的Request Headers,大家看一下,問題出在哪裏。

請求所抓取的Request Headers

下面的代碼是正確的用GET方法調用無參數的WebService方法。

        function fun3() {
            $.ajax({
                url: "WebService1.asmx/WS3",
                dataType: "json",
                data: "",
                beforeSend: function(x) {
                    x.setRequestHeader("Content-Type", "application/json; charset=utf-8");
                },
                success: function(json) {
                    alert(json.d);
                },
                error: function(x, e) {
                    alert(x.responseText);
                },
                complete: function(x) {
                    alert(x.responseText);
                }
            });
        }

下面的代碼是正確的用GET方法調用有參數的WebService方法。

        function fun4() {
            $.ajax({
                url: "WebService1.asmx/WS4",
                dataType: "json",
                data: encodeURI("s='GET有參數'"),
                beforeSend: function(x) {
                    x.setRequestHeader("Content-Type", "application/json; charset=utf-8");
                },
                success: function(json) {
                    alert(json.d);
                },
                error: function(x, e) {
                    alert(x.responseText);
                },
                complete: function(x) {
                    alert(x.responseText);
                }
            });
        }

下圖是正確的用GET方法(有參數和無參數)調用WebService方法所抓取的Request Headers。

用GET方法

從上圖可以看到,用GET方法請求,不管是有參數還是無參數,都是沒有Content-Length的,所以jquery也就不能爲我們設置 Content-Type了,我只能手工設置Content-Type,所以我們也就沒有必要設置jquery ajax的contentType了。

需要注意的是,GET方法與POST方法不同,有參數的時候,如果參數的值不是ASCII字符,要用encodeURI編一下碼,要不服務端接收到的數據爲亂碼。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章