返璞归真 asp.net mvc (10) - asp.net mvc 4.0 新特性之 Web API

简介: 原文:返璞归真 asp.net mvc (10) - asp.net mvc 4.0 新特性之 Web API[索引页][源码下载] 返璞归真 asp.net mvc (10) - asp.net mvc 4.
原文: 返璞归真 asp.net mvc (10) - asp.net mvc 4.0 新特性之 Web API

[索引页]
[源码下载]


返璞归真 asp.net mvc (10) - asp.net mvc 4.0 新特性之 Web API



作者:webabcd


介绍
asp.net mvc 之 asp.net mvc 4.0 新特性之 Web API

  • 开发一个 CRUD 的 Demo,服务端用 Web API,并使其支持 jsonp 协议,客户端用 jQuery



示例
1、自定义一个 JsonMediaTypeFormatter,以支持 jsonp 协议
MyJsonFormatter.cs

/*
 * 自定义一个 JsonMediaTypeFormatter,以支持 jsonp 协议
 */

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Net.Http.Formatting;
using System.Net.Http.Headers;
using System.Threading.Tasks;
using System.Web;

namespace MVC40.Controllers
{
    public class MyJsonFormatter : JsonMediaTypeFormatter
    {
        // jsonp 回调的函数名称
        private string JsonpCallbackFunction;

        public MyJsonFormatter()
        {

        }

        public override bool CanWriteType(Type type)
        {
            return true;
        }

        // 每个请求都先来这里
        public override MediaTypeFormatter GetPerRequestFormatterInstance(Type type, System.Net.Http.HttpRequestMessage request, MediaTypeHeaderValue mediaType)
        {
            var formatter = new MyJsonFormatter()
            {
                JsonpCallbackFunction = GetJsonCallbackFunction(request)
            };

            // 增加一个转换器,以便枚举值与枚举名间的转换
            formatter.SerializerSettings.Converters.Add(new Newtonsoft.Json.Converters.StringEnumConverter());

            // 增加一个转换器,以方便时间格式的序列化和饭序列化
            var dateTimeConverter = new Newtonsoft.Json.Converters.IsoDateTimeConverter();
            dateTimeConverter.DateTimeFormat = "yyyy-MM-dd HH:mm:ss";
            formatter.SerializerSettings.Converters.Add(dateTimeConverter);

            // 排版返回的 json 数据,使其具有缩进格式,以方便裸眼查看
            formatter.SerializerSettings.Formatting = Newtonsoft.Json.Formatting.Indented;

            return formatter;
        }

        // 序列化的实现
        public override Task WriteToStreamAsync(Type type, object value, Stream stream, HttpContent content, TransportContext transportContext)
        {
            if (string.IsNullOrEmpty(JsonpCallbackFunction))
                return base.WriteToStreamAsync(type, value, stream, content, transportContext);

            StreamWriter writer = null;

            try
            {
                writer = new StreamWriter(stream);
                writer.Write(JsonpCallbackFunction + "(");
                writer.Flush();
            }
            catch (Exception ex)
            {
                try
                {
                    if (writer != null)
                        writer.Dispose();
                }
                catch { }

                var tcs = new TaskCompletionSource<object>();
                tcs.SetException(ex);
                return tcs.Task;
            }

            return base.WriteToStreamAsync(type, value, stream, content, transportContext)
                       .ContinueWith(innerTask =>
                            {
                                if (innerTask.Status == TaskStatus.RanToCompletion)
                                {
                                    writer.Write(")");
                                    writer.Flush();
                                }

                            }, TaskContinuationOptions.ExecuteSynchronously)
                        .ContinueWith(innerTask =>
                            {
                                writer.Dispose();
                                return innerTask;

                            }, TaskContinuationOptions.ExecuteSynchronously)
                        .Unwrap();
        }

        // 从请求 url 中获取其参数 callback 的值
        private string GetJsonCallbackFunction(HttpRequestMessage request)
        {
            if (request.Method != HttpMethod.Get)
                return null;

            var query = HttpUtility.ParseQueryString(request.RequestUri.Query);
            var queryVal = query["callback"];

            if (string.IsNullOrEmpty(queryVal))
                return null;

            return queryVal;
        }
    }
}


2、在 Global 中做的一些配置
Global.asax.cs

using MVC40.Controllers;
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Http.Formatting;
using System.Net.Http.Headers;
using System.Web;
using System.Web.Http;
using System.Web.Mvc;
using System.Web.Optimization;
using System.Web.Routing;

namespace MVC40
{
    public class WebApiApplication : System.Web.HttpApplication
    {
        protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();

            WebApiConfig.Register(GlobalConfiguration.Configuration);
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);
            


