一、媒体类型

  媒体类型(也称为MIME类型)标识一段数据的格式。在HTTP中,媒体类型描述了消息体的格式。媒体类型由两个字符串组成,一个类型和一个子类型。例如:text / html; image/ PNG ;application/ JSON
在HTTP中,Response的 Content-Type 标头指定消息主体的格式。这告诉接收者如何解析消息体的内容。

HTTP/1.1  OK
Content-Length:
Content-Type: image/png

Request中的 Accept 标头告诉服务器客户端希望从服务器获取哪些媒体类型。

Accept: text/html,application/xhtml+xml,application/xml

此标头告诉服务器客户端需要HTML,XHTML或XML。
  媒体类型用于确定Web API如何序列化和反序列化HTTP消息正文。Web API内置支持XML,JSON,BSON和表单urlencoded数据,我们也可以编写 媒体格式化程序(media-type formatter ) 来支持其他媒体器。媒体格式化器的作用是:

Read CLR objects from an HTTP message body
Write CLR objects into an HTTP message body

要创建媒体格式化器,请从以下类之一派生:

MediaTypeFormatter。该类使用异步读写方法。
BufferedMediaTypeFormatter。此类派生自MediaTypeFormatter,使用同步读/写方法,简单但是会读写期间阻塞线程。

从BufferedMediaTypeFormatter派生更简单,因为没有异步代码,但它也意味着调用线程会在I / O期间阻塞。

二、一个创建媒体格式的栗子

需求:为Book模型创建CSV媒体格式化程序,用于将Book对象序列化为逗号分隔值格式的媒体类型(CSV)。以下是Book对象的定义:

public class Book
{
  public int BookId { get; set; }
  public string Title{ get; set; }
  public string Genre { get; set; }
  public decimal Price { get; set; }
}

第一步:创建一个BookCsvFormatter

BookCsvFormatter继承于BufferedMediaTypeFormater类:

public class BookCsvFormatter : BufferedMediaTypeFormatter
{
//构造函数
public BookCsvFormatter()
{
      //添加支持的Mime类型
SupportedMediaTypes.Add(new MediaTypeHeaderValue("text/csv"));
// 添加支持的编码类型
    SupportedEncodings.Add(new UTF8Encoding(encoderShouldEmitUTF8Identifier: false));
    SupportedEncodings.Add(Encoding.GetEncoding("iso-8859-1"));
}
//实现BufferedMediaTypeFormatter抽象方法,是否可读
public override bool CanReadType(Type type)
{
//不可读
return false;
}    //实现BufferedMediaTypeFormatter抽象方法,是否可写
   public override bool CanWriteType(Type type)
  {
if (type == typeof(Book))
{ return true;}
else
    {
      //支持csv类型集合的读取
      Type enumbook = typeof(IEnumerable<Book>);
      return enumbook.IsAssignableFrom(type);
      }
    }      //重写写入方法
    public override void WriteToStream(Type type, object value, Stream writeStream, HttpContent content)
    {
      //通过httpcontent获取编码类型
      Encoding currentEncoding = SelectCharacterEncoding(content.Headers);
      using (var writer=new StreamWriter(writeStream, currentEncoding))
      {
        var books = value as IEnumerable<Book>;
        if (books!=null)
        {
          foreach (var book in books)
          {
            WriteItem(book, writer);
          }
         }
        else
        {
          var singleBook = value as Book;
          if (singleBook == null)
          {
            throw new InvalidOperationException("不能序列化该类型");
          }
          WriteItem(singleBook, writer);
        }
       }
      }
    public void WriteItem(Book book, StreamWriter writer) {
      writer.WriteLine($"{book.BookId.ToString()},{book.Title.ToString()},{book.Genre.ToString()},{book.Price.ToString()}");
      }
    }

第二步:将BookCsvFormatter程序添加到Web API管道

打开WebApiConfig.cs,添加以下代码

public static void ConfigureApis(HttpConfiguration config)
{
//添加这行代码
config.Formatters.Add(new BookCsvFormatter ());
}

第三步:测试

在BooksController中添加控制器如下

 [Route("~/api/books/helloCsv")]
public HttpResponseMessage GetBook()
{
Book book = new Book { BookId=,Title="哈利波特",Genre="玄幻",Price=59.99M};
return Request.CreateResponse(HttpStatusCode.OK, book, "text/csv");
}

重新生成后启动在url填入http://localhost:60398/api/books/helloCsv,会直接下载一个名字为helloCsv的文件

打开记事本打开helloCsv,内容是:1,哈利波特,玄幻,59.99 ,测试成功。

本文实现了一个简单的添加媒体类型的方法,如果想了解更多请查看官网

