ajax請求基於restFul的WebApi(post、get、delete、put)

来源:https://www.cnblogs.com/wangjifeng23/archive/2018/12/05/10071880.html
-Advertisement-
Play Games

近日逛招聘軟體,看到部分企業都要求會編寫、請求restFul的webapi。正巧這段時間較為清閑,於是乎打開vs準備開擼。 1.何為restFul? restFul是符合rest架構風格的網路API介面。 rest是一種軟體架構的編碼風格,是根據網路應用而去設計和開發的一種可以降低開發複雜度的編碼方 ...


近日逛招聘軟體,看到部分企業都要會編寫、請求restFul的webapi。正巧這段時間較為清閑,於是乎打開vs準備開擼。

 

1.何為restFul?

restFul是符合rest架構風格的網路API介面。

rest是一種軟體架構的編碼風格,是根據網路應用而去設計和開發的一種可以降低開發複雜度的編碼方式,並且可以提高程式的可伸縮性(增減問題)。

幾種較為常見的操作類型:get(查詢)、post(新增)、put(修改)、delete(刪除)

 

2.restFul標準的WebApi搭建以及部署在iis上

在這裡為了方便,使用的ef框架,在創建讀/寫控制器時直接引用了模型類

 

 

控制器直接幫我幫crud的方法都寫好了,按照註釋的請求規則,可直接使用。代碼如下:

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.Entity;
using System.Data.Entity.Infrastructure;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Web.Http;
using System.Web.Http.Description;
using webapi;

namespace webapi.Controllers
{
    public class ProductsController : ApiController
    {
        private DtoolsEntities db = new DtoolsEntities();

        // GET: api/Products
        [HttpGet]
        public IQueryable<Product> GetProduct()
        {
            return db.Product.OrderByDescending(p => p.AddDate).Take(10);
        }

        // GET: api/Products/5
        [HttpGet]
        [ResponseType(typeof(Product))]
        public IHttpActionResult GetProduct(int id)
        {
            Product product = db.Product.Find(id);
            if (product == null)
            {
                return NotFound();
            }

            return Ok(product);
        }

        // PUT: api/Products/5  update 
        [HttpPut]
        [ResponseType(typeof(void))]
        public IHttpActionResult PutProduct(int id, Product product)
        {
            if (!ModelState.IsValid)
            {
                return BadRequest(ModelState);
            }

            if (id != product.Id)
            {
                return BadRequest();
            }
            product.AddDate = DateTime.Now;
            db.Entry(product).State = EntityState.Modified;

            try
            {
                db.SaveChanges();
            }
            catch (DbUpdateConcurrencyException)
            {
                if (!ProductExists(id))
                {
                    return NotFound();
                }
                else
                {
                    throw;
                }
            }

            return StatusCode(HttpStatusCode.NoContent);
        }


        // POST: api/Products  
        [HttpPost]
        [ResponseType(typeof(Product))]
        public IHttpActionResult PostProduct(Product product)
        {
            if (!ModelState.IsValid)
            {
                return BadRequest(ModelState);
            }
            product.AddDate = DateTime.Now;
            db.Product.Add(product);
            db.SaveChanges();

            return CreatedAtRoute("DefaultApi", new { id = product.Id }, product);
        }

        // DELETE: api/Products/5  
        [HttpDelete]
        [ResponseType(typeof(Product))]
        public IHttpActionResult DeleteProduct(int id)
        {
            Product product = db.Product.Find(id);
            if (product == null)
            {
                return NotFound();
            }

            db.Product.Remove(product);
            db.SaveChanges();

            return Ok(product);
        }

        protected override void Dispose(bool disposing)
        {
            if (disposing)
            {
                db.Dispose();
            }
            base.Dispose(disposing);
        }

        private bool ProductExists(int id)
        {
            return db.Product.Count(e => e.Id == id) > 0;
        }
    }
}

 

每個控制器前根據類型最好指定[HttpGet]  [HttpPost]   [HttpPut]  [HttpDelete],因為伺服器是根據請求類型自動映射匹配控制器名稱,加上特性,避免出錯。