            // 添加一个转换器 IsoDateTimeConverter,其用于日期数据的序列化和反序列化
            var dateTimeConverter = new IsoDateTimeConverter();
            dateTimeConverter.DateTimeFormat = "yyyy-MM-dd HH:mm:ss";
            JsonSerializerSettings serializerSettings = new JsonSerializerSettings();
            serializerSettings.Converters.Add(dateTimeConverter);

            // 清除全部 Formatter(默认有 4 个,分别是:JsonMediaTypeFormatter, XmlMediaTypeFormatter, FormUrlEncodedMediaTypeFormatter, JQueryMvcFormUrlEncodedFormatter)
            // GlobalConfiguration.Configuration.Formatters.Clear();

            // 如果请求 header 中有 accept: text/html 则返回这个新建的 JsonMediaTypeFormatter 数据
            var jsonFormatter = new JsonMediaTypeFormatter();
            jsonFormatter.SerializerSettings = serializerSettings;
            // jsonFormatter.SupportedMediaTypes.Add(new MediaTypeHeaderValue("text/html")); 
            jsonFormatter.MediaTypeMappings.Add(new RequestHeaderMapping("accept", "text/html", StringComparison.InvariantCultureIgnoreCase, true, new MediaTypeHeaderValue("text/html")));
            GlobalConfiguration.Configuration.Formatters.Insert(0, jsonFormatter);


            // 请求 url 中如果带有参数 xml=true,则返回 xml 数据
            GlobalConfiguration.Configuration.Formatters.XmlFormatter.MediaTypeMappings.Add(new QueryStringMapping("xml", "true", "application/xml"));


            // 请求 url 中如果带有参数 jsonp=true,则返回支持 jsonp 协议的数据(具体实现参见 MyJsonFormatter.cs)
            MyJsonFormatter formatter = new MyJsonFormatter();
            formatter.MediaTypeMappings.Add(new QueryStringMapping("jsonp", "true", "application/javascript"));
            GlobalConfiguration.Configuration.Formatters.Add(formatter);
        }
    }
}

关于项目模版生成的代码的简短说明

<p>
    项目模板更新了,原来堆在 Global.asax.cs 中的配置都分出去了
    <br />
    在 App_Start 文件夹里自动生成的 BundleConfig.cs 用于对多个 css 或 js 做打包和压缩
    <br />
    在 App_Start 文件夹里自动生成的 FilterConfig.cs 用于配置全局的 Action Filter
    <br />
    在 App_Start 文件夹里自动生成的 RouteConfig.cs 用于配置路由
    <br />
    在 App_Start 文件夹里自动生成的 WebApiConfig.cs 用于为 web api 配置路由
</p>

web api 的路由配置
WebApiConfig.cs

/*
 * web api 的路由配置
 */

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web.Http;

namespace MVC40
{
    public static class WebApiConfig
    {
        public static void Register(HttpConfiguration config)
        {
            // 此配置为默认生成的配置
            config.Routes.MapHttpRoute(
                name: "DefaultApi",
                routeTemplate: "api/{controller}/{id}",
                defaults: new { id = RouteParameter.Optional }
            );


            // 由于默认为 web api 生成的路由配置,无 action 配置,所以只能通过 http 方法来匹配 action
            // 如果需要带 action 的路由则使用以下配置即可
            /*
            routes.MapHttpRoute(
                name: "ActionApi",
                routeTemplate: "api/{controller}/{action}/{id}",
                defaults: new { id = RouteParameter.Optional }
            );
            */
        }
    }
}


3、提供 Web API 服务的 Controller(数据层用 Entity Framework 5.0 来实现)
ProductsController.cs

/*
 * ASP.NET Web API
 * 
 * c - POST - 创建
 * r - GET - 读取
 * u - PUT - 更新
 * d - DELETE - 删除
 * 
 * 注:win8 的 iis 默认不会安装 asp.net,需要在“程序和功能”中手动添加
 */

using MVC40.Models;
using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Threading.Tasks;
using System.Web.Http;

namespace MVC40.Controllers
{
    /*
     * Web API 的 Controller 要从 ApiController 继承
     * 
     * 默认:http get 找 controller 的 get(), http post 找 controller 的 post(), http put 找 controller 的 put(), http delete 找 controller 的 delete()
     */
    public class ProductsController : ApiController
    {
        // 获取全部 Product 数据:get http://localhost:17612/api/products
        public IEnumerable<Product> Get()
        {
            NorthwindEntities db = new NorthwindEntities();

            var products = from p in db.Products
                           orderby p.ProductID descending
                           select p;

            return products.ToList();
        }

        // 根据 ProductId 获取指定的 Product 数据:get http://localhost:17612/api/products/3
        public Product Get(int id)
        {
            NorthwindEntities db = new NorthwindEntities();

            var product = db.Products.SingleOrDefault(p => p.ProductID == id);

            return product;
        }

