前言

之前对于用SelfHost来手动实现Web API的宿主模式,似乎不是太深入,所以本篇文章我们一起来讨论关于利用HttpClient来访问Web API上的资源来进行探讨以及注意相关事项,希望此文对你也有收获。

来自XML或Json Content的简单参数

当Web API方法中接受如String、Datetime、Int等参数类型时,但是默认情况下这些方法不会接收来自XML或者JSON Body的参数,那么结果就是导致请求失败。接下来我们来进行演示。

我们取名为Default的Web API控制器,方法以及路由注册如下:

        [HttpPost]public string GetString(string message){return message;}config.Routes.MapHttpRoute(name: "DefaultApi",routeTemplate: "api/{controller}/{action}/{message}",defaults: new { message = RouteParameter.Optional });

接下来我们用客户端发出POST请求以及数据来请求Web API来并返回该数据,如下:

            var client = new HttpClient();var postUrl = "http://localhost:23133/api/default/GetString";var task = client.PostAsJsonAsync<string>(postUrl,"Hello World").Result; //将参数进行序列化为JSON并发送到Request中的Body中

通过上述路由以及请求方式的配置理论上来说是能通过URI能访问到资源,但是结果却令你大跌眼镜

由上可得知,参数Message的值Hello World已经传递过来,但是此时请求响应的资源上方法的参数并未接收到且返回404未找到该方法。

上面可能效果不是太明显,我们利用Ajax传递时间来进行请求并返回该数据,方法如下:

        public HttpResponseMessage GetDateTime(DateTime time){return Request.CreateResponse<DateTime>(HttpStatusCode.OK, time);}

前台请求如下(路由配置就此略过):

       $("#btn").click(function () {$.ajax({type: "get",url: "http://localhost:23133/api/Default/GetDateTime",dataType: "json",contentType: "application/json; charset=utf-8",data: { "time": new Date("2015","09","22") },cache: false,success: function (r) {console.log(r);}});});

此时响应信息如下:

{"Message":"请求无效。","MessageDetail":"对于“WebApplication2.Controllers.DefaultController”中方法“System.Net.Http.HttpResponseMessage GetDateTime(System.DateTime)”的不可以为 null 的类型“System.DateTime”的参数“time”,参数字典包含一个 null 项。可选参数必须为引用类型、可以为 null 的类型或声明为可选参数。"}

出错原因是对于简单的参数(如String、Int以及DateTime等)当进行请求时不会进行解析其属性而会将其设置为null,所以传递过来的时间为null,而方法的时间参数又不能为空,所以会出现错误。对于这种情况我们使用特性[FromBody]来解决【注意】当在前台传递值为字符串或者数字时则是可以解析的,能够成功传递到方法的参数时,而利用HttpClient进行传递时则会出现404。

鉴于上述,我们将其方法参数进行如下修饰即可成功

  public HttpResponseMessage GetDateTime([FromBody] DateTime time)

建议

对于前台进行简单参数进行传递以及和利用HttpClient来进行简单参数的传递出现的情况不同,建议在其参数上加上特性 FromBody  ,这样两者皆可避免。

补充

当然有FromBody也就对应的有FromURI,添加此特性就可获得URL上该特性参数对应的值,URI如下:

    var postUrl = "http://localhost:23133/api/default/GetString?message='xpy0928'";

被请求的资源如下:

        [HttpPost]public string GetString([FromUri] string message){return message;}

此时请求时该message对应的值就为xpy0928。

UrlEncoded表单字符串解析

我们在操作表单时肯定会有将表单数据作为字典进行传递的情况,下面我们就来演示这种情况。

如下为Web API需要被请求的方法

        [HttpPost]public string GetString(string cnblogs){return cnblogs;}

接下来我们利用客户端来访问上述该方法:

            var formVars = new Dictionary<string, string>();formVars.Add("cnblogs", "xpy0928");var content = new FormUrlEncodedContent(formVars);  var client = new HttpClient();var postUrl = "http://localhost:23133/api/default/GetString";var task = client.PostAsync(postUrl, content).Result;  //参数未进行序列化

从上述可以看出理论上是可行的,我们通过URI将表单数据以字典形式进行传递去请求Web API,并获取键cnblogs的值。结果会虐死你,如下:

因为这种对于cnblogs参数值映射到方法上的cnblogs参数上在Web API是行不通的,Web API不会进行映射 。

