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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...