        // 新建 Product:post 一个 product 到 http://localhost:17612/api/products
        public void Post(Product product)
        {
            NorthwindEntities db = new NorthwindEntities();

            db.Products.Add(product);

            db.SaveChanges();

        }

        // 更新 Product:put 一个 product 到 http://localhost:17612/api/products
        public void Put(Product product)
        {
            NorthwindEntities db = new NorthwindEntities();

            db.Products.Attach(product);
            var entry = db.Entry(product);
            entry.State = EntityState.Modified;

            db.SaveChanges();
        }

        // 根据 ProductId 删除指定的 Product 数据:delete http://localhost:17612/api/products/3
        public void Delete(int id)
        {
            NorthwindEntities db = new NorthwindEntities();

            var product = db.Products.SingleOrDefault(p => p.ProductID == id);
            db.Products.Remove(product);

            db.SaveChanges();
        }
    }
}


4、调用 Web API 的客户端(用 jQuery 实现)
CRUDDemo.cshtml

@{
    Layout = null;
}

<!DOCTYPE html>

<html>
<head>
    <title>演示如何通过 jQuery 调用 asp.net web api 做 crud 操作</title>
    <script src="../Scripts/jquery-1.7.1.js" type="text/javascript"></script>
</head>
<body>
    <table id="tblProducts" border="1">
        <tr>
            <th>Product Id</th>
            <th>Product Name</th>
            <th>Unit Price</th>
            <th>Actions</th>
        </tr>
        <tr>
            <td>
                <input type="text" id="txtProductId" size="5" disabled /></td>
            <td>
                <input type="text" id="txtProductName" /></td>
            <td>
                <input type="text" id="txtUnitPrice" /></td>
            <td>
                <input type="button" name="btnInsert" value="Insert" /></td>
        </tr>
    </table>

    <script type="text/javascript">
        $(document).ready(function () {
            loadProductsAsync();
        });

        // 异步获取全部 Product 数据
        function loadProductsAsync() {
            $.getJSON("/api/products?jsonp=true&callback=?", loadProductsCompleted).error(function () { alert('error') });
        }

        // 显示获取到的 Product 数据
        function loadProductsCompleted(data) {
            $("#tblProducts").find("tr:gt(1)").remove();
            $.each(data, function (key, val) {
                var tableRow = '<tr>' +
                                '<td>' + val.ProductID + '</td>' +
                                '<td><input type="text" value="' + val.ProductName + '"/></td>' +
                                '<td><input type="text" value="' + val.UnitPrice + '"/></td>' +
                                '<td><input type="button" name="btnUpdate" value="Update" />  <input type="button" name="btnDelete" value="Delete" /></td>' +
                '</tr>';
                $('#tblProducts').append(tableRow);
            });

            $("input[name='btnInsert']").click(onInsert);
            $("input[name='btnUpdate']").click(onUpdate);
            $("input[name='btnDelete']").click(onDelete);
        }


        // 新增 Product 数据
        function onInsert(evt) {
            var productName = $("#txtProductName").val();
            var unitPrice = $("#txtUnitPrice").val();

            // 构建需要 post 的数据,即需要添加的数据
            var data = '{"ProductName":"' + productName + '","UnitPrice":' + unitPrice + '}';

            $.ajax({
                type: 'POST',
                url: '/api/products/',
                data: data,
                contentType: "application/json; charset=utf-8",
                dataType: 'json',
                success: function (results) {
                    $("#txtProductName").val('');
                    $("#txtUnitPrice").val('');

                    alert('Product Added');

                    loadProductsAsync();
                }
            })
        }


        // 更新 Product 数据
        function onUpdate(evt) {
            var productId = $(this).parent().parent().children().get(0).innerHTML;

            var cell = $(this).parent().parent().children().get(1);
            var productName = $(cell).find('input').val();

            cell = $(this).parent().parent().children().get(2);
            var unitPrice = $(cell).find('input').val();

            // 构建需要 put 的数据,即需要更新的数据
            var data = '{"ProductID":"' + productId + '","ProductName":"' + productName + '","UnitPrice":' + unitPrice + '}';
          
            $.ajax({
                type: 'PUT',
                url: '/api/products/',
                data: data,
                contentType: "application/json; charset=utf-8",
                dataType: 'json',
                success: function (results) {
                    alert('Product Updated');
                }
            })
        }


        // 删除指定的 Product 数据
        function onDelete(evt) {
            var productId = $(this).parent().parent().children().get(0).innerHTML;

            $.ajax({
                type: 'DELETE',
                url: '/api/products/' + productId,
                contentType: "application/json; charset=utf-8",
                dataType: 'json',
                success: function (results) {
                    alert('Product Deleted');

                    loadProductsAsync();
                }
            })

        }
    </script>
</body>
</html>



OK
[源码下载]