weiapi設置中指定json格式,避免數據類型異常

webapi的搭建基本沒有問題了。接下來就是部署在iis上,這裡不多做描述,不懂如何部署iis可點擊這裡 (會被揍嗎?)

 

3.前臺ajax請求頁面的編寫

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
    <title></title>
    <script src="js/jquery-3.3.1.js"></script>
    <script type="text/javascript">

        //部署在iis上的webapi介面具體請求路徑
        var httpUrl = "http://192.168.0.142:8018/api/Products";

        $(function () {
            $.ajax({
                url: httpUrl,
                type: "GET",
                dataType: "json",
                success: function (data) {
                    $.each(data, function (index, item) {
                        var tr = $("<tr/>");
                        $("<td/>").html(item["ProductName"]).appendTo(tr);
                        $("<td/>").html(item["Brand"]).appendTo(tr);
                        $("<td/>").html(item["ASIN"]).appendTo(tr);
                        $("<td/>").html(item["SKU"]).appendTo(tr);
                        $("<button id ='d' onclick='del(" + item["Id"] + ")'>").html("刪除").appendTo(tr);
                        $("<button id ='u' onclick='update(" + item["Id"] + ")'>").html("更新").appendTo(tr);
                        tr.appendTo("#tab");
                    });
                },
                error: function (XMLHttpRequest, textStatus, errorThrown) {
                    alert(XMLHttpRequest + "," + textStatus + "," + errorThrown);
                }
            });

        });

        //修改
        function update(id) {
            $.ajax({
                url: httpUrl + "?id=" + id,
                type: "Put",
                data: { "id": id, "ProductName": '男士領帶', "Brand": '海瀾之家', "ASIN": 'SAD498AE1', "SKU": '98DA7E9QE-SDAE', "StoreName": '海瀾之家京東自營店' },
                dataType: "json",
                success: function (data) {
                    location.reload();
                }
            })
        }

        //新增
        function add() {
            $.ajax({
                url: httpUrl,
                type: "Post",
                data: { "ProductGuid": newGuid(), "ProductName": '男士襯衫', "Brand": '海瀾之家', "ASIN": 'SAD498AE1', "SKU": '98DA7E9QE-SDAE', "StoreName": '海瀾之家天貓旗艦店' },
                dataType: "json",
                success: function (data) {
                    location.reload();
                }
            })
        }

        //刪除
        function del(id) {
            $.ajax({
                url: httpUrl+"/" + id,
                type: "Delete",
                dataType: "json",
                success: function (data) {
                    location.reload();
                }
            });
        }

        //生成guid
        function newGuid() {
            var guid = "";
            for (var i = 1; i <= 32; i++) {
                var n = Math.floor(Math.random() * 16.0).toString(16);
                guid += n;
                if ((i == 8) || (i == 12) || (i == 16) || (i == 20))
                    guid += "-";
            }
            return guid;
        }
    </script>
</head>
<body>
    <div>
        <table id="tab">
            <tr>
                <th>產品名稱</th>
                <th>產品品牌</th>
                <th>ASIN</th>
                <th>SKU</th>
            </tr>
        </table>
        <button id="add" onclick="add()">add</button>
    </div>

</body>
</html>

 

前端,後端代碼都寫完了。只剩測試了。想都不用想肯定會出問題的,因為涉及到了跨域請求等,接下來就為大家解決問題。

問題一:ajax請求,涉及到cors跨域,請求失敗

  問題介紹及原因分析:

           CORS是一個W3C標準,全稱是”跨域資源共用”。它允許瀏覽器向跨源伺服器,發出XMLHttpRequest請求,基本上目前所有的瀏覽器都實現了CORS標準,其實目前幾乎所有的瀏覽器ajax請求都是基於CORS機制的。
         跨域問題一般發生在Javascript發起AJAX調用,因為瀏覽器對於這種請求,所給予的許可權是較低的,通常只允許調用本域中的資源,除非目標伺服器明確地告知它允許跨域調用。所以,跨域的問題雖然是由瀏覽器的行為產生出來的,但解決的方法卻            是在服務端。因為不可能要求所有客戶端降低安全性。

        解決方案:

                           web.config修改配置文件,使服務端支持跨域請求

 