既然简单的参数来接收不行下面我们尝试利用对象来获得该参数的值就像MVC中提交参数到实体上一样。

首先给出一个类,并且该类中含有cnblogs属性,如下:

        public class CnblogsModel{public string Cnblogs { get; set; }}

将其方法及参数修改如下:

       [HttpPost]public string GetString(CnblogsModel model){return model.Cnblogs;}

接下来就是见证奇迹的时刻到了,给力啊AngelaBaby,皇天不负有心人,如下:

【注意】由上可知成功的关键是对象中的字段必须要和POST表单中的键必须相匹配,如果有多个键,那么对象中的字段也要与之对应。

上述的解决方案完全是可行的,但是让人忍俊不禁的是这么做似乎有点太操蛋,接受表单中的参数还得建立对象并与之对应,所以得想更好的出路,既然是针对表单的,难道表单中对于接受这样的字典就没有对应的类来进行处理吗,于是乎,开始找啊找,结果找到了一个 FormDataCollection (表面意思是表单数据集合)。貌似可行,来我们继续将其代码进行改写,如下:

       [HttpPost]public string GetString(FormDataCollection collection){return collection.Get("cnblogs");}

验证通过,如下:

这样就比第一个解决方案可行多了,只是对于这样的集合类有点让人想不明白,为什么不直接通过索引来获取数据而非得用对于单个用Get或者对于多个字段用GetValues()方法

你是不是就以为这样简洁就是可行的呢?那你可就错了,当我们在ASP.NET上运行Web API我们完全可以通过如下来进行获取表单数据:

            var request = HttpContext.Current.Request;var formKeys = request.Form.AllKeys;

建议

当获取表单数据时建议用 HttpContext.Current.Request  ,虽然它仅仅只能在ASP.NET上进行应用而无法在SelfHost上使用,但是对于一些复杂的场景下,比如说你需要获得一些Web API上的Request没有暴露的数据,此时用此种方法将是不可替代的,同时HttpContext无论是在ASP.NET还是MVC或者Web API中都是存在的。所以尽量避免使用FormDataCollection,即使看起来在路由上使用它更简单。

补充(1)

感谢园友【敷衍不起】的疑问,刚开始以为挺简单的,结果就是被打脸了,后经过研究对于复杂类型的数据的传递以及接收,若是具体类,利用FormDataCollection是行不通的,我能想到的解决方案是进行如下操作【期待你更好的解决方案】:

            var p = new Person() { Name = "person", Student = new Student() { StuName = "xpy0928" } };var client = new HttpClient();var postUrl = new Uri("http://localhost:23133/api/default/GetString");var task = client.PostAsJsonAsync(postUrl, p).Result;  //利用PostAsJsonAsync方法来进行序列化类

在Web API上进行接收,如下:

     [HttpPost]public string GetString(Person p){return p.Name;}

至于用FormDataCollection接收不行的原因是 FormUrlEncodedContent 的参数键和值必须都是string类型,若类不是具体类直接添加键和值都是字符串则是可行的,但是这种情况似乎不太现实,所以对于具体类还是利用 PostAsJsonAsync 来进行解决。

补充(2)

感谢园友【YoMe和kennywangjin】的提醒,上述必须显式通过路由来进行参数传递才会成功,如果通过Request中body来传递就会出现不解析的情况。

访问QueryString

在命名空间System.Net.Http下有一扩展方法如下:

            var requestUri = Request.RequestUri.ParseQueryString();var paramValue = requestUri["param"];

在大多数场景下使用此方法是比较高效的,但是如果在URI中有许多参数,此时你不得不去显式去解析这些参数。

总结

如上述,Web API对于相关参数的映射是比较简单的,但是在一些不同的情况下,你不得不手动做一些工作来进行处理,同时我们也许意识到了Web API并没有提供任何全局上下文对象,例如Request以及过滤中的Formatters等,都是基于Web API来提供特定的方法来进行操作即方法都是为Web API而量身定制,所以在看到Web API强大的同时也要看到其软肋,这样我们才能更好的对于不同场景下作出不同的处理。

本文转自Jeffcky博客园博客,原文链接:http://www.cnblogs.com/CreateMyself/p/4827171.html,如需转载请自行联系原作者