目录
相关文章
|
3月前
|
Java API 数据库
构建RESTful API已经成为现代Web开发的标准做法之一。Spring Boot框架因其简洁的配置、快速的启动特性及丰富的功能集而备受开发者青睐。
【10月更文挑战第11天】本文介绍如何使用Spring Boot构建在线图书管理系统的RESTful API。通过创建Spring Boot项目,定义`Book`实体类、`BookRepository`接口和`BookService`服务类,最后实现`BookController`控制器来处理HTTP请求,展示了从基础环境搭建到API测试的完整过程。
63 4
|
3月前
|
XML JSON API
ServiceStack:不仅仅是一个高性能Web API和微服务框架,更是一站式解决方案——深入解析其多协议支持及简便开发流程,带您体验前所未有的.NET开发效率革命
【10月更文挑战第9天】ServiceStack 是一个高性能的 Web API 和微服务框架,支持 JSON、XML、CSV 等多种数据格式。它简化了 .NET 应用的开发流程,提供了直观的 RESTful 服务构建方式。ServiceStack 支持高并发请求和复杂业务逻辑,安装简单,通过 NuGet 包管理器即可快速集成。示例代码展示了如何创建一个返回当前日期的简单服务,包括定义请求和响应 DTO、实现服务逻辑、配置路由和宿主。ServiceStack 还支持 WebSocket、SignalR 等实时通信协议,具备自动验证、自动过滤器等丰富功能,适合快速搭建高性能、可扩展的服务端应用。
171 3
|
22天前
|
Kubernetes 安全 Devops
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
53 10
有效抵御网络应用及API威胁,聊聊F5 BIG-IP Next Web应用防火墙
|
1月前
|
存储 Java 数据挖掘
Java 8 新特性之 Stream API:函数式编程风格的数据处理范式
Java 8 引入的 Stream API 提供了一种新的数据处理方式,支持函数式编程风格,能够高效、简洁地处理集合数据,实现过滤、映射、聚合等操作。
56 6
|
3月前
|
缓存 JavaScript 前端开发
深入理解 Vue 3 的 Composition API 与新特性
本文详细探讨了 Vue 3 中的 Composition API,包括 setup 函数的使用、响应式数据管理(ref、reactive、toRefs 和 toRef)、侦听器(watch 和 watchEffect)以及计算属性(computed)。我们还介绍了自定义 Hooks 的创建与使用,分析了 Vue 2 与 Vue 3 在响应式系统上的重要区别,并概述了组件生命周期钩子、Fragments、Teleport 和 Suspense 等新特性。通过这些内容,读者将能更深入地理解 Vue 3 的设计理念及其在构建现代前端应用中的优势。
52 1
深入理解 Vue 3 的 Composition API 与新特性
|
2月前
|
前端开发 API 开发者
Python Web开发者必看!AJAX、Fetch API实战技巧,让前后端交互如丝般顺滑!
在Web开发中,前后端的高效交互是提升用户体验的关键。本文通过一个基于Flask框架的博客系统实战案例,详细介绍了如何使用AJAX和Fetch API实现不刷新页面查看评论的功能。从后端路由设置到前端请求处理,全面展示了这两种技术的应用技巧,帮助Python Web开发者提升项目质量和开发效率。
57 1
|
2月前
|
JavaScript 前端开发 API
Vue 3新特性详解:Composition API的威力
【10月更文挑战第25天】Vue 3 引入的 Composition API 是一组用于组织和复用组件逻辑的新 API。相比 Options API,它提供了更灵活的结构,便于逻辑复用和代码组织,特别适合复杂组件。本文将探讨 Composition API 的优势,并通过示例代码展示其基本用法,帮助开发者更好地理解和应用这一强大工具。
38 2
|
2月前
|
JSON API 数据格式
如何使用Python和Flask构建一个简单的RESTful API。Flask是一个轻量级的Web框架
本文介绍了如何使用Python和Flask构建一个简单的RESTful API。Flask是一个轻量级的Web框架,适合小型项目和微服务。文章从环境准备、创建基本Flask应用、定义资源和路由、请求和响应处理、错误处理等方面进行了详细说明,并提供了示例代码。通过这些步骤,读者可以快速上手构建自己的RESTful API。
133 2
|
3月前
|
监控 负载均衡 API
Web、RESTful API 在微服务中有哪些作用?
在微服务架构中,Web 和 RESTful API 扮演着至关重要的角色。它们帮助实现服务之间的通信、数据交换和系统的可扩展性。
62 2
|
4月前
|
存储 JavaScript 前端开发
敲黑板!vue3重点!一文了解Composition API新特性:ref、toRef、toRefs
该文章深入探讨了Vue3中Composition API的关键特性,包括`ref`、`toRef`、`toRefs`的使用方法与场景,以及它们如何帮助开发者更好地管理组件状态和促进逻辑复用。
敲黑板!vue3重点!一文了解Composition API新特性:ref、toRef、toRefs