<system.webServer>
    <httpProtocol>
      <customHeaders>
        <!--伺服器端返回Response header  後接URL或*  表示允許-->
        <add name="Access-Control-Allow-Origin" value="*" />
        <add name="Access-Control-Allow-Headers" value="Content-Type" />
     <!--支持請求的類型--> <add name="Access-Control-Allow-Methods" value="GET, POST, PUT, DELETE, OPTIONS" /> </customHeaders> </httpProtocol> <handlers> <remove name="ExtensionlessUrlHandler-Integrated-4.0" /> <add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="*" type="System.Web.Handlers.TransferRequestHandler" preCondition="integratedMode,runtimeVersionv4.0" /> </handlers> </system.webServer>

 

 

問題二:get、post 可正常請求,put、delete 出現405(Method Not Allowed)  註意:我提交Put的請求,瀏覽器響應的是Request Method:PUT

  問題介紹及原因分析:

           有些人會問:我剛剛在服務端設置了允許put delete請求了,瀏覽器上服務端的響應也看到了支持put delete,為啥服務端還是拒絕呢?

           一切都是iis的WebDAV(Web Distribution Authorization Versioning) Publish惹的禍,WebDAV是基於HTTP協議的拓展,添加了很多Method用於管理伺服器上的文件。若安裝了WebDAV,那麼iis所有的site都會預設使用WebDAV Module與WebDAV Handler。

          WebDAV Handler的預設配置是處理如下 Method:PROPFIND,PROPPATCH,MKCOL,PUT,COPY,DELETE,MOVE,LOCK,UNLOCK。所以瀏覽器發送的put delete請求都會被攔截並交給WebDAV Handler來處理,並且WebDAV Handler會預設拒絕請求

        解決方案:

        既然我們找到了問題所在點,那麼解決方案應然而生,那就是在配置文件里移除ebDAVModule和WebDAVHandler或者在系統中直接移除WebDAV

 <system.webServer>
    <!--以下配置為了讓IIS 7+支持Put/Delete方法-->
    <httpProtocol>
      <customHeaders>
        <!--伺服器端返回Response header  後接URL或*  表示允許-->
        <add name="Access-Control-Allow-Origin" value="*" />
        <add name="Access-Control-Allow-Headers" value="Content-Type" />
        <add name="Access-Control-Allow-Methods" value="GET, POST, PUT, DELETE, OPTIONS" />
      </customHeaders>
    </httpProtocol>
    <validation validateIntegratedModeConfiguration="false"/>
    <handlers>
      <!--移除WebDAV協議-->
      <remove name="WebDAV"/>
      <remove name="ExtensionlessUrlHandler-Integrated-4.0" />
      <!--支持所有方式的請求-->
      <add name="ExtensionlessUrlHandler-Integrated-4.0" path="*." verb="*" type="System.Web.Handlers.TransferRequestHandler" preCondition="integratedMode,runtimeVersionv4.0" />
    </handlers>
    <!--IIS7/7.5上必須加這個配置,否則訪問報錯-->
    <modules  runAllManagedModulesForAllRequests="true">
      <!--移除WebDAV協議-->
      <remove name="WebDAVModule"/>
      <remove name="TelemetryCorrelationHttpModule" />
      <add name="TelemetryCorrelationHttpModule" type="Microsoft.AspNet.TelemetryCorrelation.TelemetryCorrelationHttpModule, Microsoft.AspNet.TelemetryCorrelation" preCondition="integratedMode,managedHandler" />
    </modules>
  </system.webServer>

 