Web APi之HttpClient注意事项以及建议(四)相关推荐

  1. Web APi之HttpClient注意事项以及建议

    Web APi之HttpClient注意事项以及建议 前言 之前对于用SelfHost来手动实现Web API的宿主模式,似乎不是太深入,所以本篇文章我们一起来讨论关于利用HttpClient来访问W ...

  2. .NET Core Web API使用HttpClient提交文件的二进制流(multipart/form-data内容类型)

    需求背景: 拼多多优惠券 m.fenfaw.net 在需要通过服务端请求传递文件二进制文件流数据到相关的服务端保存时,如对接第三方接口很多情况下都会提供一个上传文件的接口,但是当你直接通过前端Ajax ...

  3. 创建包含CRUD操作的Web API接口-第一部

    在这里,我们将创建一个新的Web API项目,它将使用实体框架实现Get,POST.PUT和DELETE方法来实现CRUD操作. 首先,在Visual Studio 2013 for Web expr ...

  4. ASP.NET Web API 2 入门教程

    译者:jiankunking 出处:http://blog.csdn.net/jiankunking 源码下载 HTTP不仅提供web页面服务,在构建公开服务和数据api方面,它也是一个强大的平台.H ...

  5. java调用asp.net webapi_通过HttpClient 调用ASP.NET Web API示例

    在前面两篇文章中我们介绍了ASP.NET Web API的基本知识和原理,并且通过简单的实例了解了它的基本(CRUD)操作.我们是通过JQuery和Ajax对Web API进行数据操作.这一篇我们来介 ...

  6. 使用HttpClient消费ASP.NET Web API服务

    本篇体验使用HttpClient消费ASP.NET Web API服务,例子比较简单. 依次点击"文件","新建","项目". 选择&quo ...

  7. 使用HttpClient 调用Web Api

    C#4.5 添加了异步调用Web Api . 如果你的项目是4.5以上版本,可以直接参考官方文档. http://www.asp.net/web-api/overview/web-api-client ...

  8. 【ASP.NET Web API教程】3.4 HttpClient消息处理器

    [ASP.NET Web API教程]3.4 HttpClient消息处理器 原文:[ASP.NET Web API教程]3.4 HttpClient消息处理器 注:本文是[ASP.NET Web A ...

  9. ASP.NET Web API 入门 (API接口、寄宿方式、HttpClient调用)

    一.ASP.NET Web API接口定义 ASP.NET Web API默认实现了Action方法和HTTP方法的映射,Action方法方法名体现了其能处理的请求必须采用的HTTP方法 二.寄宿方式 ...

最新文章

  1. Android SpannableString 给TextView 设置颜色,删除线等
  2. oracle 11.2 安装asm,oracle11r2安装asm+rac配置步骤.doc
  3. 常用的web安全处理
  4. eclipse启动了tomcat,但是浏览器打不开欢迎页
  5. junit5和junit4_JUnit 5 –基础
  6. oracle窗帘位图索引,Greenplum数据库设计开发规范参考.docx
  7. Linux查看占用mem的进程脚本
  8. Android自动打包、签名、优化、上传ANT脚本
  9. 产业变革新机遇,与5G同行,驰骋在浪潮之巅
  10. 利用canvas的getImageData()方法制作《在线取色器》
  11. 不是一个PDF文件或该文件已损坏
  12. Programiranje
  13. 抖音、腾讯世纪大和解,透露了哪些信息?
  14. AcWing120防线 经典题二分+前缀和+等差数列
  15. 把英文单词转换成ASCII画
  16. POJ海贼王之伟大航路(dfs)
  17. NLP词向量和句向量方法总结及实现
  18. EXCEL函数篇之一文读懂VLOOKUP精准查找、近似查找、模糊查找的区别
  19. 电子密码锁——数电课设
  20. .设计一个Student类,该类中包括学生的姓名和成绩。创建Student类的5个对象,如下所示: 姓名 成绩 刘德华 85 张学友 100 刘杰 65 章子怡 58 周迅 76 将以上5个对象

热门文章

  1. Java线程和进程区别
  2. 系统困境与软件复杂度:为什么我们的系统会如此复杂?
  3. Xcode SDK模拟器安装及安装路径
  4. android sdk模拟器在linux和windows下的性能比较
  5. 《如何高效记忆》阅读笔记
  6. Android集成常见问题,android软件开发教程
  7. 清华大学自动化系科技营暨清华大学自动化系智能机器人挑战赛 | 无线电轨比赛实施方案
  8. 比亚迪在连遭挫败之后,4月份迎来喜报,夺下热销榜第一名
  9. latex数学公式总结
  10. 批量下载500首音乐走起(python多线程)分分钟下完