WEB API系列(一):WEB API的适用场景、第一个实例

在我前一篇博客《WebAPI前置知识:HTTP与RestfulAPI》中已经给各位简单介绍了HTTP协议与RestFul API的关系,以及一些基本的HTTP协议知识,在这些知识的铺垫下,今天,我们一起来讨论一下WEB API的适用场景,然后写我们第一个WEB API接口,并演示如何对其进行简单调用。

    很多人都很迷惑,既然有了WCF为什么还要有WEB API?WEB API会不会取代WCF?

    就我的看法,WCF提供的是一种RPC实现的集合,WCF的设计更多地考虑了SOA的场景,以及各种RPC的问题。很多人也会说,RestFul API也是一种RPC啊,并且WCF中也有关于RestFul 的实现啊。很多资料中RPC和RestFul在风格概念上是有一些区别的,其实我觉得这两者的区别比较主观,过度纠结这些就学院派了;我主要关注了实际使用上的一些问题,在WCF中,支持的协议很多,WS-*系列协议,以及一些更简洁的协议,其中提供了一些专用通信协议的性能是非常高的,并且WCF还提供了服务发现等功能,我认为WCF更适合内部系统间的高性能调用,社区中也有其他一些RPC方案可以选择,例如gRPC,Avor,thrift都是和WCF定位相同的产品;而WEB API是关注于HTTP RestFul风格的产品,在此基础上,任何语言、任何终端都能非常容易地进行对接,并且能利用非常成熟的各种HTTP基础设施和解决方案来进行开发、调试、负载均衡、内容分发。所以,WEB API是一种针对HTTP的,偏重于快速开发RestFul风格开放式API的开发框架。目前看来,他并不能取代WCF,他们各有适合的场景,不能认为WEB API是WCF的替代产品。

       OK,现在我们来开发第一组WEB API接口!使用VS2012以后的版本都有现成的WEB API创建模板,大家跟着创建就好了,创建出来后,项目中会有MVC、WEB API的项目,WEB API对MVC有依赖,不能单独创建!而WEB API和MVC都是利用类似的路由机制,所以在默认路由中,WEB API 使用

/api/{controller}/{id} 

       作为路由,添加了/api/节以区分MVC和web api。

       接下来,我们添加一个WEB API的Controller,取名为PersonController,他继承于ApiController;在创建这个Controller的时候,我们就定义了一种资源:Person,在PersonController里的所有操作均围绕着Person这个资源来的。接下来我们开始定义一组增删改查操作。

       在Web API中,默认路由采用了一种约定:根据谓词来进行路由,而方法名的前缀就是调用该方法对应使用的HTTP谓词。代码示例如下:

/// <summary>
///
Person 为资源,对Person进行的一组操作
/// </summary>public class PersonController : ApiController    {        
       
private static List<Person> _personLst = new List<Person>();              /// <summary>/// 获取一个Person
       /
// </summary>/// <param name="id">Person的ID</param>/// <returns>Person</returns>public Person GetPerson(long id)        {
            
return _personLst.Find(x => x.Id == id);        }        /// <summary>/// 添加一个Person
       
/// </summary>/// <param name="person">Person</param>public void PostAddPerson(Person person)        {            _personLst.Add(person);        }        /// <summary>/// 修改一个
       
/// </summary>/// <param name="id">Person Id</param>/// <param name="person"></param>public void PutModifyPerson(long id, Person person)        {  
           
var p = _personLst.Find(x => x.Id == id);            p.Age = person.Age;            p.Name = person.Name;            p.Sex = person.Sex;        }        /// <summary>/// 删除一个Person
       /// </summary>/// <param name="id">Person ID</param>public void DeletePerson(long id)        {            _personLst.RemoveAll(x => x.Id == id);        } }

一个简单的针对资源的CRUD操作的API就好了,不用解析输入,不用拼接输出,就是那么简单!让我们来遛一遛! 

发送请求:谓词为POST,语义创建Person,Person描述在Body里,head中声明了Body通过Json序列化。

收到响应:响应码204,属于2XX类型执行成功,Body里没有数据

 

发送请求:谓词为GET,语义为查询Person资源,Id为1的,head中声明希望接收使用XML序列化的数据

收到响应:响应码为200,执行成功,Body中有数据,数据使用XML序列化

 

发送请求:谓词为PUT,语义为修改ID为1的Person资源,修改内容在Body中,Content-Type标明Body使用Json序列化,在Body中我们将Name修改为Test1Changed

收到响应,响应码为204,执行成功

 