Web API中给领域模型添加媒体类型支持的更多相关文章

  1. Web API中的返回值类型

    WebApi中的返回值类型大致可分为四种: Void/ IHttpActionResult/ HttpResponseMessage /自定义类型 一.Void void申明方法没有返回值,执行成功后 ...

  2. Entity Framework 6 Recipes 2nd Edition(9-3)译->找出Web API中发生了什么变化

    9-3. 找出Web API中发生了什么变化 问题 想通过基于REST的Web API服务对数据库进行插入,删除和修改对象图,而不必为每个实体类编写单独的更新方法. 此外, 用EF6的Code Fri ...

  3. 使用ASP.NET Web Api构建基于REST风格的服务实战系列教程【五】——在Web Api中实现Http方法(Put,Post,Delete)

    系列导航地址http://www.cnblogs.com/fzrain/p/3490137.html 前言 在Web Api中,我们对资源的CRUD操作都是通过相应的Http方法来实现——Post(新 ...

  4. 【ASP.NET Web API教程】6.2 ASP.NET Web API中的JSON和XML序列化

    谨以此文感谢关注此系列文章的园友!前段时间本以为此系列文章已没多少人关注,而不打算继续下去了.因为文章贴出来之后,看的人似乎不多,也很少有人对这些文章发表评论,而且几乎无人给予“推荐”.但前几天有人询 ...

  5. Asp.Net Web API 2第十三课——ASP.NET Web API中的JSON和XML序列化

    前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok/p/3446289.html 本文描述ASP.NET W ...

  6. 【ASP.NET Web API教程】6.1 媒体格式化器

    http://www.cnblogs.com/r01cn/archive/2013/05/17/3083400.html 6.1 Media Formatters6.1 媒体格式化器 本文引自:htt ...

  7. Web Api中实现Http方法(Put,Post,Delete)

    在Web Api中实现Http方法(Put,Post,Delete) 系列导航地址http://www.cnblogs.com/fzrain/p/3490137.html 前言 在Web Api中,我 ...

  8. ASP.NET Web API中的JSON和XML序列化

    ASP.NET Web API中的JSON和XML序列化 前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok ...

  9. Web API中的内容协商

    一.内容协商的概念 HTTP规范将内容协商定义为“当有多个格式可用时为给定响应选择最佳格式的过程”.HTTP中内容协商的主要机制是这些请求标头: Accept:响应可接受哪些媒体类型,例如“appli ...

随机推荐

  1. 【BZOJ3813】【清华集训2014】奇数国 线段树 数学

    题目描述 给你一个长度为\(n\)的数列,第\(i\)个数为\(a_i\).每个数的质因子都只有前\(60\)个质数.有\(q\)个询问,每次给你\(l,r\),求\(\varphi(\prod_{i ...

  2. NORMA2 - Norma [cdq分治]

    题面 洛谷 你有一个长度为n的序列,定义这个序列中每个区间的价值是 \(Cost(i,j)=Min(Ai...Aj)∗Max(Ai...Aj)∗(j−i+1)Cost(i,j)=Min(A_{i}.. ...

  3. Codeforces Round #449 (Div. 1) Willem, Chtholly and Seniorious (ODT维护)

    题意 给你一个长为 \(n\) 的序列 \(a_i\) 需要支持四个操作. \(1~l~r~x:\) 把 \(i \in [l, r]\) 的 \(a_i\) 加 \(x\) . \(2~l~r~x: ...

  4. 【BZOJ4868】[六省联考2017]期末考试(贪心)

    [BZOJ4868][六省联考2017]期末考试(贪心) 题面 BZOJ 洛谷 题解 显然最终的答案之和最后一个公布成绩的课程相关. 枚举最后一天的日期,那么维护一下前面有多少天可以向后移,后面总共需 ...

  5. 「NOI2014」购票 解题报告

    「NOI2014」购票 写完了后发现写的做法是假的...然后居然过了,然后就懒得管正解了. 发现需要维护凸包,动态加点,询问区间,强制在线 可以二进制分组搞,然后你发现在树上需要资瓷撤回,然后暴力撤回 ...

  6. bandwagon host

    104.20.6.63 bandwagonhost.com 104.20.6.63 bwh1.net

  7. CF1139D Steps to One(DP,莫比乌斯反演,质因数分解)

    stm这是div2的D题……我要对不住我这个紫名了…… 题目链接:CF原网  洛谷 题目大意:有个一开始为空的序列.每次操作会往序列最后加一个 $1$ 到 $m$ 的随机整数.当整个序列的 $\gcd ...

  8. [CTSC2018]暴力写挂——边分树合并

    [CTSC2018]暴力写挂 题面不错 给定两棵树,两点“距离”定义为:二者深度相加,减去两棵树上的LCA的深度(深度指到根节点的距离) 求最大的距离. 解决多棵树的问题就是降维了. 经典的做法是边分 ...

  9. APP reset.css

    html { box-sizing: border-box; } * { user-select: none; -webkit-tap-highlight-color: rgba(255,255,25 ...

  10. MyQR库自动为网址生成二维码

    首先安装MyQR库: pip install MyQR #导包 from MyQR import myqr #生成二维码 words=你要为那个网址生成二维码 save_name=保存后的图片名 pi ...