Asp.Net Web API(四)

HttpResponseException-----HTTP响应异常

    如果Web API控制器抛出一个未捕捉的异常,会发生什么呢?在默认情况下,大多数异常都会转换为一个带有状态码500的内部服务器错误的HTTP响应。

这个HTTPResponseException类型是一个特殊的类型。这种异常会返回你在异常构造器中指定的任何HTTP状态码。例如,在以下方法中,如果这个id参数无效,那么会返回“404---未找到”

1  public Product GetProduct(int id)
2 {
3             var item = repository.Get(id);
4             if (item == null)
5                 //未找到返回一个404的状态码
6                 throw new HttpResponseException(HttpStatusCode.NotFound);
7             return item;
8 }

为了响应进行更多的控制,你也可以构造整个响应消息,并用HTTPResponseMessage来包含它:

 1 public Product GetProduct(int id)
 2 {
 3             var item = repository.Get(id);
 4             if (item == null)
 5             //未找到返回一个404的状态码
 6             {
 7                 var resp = new HttpResponseMessage(HttpStatusCode.NotFound)
 8                 {
 9                     Content = new StringContent(String.Format("No product with ID={0}", id)),
10                     ReasonPhrase = "Product ID Not Found"
11                 };
12                 throw new HttpResponseException(resp);
13             }
14             return item;
15 }

Exception Filters---异常过滤器

   通过编写一个异常过滤器,你可以定制Web API如何处理异常。当一个控制器抛出一个未处理异常,且这个异常不是一个HttpResponseException异常时,一个异常过滤器会被执行。HttpResponseException类型是一个特殊情况,因为它是专门设计用来返回一个HTTP响应的。

   异常过滤器实现System.Web.Http.Filters.IExceptionFilter接口。编写异常过滤器最简单的方法是通过System.Web.Fitlers.ExceptionFilterAttribute类进行派生,并重写OnException方法。

      注意:ASP.NET Web API中的异常过滤器与ASP.NET MVC中是及其相似的。然后,它们被声明在不用的命名空间下,且功能也是独立的。特别强调以下,ASP.NET MVC中使用的HandlerErrorFilterAttribute不会处理Web API控制器抛出的异常。

  以下是将NotImplementedException异常转换成HTTP状态码“501 - 未实现”的过滤器:

 1 namespace WebAPIDemo.Filter
 2 {
 3     public class NotImpleExceptionFilterAttribute:ExceptionFilterAttribute
 4     {
 5         public override void OnException(HttpActionExecutedContext actionExecutedContext)
 6         {
 7             if (actionExecutedContext.Exception is NotImplementedException)
 8                 actionExecutedContext.Response = new System.Net.Http.HttpResponseMessage(System.Net.HttpStatusCode.NotImplemented);
 9         }
10     }
11 }

HttpActionExecutedContext对象的Response属性含有发送到客户端的HTTP响应消息

Registering  Exception Filters --- 注册异常过滤器

   以下是注册Web API异常过滤器的几种方式

       1.通过Action注册

       2.通过Controller注册

       3.通过全局注册

   要把过滤器应用特定的Action,在Action中添加过滤器的注解属性

 1 [NotImpleExceptionFilter]
 2 public Product GetProduct(int id)
 3  {
 4           var item = repository.Get(id);
 5            if (item == null)
 6             //未找到返回一个404的状态码
 7             {
 8                 var resp = new HttpResponseMessage(HttpStatusCode.NotFound)
 9                 {
10                     Content = new StringContent(String.Format("No product with ID={0}", id)),
11                     ReasonPhrase = "Product ID Not Found"
12                 };
13                 throw new HttpResponseException(resp);
14             }
15            return item;
16 }

要把过滤器应用于一个控制器的所有Action,可以在Controller上添加过滤器的注解属性

[NotImpleExceptionFilter]
public class ProductController : ApiController
{
   
}

要全局性的把过滤器运用于所有的Web API控制器将该过滤器的一个实例添加GlobalConfiguration.Configuration.Filter集合。这个集合中的所有异常过滤器会应用于任何Web API控制器Action

public class WebApiApplication : System.Web.HttpApplication
{
        protected void Application_Start()
        {
            //注册全局异常过滤器
            GlobalConfiguration.Configuration.Filters.Add(new NotImpleExceptionFilterAttribute());
        }
 }

HttpError----HTTP错误

HttpError对象为在响应正文中返回错误消息提供了响应的方法。以下实例演示了如何用HttpError在响应中返回HTTP状态码“404--未找到”:

1 public HttpResponseMessage GetProduct(int id)
2 {
3             var item = repository.Get(id);
4             if (item != null)
5                 return Request.CreateResponse(HttpStatusCode.OK, item);
6             var message = string.Format("Product with id = {0} not found", id);
7             HttpError httpError = new HttpError(message);
8             return Request.CreateResponse(HttpStatusCode.NotFound, httpError);
9 }