发送请求:谓词为GET,语义为查询ID为1的Person资源,Accept标明希望接收到Json数据

收到响应:可以看到Body为使用Json序列化的内容,Name属性已经变更为Test1Changed

 

发送请求:谓词为DELETE,语义为删除ID为1的Person资源

收到响应:响应码204,执行成功

 

发送请求:谓词为GET,语义为查询ID为1的Person资源,Accept标明希望接收到Json数据

收到响应:响应码为200,执行成功,响应内容为null,资源已删除 

这就是我用Fiddler来发送、调用的一组RestFul接口,大家可以看到,整个调用过程使用到了HTTP的语义,用到了谓词路由、内容协商。在增、删、改操作中,我都是使用void作为返回值,根据HTTP Code 判断,大家也可以自定义一些返回数据来做出更进一步的操作描述。 

在写了这些API后,我们需要在程序中调用,我以C#为例写一组对这些接口调用的实现。在C#中,传统调用HTTP接口一般有两种办法: WebRequest/WebResponse组合的方法调用和WebClient类进行调用。第一种方法抽象程度较低,使用较为繁琐;而WebClient主要面向了WEB网页场景,在模拟Web操作时使用较为方便,但用在RestFul场景下却比较麻烦,在Web API发布的同时,.NET提供了两个程序集:System.Net.Http和System.Net.Http.Formatting。这两个程序集中最核心的类是HttpClient。在.NET4.5中带有这两个程序集,而.NET4需要到Nuget里下载Microsoft.Net.Http和Microsoft.AspNet.WebApi.Client这两个包才能使用这个类,更低的.NET版本就只能表示遗憾了只能用WebRequest/WebResponse或者WebClient来调用这些API了。

       在使用中,System.Net.Http这个程序集提供了HttpClient类以及相关的HTTP调用,而System.Net.Http.Formatting提供了一些针对HttpClient的帮助扩展,更好地支持了内容协商、Content创建等功能。下面我就和大家一起写一下这个例子:

       我们新建一个控制台程序:

       代码如下:

public class Person    {        
       
public long Id { get; set; }  
      
      
public string Name { get; set; }        public int Age { get; set; }        public string Sex { get; set; }        public override string ToString()        {          
              
return $"Id={Id} Name={Name} Age={Age} Sex={Sex}";        }    }    class Program    {      
       
static void Main(string[] args)        {    
           
var client = new HttpClient();            client.BaseAddress = new Uri("http://localhost:22658/"); //基本的API URL            client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); //默认希望响应使用Json序列化            Run(client);            Console.ReadLine();        }        static async void Run(HttpClient client)        {          
           
var result = await AddPerson(client);            Console.WriteLine($"添加结果:{result}"); //添加结果:truevar person = await GetPerson(client);            Console.WriteLine($"查询结果:{person}"); //查询结果:Id=1 Name=test Age=10 Sex=F            result = await PutPerson(client);            Console.WriteLine($"更新结果:{result}"); //更新结果:true            result = await DeletePerson(client);            Console.WriteLine($"删除结果:{result}"); //删除结果:true        }        static async Task<bool> AddPerson(HttpClient client)        {            return await client.PostAsJsonAsync("api/Person", new Person() { Age = 10, Id = 1, Name = "test", Sex = "F" }) //向Person发送POST请求,Body使用Json进行序列化                                     .ContinueWith(x => x.Result.IsSuccessStatusCode);  //返回请求是否执行成功,即HTTP Code是否为2XX        }        static async Task<Person> GetPerson(HttpClient client)        {            
         
return await await client.GetAsync("api/Person/1") //向Person发送GET请求                .ContinueWith(x => x.Result.Content.ReadAsAsync<Person>(                              //获取返回Body,并根据返回的Content-Type自动匹配格式化器反序列化Bodynew List<MediaTypeFormatter>() {new JsonMediaTypeFormatter()/*这是Json的格式化器*/                                                    ,new XmlMediaTypeFormatter()/*这是XML的格式化器*/}));        }        static async Task<bool> PutPerson(HttpClient client)        {            return await client.PutAsJsonAsync("api/Person/1", new Person() { Age = 10, Id = 1, Name = "test1Change", Sex = "F" }) //向Person发送PUT请求,Body使用Json进行序列化                                    .ContinueWith(x => x.Result.IsSuccessStatusCode);  //返回请求是否执行成功,即HTTP Code是否为2XX        }        static async Task<bool> DeletePerson(HttpClient client)        {            return await client.DeleteAsync("api/Person/1") //向Person发送DELETE请求                .ContinueWith(x => x.Result.IsSuccessStatusCode); //返回请求是否执行成功,即HTTP Code是否为2XX        } }

 

    这就完成了这组API的调用,是不是非常简单方便?HTTPClient使用全异步的方法,并且他有良好的扩展性,我会在之后的博客中再聊这个问题。

    OK,到此为止一组简单的Restful API和C#的调用客户端就完成了,但这只是开始,Web API是一个很强大的框架,他的扩展点非常丰富,这些扩展能为我们的开发提供很多的帮助,下一篇博文我将为大家带来WEB API中Filter的使用。

    博文中如有不正确的地方欢迎大家指正。

原文地址:http://www.cnblogs.com/UliiAn/p/5373854.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/329696.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

rpa文件怎么提取内容_怎么编辑pdf文件内容?有什么软件可以编辑pdf文件吗?

怎么编辑pdf文件内容&#xff1f;我是2020年大学新生&#xff0c;选择学习的专业是财务管理。为了更快、更深入地了解更多专业知识&#xff0c;我上网查找、下载了很多相关资料。不过&#xff0c;从网上保存下来的资料文件大多是pdf格式的&#xff0c;想要编辑它还得先转word。…

Future取消线程执行

【README】 本文总结于 《java并发编程实战》 page121&#xff0c;非常棒的一本书&#xff1b; 【1】Future 1&#xff0c;介绍&#xff1a;future 用于管理任务的生命周期&#xff0c;处理异常&#xff0c;以及实现取消&#xff1b; 2&#xff0c;future.cancel() 取消方法…

Nancy之结合tinyfox给我们的应用提供简单的数据服务

说到提供数据服务给我们的一些应用&#xff0c;估计用的最多的也就是json和xml这两种数据格式 实现的方法也是多种多样&#xff0c;web api,mvc的jsonresult和contentresult...等等 本文是结合Nancy、TInyFox、Owin等来实现的 一、前提工作 新建一个空的web应用程序 添加相应的…

javaweb调用python算法_请教怎么用java远程调用python? 多谢

请问如何用java远程调用python? 谢谢&#xff01;本帖最后由 blackkettle 于 2015-05-07 13:00:41 编辑比如有一台机器 A上安装了python, 另一台机器B要用java 调用A 上的python的某个函数&#xff0c;输入数据在B机器上&#xff0c;所有的计算在A机器完成&#xff0c;结果返回…

(转)如何查看java本地方法

在线查看本地代码&#xff0c; refer2 http://hg.openjdk.java.net/jdk8/jdk8/hotspot/file/tip/src/os/linux/vm/os_linux.cpp 我们知道在java中查看java源码时看到native方法在java层面上就到头的&#xff0c;如果还想继续往下看就需要看jdk是如何实现的&#xff0c;今天就分…

统一配置中心的设计方案

转载自 统一配置中心的设计方案对于配置文件&#xff0c;我们不陌生&#xff0c;它提供我们可以动态修改程序运行能力。引用别人的一句话就是&#xff1a;系统运行时(runtime)飞行姿态的动态调整。我可以把我们的工作称之为在快速飞行的飞机上修理零件。我们人类总是无法掌控和…

微软CEO纳德拉恢弘计划:让开发者始终忘不了微软

BI中文站 4月11日 报道 当微软前CEO史蒂夫•鲍尔默&#xff08;Steve Ballmer&#xff09;在2000年喊出“开发者&#xff0c;开发者&#xff0c;开发者”的口号时&#xff0c;他可能有点儿激动&#xff0c;但是他的观点是对的。 30年前当Windows 1.0推出的时候&#xff0c;这种…

python scratch unity_Unity3D研究院之2D游戏开发制作原理(二十一)

经过了4个月不懈的努力&#xff0c;我和图灵教育合作的这本3D游戏开发书预计下个月就要出版了。这里MOMO先打一下广告&#xff0c;图灵的出版社编辑成员都非常给力&#xff0c;尤其是编辑小花为这本书付出了很大的努力&#xff0c;还有杨海玲老师&#xff0c;不然我也无法完成这…

原码补码与反码

【README】 1.本文内容总结自“哈工大刘宏伟”老师的mooc视频《计算机组成原理》on bilibili&#xff1b; 2.为了便于理解&#xff0c;本文引入了逗号分割符号部分和数值部分&#xff0c;计算机存储数据的时候没有逗号&#xff1b; 【1】原码表示法 原码定义&#xff1a; 原…

第一篇 Entity Framework Plus 之 Audit

一般系统会有登陆日志&#xff0c;操作日志&#xff0c;异常日志&#xff0c;已经满足大部分的需求了。但是有时候&#xff0c;还是需要Audit 审计日志&#xff0c;审计日志&#xff0c;主要针对数据增&#xff0c;改&#xff0c;删操作数据变化的记录&#xff0c;主要是对数据…

本想试试看,结果却拿到了京东的Offer

转载自 本想试试看&#xff0c;结果却拿到了京东的Offer 最近&#xff0c;春招已经基本接近尾声了&#xff0c;我找了几位拿到名企Offer的粉丝&#xff0c;请他们总结了面试经验&#xff0c;近期会分批的推送给大家。希望给那些正在准备秋招的同学提供些帮助。 今天给大家分享的…

ping 命令使用代理_网络检测知识篇:ping命令使用知识,你知道几点?

Ping命令其实是一个非常好的网络故障诊断工具&#xff0c;相信阅读完本文对大家一定有些帮助。如果大家网络遇到问题&#xff0c;不妨试试以下方法。首先使用Ping命令诊断本地TCP/IP协议是否安装正常&#xff0c;检测方法如下&#xff1a;从电脑开始里找到运行&#xff0c;快捷…

8.4-中断系统小结(cpu中断七个问题)

【README】 本文转自bilibili《计算机组成原理&#xff08;哈工大刘宏伟&#xff09;》的视频讲解&#xff0c;非常棒&#xff0c;墙裂推荐&#xff1b; 【1】中断介绍 1&#xff09;作用&#xff1a;用中断系统实现了外设数据的输入输出&#xff1b; 还可以用于程序调试&…

第二篇 Entity Framework Plus 之 Query Future

从性能的角度出发&#xff0c;能够减少 增&#xff0c;删&#xff0c;改&#xff0c;查&#xff0c;跟数据库打交道次数&#xff0c;肯定是对性能会有所提升的&#xff08;这里单纯是数据库部分&#xff09;。 今天主要怎样减少Entity Framework查询跟数据库打交道的次数&#…

python长整数相乘_python写的大整数相乘的方法

输入72106547548473106236 982161082972751393 两个大整数输出结果70820244829634538040848656466105986748解题思路首先根据 大整数相乘的原理的基础上&#xff0c;把大整数进行优化拆分&#xff0c;拆分的长度&#xff0c;要考虑语言中整形的长度。这里用的python&#xff0c…

Java Web应用的代码分层最佳实践

转载自 Java Web应用的代码分层最佳实践代码分层&#xff0c;对于任何一个Java Web开发来说应该都不陌生。一个好的层次划分不仅可以能使代码结构更加清楚&#xff0c;还可以使项目分工更加明确&#xff0c;可读性大大提升&#xff0c;更加有利于后期的维护和升级。从另外一个角…

中断屏蔽技术

【README】 本文总结自bilibili《计算机组成原理&#xff08;哈工大刘宏伟&#xff09;》的视频讲解&#xff0c;非常棒&#xff0c;墙裂推荐&#xff1b; 【1】中断屏蔽 1&#xff0c;中断屏蔽的意思是&#xff0c;在中断1的服务程序执行过程中&#xff0c;不允许被其他中断…

.NET FM的未来计划

2016年3月21日&#xff0c;我们启动了.NET FM这档独立播客来服务中文.NET和微软技术社区。如同早先感谢信所言&#xff0c;能够得到社区的肯定和全力支持&#xff0c;二位主播是受宠若惊。关于未来的计划&#xff0c;下面简单和大家分享一下。 首先&#xff0c;关于节目播出频率…

Java开发必须掌握的日志分析命令

转载自 Java开发必须掌握的日志分析命令 对于大型网站来说&#xff0c;很多网站在可用性方面提出4个9或者5个9的要求&#xff0c;如果是4个9&#xff0c;那么网站全年的不可用时间不能超过52.6分钟&#xff0c;如果是5个9&#xff0c;全年不可用时间不能超过5.2分钟。这其实是很…

指令寻址方式与数据寻址方式

【README】 1.本文总结自bilibili《计算机组成原理&#xff08;哈工大刘宏伟&#xff09;》的视频讲解&#xff0c;非常棒&#xff0c;墙裂推荐&#xff1b; 2. cpu访存寄存器耗费 10ns&#xff0c;访问缓存&#xff08;高速缓冲寄存器&#xff09;耗费20ns&#xff0c;访问主…