問題三:put delete請求,又出現了405(Method Not Allowed)

  問題介紹及原因分析:

           大家發現沒有,問題二put提交,Request Method就是put,Delete提交,Request Method就是Delete。然而在這裡統統都是OPTIONS。那麼這個OPTIONS到底是個啥呢?

           Preflighted Requests(預檢請求)
           Preflighted Requests是CORS中一種透明伺服器驗證機制。預檢請求首先需要向另外一個功能變數名稱的資源發送一個 HTTP OPTIONS 請求頭,其目的就是為了判斷實際發送的請求是否是安全的
          下麵的2種情況需要進行預檢:
          1、簡單請求,比如使用Content-Type 為 application/xml 或 text/xml 的 POST 請求;
          2、請求中設置自定義頭,比如 X-JSON、X-MENGXIANHUI 等。

          原來put、delete請求如果頭部設置了XMLHttpRequest.setRequestHeader("Content-Type", "application/json")之前還發送了一次預檢請求。

        解決方案:

          既然是多的一次請求,那我們就在服務端過濾掉就好了。

          Global.asac添加以下方法就行了

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Http;
using System.Web.Mvc;
using System.Web.Optimization;
using System.Web.Routing;

namespace webapi
{
    public class WebApiApplication : System.Web.HttpApplication
    {
        protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();
            GlobalConfiguration.Configure(WebApiConfig.Register);
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);
        }

        protected void Application_BeginRequest()
        {
            if (Request.Headers.AllKeys.Contains("Origin") && Request.HttpMethod == "OPTIONS")
            {
                Response.End();
            }
        }
    }
}

 

到這裡,ajax跨域實現RestFul請求,已經是能正常運行了。剩下的只是安全校驗、身份認證、異常記錄等,就能放到生產環境了。這裡就不多做描述了,畢竟博主還是上班族...

如有錯誤,歡迎大家指正~

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. 構造器實例化 spring容器通過bean對應的預設的構造函數來實例化bean。 2. 靜態工廠方式實例化 首先創建一個靜態工廠類,在類中定義一個靜態方法創建實例。 靜態工廠類及靜態方法: xml配置文件 3. 實例工廠方式實例化 該種方式的工廠類中,不再使用靜態方法創建Bean實例,而是採用 ...
  • using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.IO;using System.Text;using System.Web.Script. ...
  • 在 Web 應用程式開發過程中,總是無法避免涉及到文件上傳,這次我們來聊一聊怎麼去實現一個簡單方便可復用文件上傳功能;通過創建自定義綁定模型來實現文件上傳。 ...
  • 這裡貼一段VB的代碼,C#參照也是類似的,讀取C盤字體庫中的字體,判斷字體是否存在,不存在即可使用特殊方法對字體進行安裝 這對部分B/S搭建的軟體還是能起到作用的,畢竟有些軟體設置的字體在系統庫中並不存在,如果不安得話,會出現各種亂碼的情況,所以 才需要這麼做 ...
  • 寫在前面 上一篇文章中我帶著大家進行了許可權部分的極簡設計,也僅僅是一個基本的許可權設計。不過你完全可以基於這套許可權系統設計你的更複雜的許可權系統,當然更複雜的許可權系統要根據你的業務來進行,因為任何脫離實際業務的許可權設計都是耍流氓!今天這篇文章我們就對CMS系統的內容進行設計。同時下篇文章準備帶著大家理解 ...
  • 一、委托是什麼? 委托一般可以看作是持有一個或多個方法的對象。可以把委托看做是對象,其和我們使用一個對象的過程一樣。 聲明->創建委托對象->給委托賦值->調用委托。 關於委托還有另一種理解,我們可以把委托類比為C/C++中的函數指針這一概念。只是委托是類型安全的。函數指針就是 指向函數入口地址的一 ...
  • #region 【通過XDocument的方式將Xml文件遞歸到TreeView控制項中】 //讀取Xml文件(XDocument) //1.載入Xml文件 XDocument document=XDoument.Load("文件名稱.xml"); //2.先獲取跟節點 XElement rootEl ...
  • 本文采用Nginx來實現ASP.NET程式集群化。 準備環境 首先準備Nginx環境,Windows版本下載鏈接:http://nginx.org/en/download.html 解壓後文件格式如下: 修改配置文件 打開 conf 文件夾中的 nginx.conf 配置文件 配置說明: 1.lis ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...