在这个例子中,如果该方法成功,它会在HTTP响应中返回产品。但如果所请求的产品未找到,则HTTP响应会在请求体中包含一个HttpError。该响应看起来大致像这样

1 HTTP/1.1 404 Not Found
2 Content-Type: application/json; charset=utf-8
3 Date: Thu, 09 Aug 2012 23:27:18 GMT
4 Content-Length: 51
5 
6 {
7   "Message": "Product with id = 12 not found"
8 }

  注意。在这个例子中,HttpError会被序列化成JSON。使用HttpError的一个好处是,与其它强类型模型一样,会进行同样的“content-negotiation”(暂未实现)和序列过程

  直接替代创建HttpError对象的一种方法是:你可以使用CreateErrorResponse方法:

1 public HttpResponseMessage GetProduct(int id)
2 {
3             var item = repository.Get(id);
4             if (item != null)
5                 return Request.CreateResponse(HttpStatusCode.OK, item);
6             var message = string.Format("Product with id = {0} not found", id);
7             return Request.CreateErrorResponse(HttpStatusCode.NotFound, message);
8 }

CreateErrorResponseSystem.Net.Http.HttpRequsetMessageExtensions类中被定义的一个扩展方法。本质上,CreateErrorResponse会创建一个HttpError实例,然后创建一个包含该HttpError的HttpResponseMessage

  Adding Custom Key-Values to HttpError 把自定义的键值添加到HttpError

  HttpError类实际上是个键--值集合,(派生与于Dictionary<String,Object>),因此你可以添加自己的键--值对

 1 public HttpResponseMessage GetProduct(int id)
 2 {
 3             var item = repository.Get(id);
 4             if (item != null)
 5                 return Request.CreateResponse(HttpStatusCode.OK, item);
 6             var message = string.Format("Product with id = {0} not found", id);
 7             var err = new HttpError(message);
 8             err["error_sub_code"] = 42;
 9             return Request.CreateErrorResponse(HttpStatusCode.NotFound, err);
10 }

Using HttpError with HttpResponseExceptionHttpResponseException的方式来使用HttpError

前面的例子是从Action返回一个HttpResponseMessage消息,但你也可以使用HttpResponseException来返回一个HttpError。这让你能够在正常成功情况下返回强类型模型,而在错误时,仍返回HttpError

1 public Product GetProduct(int id)
2 {
3             var item = repository.Get(id);
4             if (item != null)
5                 return item;
6             var message = string.Format("Product with id = {0} not found", id);
7             throw new HttpResponseException(Request.CreateErrorResponse(HttpStatusCode.NotFound, message));
8 }

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Golang语言社区

go语言实现http服务端与客户端

go语言的net/http包的使用非常的简单优雅 (1)服务端 [plain] view plain copy package main import ...

3497
来自专栏SDNLAB

ODL应用开发之MD-SAL中级教程

1. 简介 本次我们从开始设计到最终完成一个应用的开发,主要设计datastore和RPC定义和实现。Opendaylight 开发使用了OSGi框架,OSGi...

6608
来自专栏逸鹏说道

C#进阶系列——WebApi 接口参数不再困惑:传参详解 下

(1)基础类型数组 var arr = ["1", "2", "3", "4"]; $.ajax({ type: "post", ...

2966
来自专栏微信公众号:Java团长

从并发编程到分布式系统——如何处理海量数据(上)

在这里想写写自己在学习并发处理的学习思路,也会聊聊自己遇到的那些坑,以此为记,希望鞭策自己不断学习、永不放弃!

741
来自专栏Jed的技术阶梯

Kafka 消费者旧版低级 API

Kafka 消费者总共有 3 种 API,新版 API、旧版高级 API、旧版低级 API,新版 API 是在 kafka 0.9 版本后增加的,推荐使用新版 ...

1833
来自专栏Java技术栈

Java程序员注意:Tomcat Get请求的巨坑!

2.1K2
来自专栏大内老A

ASP.NET Core管道深度剖析(3):管道是如何处理HTTP请求的?

我们知道ASP.NET Core请求处理管道由一个服务器和一组有序的中间件组成,所以从总体设计来讲是非常简单的,但是就具体的实现来说,由于其中涉及很多对象的交互...

2845
来自专栏MasiMaro 的技术博文

Vista 及后续版本的新线程池

在上一篇的博文中,说了下老版本的线程池,在Vista之后,微软重新设计了一套线程池机制,并引入一组新的线程池API,新版线程池相对于老版本的来说,它的可控性更高...

1503
来自专栏偏前端工程师的驿站

asp.net 解码gb2312下urlencode后的字符串

公司网站前期的网页用了gb2312保存用户数据,而我负责的部分用的是utf8,今天恰好要获取前期录入的数据于是毫无悬念地出现乱码问题,经过一番网上的搜索还是找不...

2085
来自专栏Core Net

ASP.NET Core 2.0 : 八.图说管道,唐僧扫塔的故事

2174

扫码关注云+社区

领取腾讯云代金券