有时候,向服务端请求一个实体,我们希望返回如下的格式:

links: [
    href: http://localhost:8901/api/user/diaries/2013-08-17,
    rel: "self",
    method: "GET",
    isTemplated: false
],
currentDate:"2013-08-17"

首先抽象出一个与Link相关的类:

public class LinkModel
{
public stirng Href{get;set;}
public stirng Rel{get;set;}
public string Method{get;set;}
public bool IsTemplated{get;set;}
}

再放到某个视图模型中:

public class DiaryModel
{
//存储和模型相关的链接
public ICollection<LinkModel> Links{get;set;}
public DateTime CurrentDate{get;set;}
} public class Diary
{
public int Id{get;set;}
public DateTime CurrentDate{get;set;}
}

ModelFactory用来实现视图模型和领域模型之间的转化。

public class ModelFactory
{ private UrlHelper _urlHelper; public ModelFactory(HttpRequestMessage request)
{
_urlHelper = new UrlHelper(request);
} //领域模型转换成视图模型
public DiaryModel Create(Diary d)
{
return new DiaryModel()
{
Links = new List<LinkModel>
{
CreateLink(_urlHelper.Link("Diaryis", new {diaryid=d.CurrentDate.ToString("yyyy-MM-dd")}),"self");
},
CurrentDate = d.CurrentDate
}
} public LinkModel CreateLink(string href, string rel, string method = "GET", bool isTemplated = false)
{
return new LinkModel()
{
Href = href,
Rel = rel,
Method = method,
IsTemplated = isTemplated
}
} //视图模型转换成领域模型
public Diary Parse(DiaryModel model)
{
try
{
var entity = new Diary();
var selfLink = model.Links.Where(l => l.Rel == "self").FirstOrDefault(); if(selfLink != null && !string.IsNullOrWhiteSpace(selfLink.Href))
{
//从Uri中取出主键
var uri = new Uri(selfLink.Href);
entity.Id = int.Parse(uri.Segments.Last());
} entity.CurrentDate = model.CurrentDate; return entity;
}
catch(Exception ex)
{ }
}
}

Diaries这个controller略,路由方面:

//api/user/diaries
//api/user/diaries/2001-01-01
config.Routes.MapHttpRoute(
name: "Diaries",
routeTemplate: "api/user/diaries/{dairyid}",
defaults: new {controller="diaries", diaryid=RouteParameter.Optional}
)

这样,在客户端发出 http://localhost:8901/api/user/diaries/2013-08-17 GET请求,得到如下的响应:

links: [
    href: http://localhost:8901/api/user/diaries/2013-08-17,
    rel: "self",
    method: "GET",
    isTemplated: false
],
currentDate:"2013-08-17"

在返回分页相关的action中,也可以返回相关的Link部分。

先定义一个基类控制器:

public abstract class BaseController : ApiController
{
ICountingKsRepository _repo;
ModelFactory _modelFactory; public BaseController(ICountingKsRepository repo)
{
_repo = repo;
//写在构造函数里的话有点迟,必须等实例化_modelFactory才有值
//_modelFactory = new ModelFactory(this.Request);
} protected ModelFactory TheModelFactory
{
get
{
if(_modelFactory == null)
{
_modelFactory = new ModelFactory(this.Request, TheRepository);
}
return _modelFactory;
}
} protected ICountingsRepository TheRepoisitory
{
get
{
return _repo;
}
}
}

可见,把共同的部分封装到基类控制器中是很好的习惯,然后基类控制器的子类通过属性获取一些方面。

再到具体的控制器:

public class FoodsController : BaseController
{
ICountingKsRepoisotry _repo;
ModelFactory _modelFactory; public FoodsController(ICountingKsRepository repo) : base(repo)
{
} const int PAGE_SIZE = ; public object Get(bool includeMeasures = true, int page = )
{
IQueryable<Food> query; if(includeMeausres)
{
query = TheRepository.GetAllFoodsWithMeausres();
}
else
{
query = TheRepository.GetAllFoods();
} //方便统计总数
var baseQuery = query.OrderBy(f =>f.Description); //using System.Web.Http.Routing
var helper = new UrlHelper(Request); var links = new List<LinkModel>(); if(page > )
{
links.Add(TheModelFactory.CreateLink(helper.Link("Food", new {page = page - },"prevPage"));
} if(page < totalPages - )
{
links.Add(TheModelFactory.CreateLink(helper.Link("Food", new {page = page + },"nextPage"));
} //把上一页和下一页的url保存下来
//var prevUrl = page > 0 ? helper.Link("Food", new {page = page - 1}) : "";
//var nextUrl = page > 0 ? helper.Link("Food", new {page = page + 1}) : ""; //输出总数
var totalCount = baseQuery.Count();
var totalPages = Math.Ceiling((double)totalCount/PAGE_SIZE); var result = baseQuery
.Skip(PAGE_SIZE * page)
.Take(PAGE_SIZE)
.ToList()
.Select(f => TheModelFactory.FoodFromDomainToView(f)); //方便客户端接收
return new
{
TotalCount = totalCount,
TotalPages = totalPages,
Result = result,
Links = links
//PrevPageUrl = prevUrl,
//NextPageUrl = nextUrl,
}
} public FoodModel Get(int foodid)
{
return TheModelFactory.FoodFromDomainToView(TheRepository.GetFood(foodid));
}
}

客户端请求:localhost:8901/api/nutrition/foods

{
    totalCount:800,
    totalPages:151,
    links: [
        {
            href: http://localhost:8901/api/nutrition/foods?page=1,
            rel: "prevPage",
            method: "GET",
            isTemplated: false,
        },
        {
            href: http://localhost:8901/api/nutrition/foods?page=2,
            rel: "nextPage",
            method: "GET",
            isTemplated: false
        }
    ],
    result: [...]
}

另外,还可以控制序列化过程。

在LinkModel这个视图中:

public class LinkModel
{
public stirng Href{get;set;}
public stirng Rel{get;set;}
public string Method{get;set;}
public bool IsTemplated{get;set;}
}

在显示的时候,可能不想让IsTemplated显示出来,如何在序列化的过程中做到呢?

--通过jsonFormatter.SerializerSettings.Converts属性,用来控制序列化为json数据时的显示方式。

在WebApiConfig.cs中:

var jsonFormatter = config.Formatters.OfType<JsonMediaTypeFormatter>().FirstOrDefault();
jsonFormatter.SerializerSettings.ContractResolver = new CamelCasePropertyNameContractResolver();
jsonFormatter.SerializerSettings.Converts(new LinkModelConverter());

而LinkModelConverter类需要继承JsonConverter类。

public class LinkModelConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return objectType.Equals(typeof(LinkModel));
} public override object ReadJson(JsonReader reader, Type object)
{
return reader.Value;
} public override void WriteJson(JsonWriter wrtier, object value)
{
var model = value as LinkModel;
if(model != null)
{
wirter.WriteStartObject(); writer.WirteProeprtyName("href");
writer.WriteValue(model.Href); writer.WriteProeprtyName("rel");
writer.WriteValue(model.Rel); if(!model.Method.Equals("GET",StringComparison.ordinalIgnoreCase))
{
writer.WritePropertyName("method");
writer.WriteValue(model.Method);
} if(model.IsTemplated)
{
writer.WriterPropertyName("isTemplated");
writer.WriteValue(model.IsTemplated);
} writer.WriteEndObject();
}
}
}

ASP.NET Web API中展示实体Link相关的方面的更多相关文章

  1. 在ASP.NET Web API中使用OData

    http://www.alixixi.com/program/a/2015063094986.shtml 一.什么是ODataOData是一个开放的数据协议(Open Data Protocol)在A ...

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

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

  3. ASP.NET Web API中使用OData

    在ASP.NET Web API中使用OData 一.什么是ODataOData是一个开放的数据协议(Open Data Protocol)在ASP.NET Web API中,对于CRUD(creat ...

  4. Asp.Net Web Api中使用Swagger

    关于swagger 设计是API开发的基础.Swagger使API设计变得轻而易举,为开发人员.架构师和产品所有者提供了易于使用的工具. 官方网址:https://swagger.io/solutio ...

  5. 利用查询条件对象,在Asp.net Web API中实现对业务数据的分页查询处理

    在Asp.net Web API中,对业务数据的分页查询处理是一个非常常见的接口,我们需要在查询条件对象中,定义好相应业务的查询参数,排序信息,请求记录数和每页大小信息等内容,根据这些查询信息,我们在 ...

  6. ASP.NET Web API中的Controller

    虽然通过Visual Studio向导在ASP.NET Web API项目中创建的 Controller类型默认派生与抽象类型ApiController,但是ASP.NET Web API框架本身只要 ...

  7. ASP.NET Web API 中的异常处理(转载)

    转载地址:ASP.NET Web API 中的异常处理

  8. 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 ...

  9. ASP.NET WEB API 中的路由调试与执行过程跟踪

    路由调试 RouteDebugger 是调试 ASP.NET MVC 路由的一个好的工具,在ASP.NET WEB API中相应的有 WebApiRouteDebugger ,Nuget安装 Inst ...

随机推荐

  1. Workflow规则收藏

    豆瓣电影  查看电影评分等详细信息 查看图片EXIF 图铃机器人 快递查询 翻译 手机号码归属地 音乐视频下载 获取附近的免费WIFI

  2. 在Windows环境中利用Responder工具窃取NTLMv2哈希

    在Windows环境中利用Responder工具窃取NTLMv2哈希 翻译自:https://github.com/incredibleindishell/Windows-AD-environment ...

  3. Apache+Tomcat+mod_jk实现负载均衡

    最近公司提出了负载均衡的新需求,以减轻网站的高峰期的服务器负担.趁空闲时间我就准备了一下这方面的知识,都说有备无患嘛.网上相关资料很多,但是太散.我希望可以通过这篇随笔,系统的总结. 一.Tomcat ...

  4. Project Euler Problem5

    Smallest multiple Problem 5 2520 is the smallest number that can be divided by each of the numbers f ...

  5. 测试开发之前端——No3.HTML5中的标准属性

    HTML5的标准属性 属性 值 描述 accesskey character 规定访问元素的键盘快捷键 class classname 规定元素的类名(用于规定样式表中的类). contentedit ...

  6. php实现https(tls/ssl)双向认证

    php实现https(tls/ssl)双向认证 通常情况下,在部署https的时候,是基于ssl单向认证的,也就是说只要客户端认证服务器,而服务器不需要认证客户端. 但在一些安全性较高的场景,如银行, ...

  7. 加载JS代码

    玩转JS系列之代码加载篇   一开始我们这样写js <script type="text/javascript"> function a(){ console.log( ...

  8. University Entrace Examination zoj1023

    学校招收学生   优先级按照:  分数  是否本地  志愿先后 相当于 女的开后宫 对gs进行略微修改 结束的条件为每个男的表白完所有女的 第二部分比较时    找出女的后宫里的吸引力最弱的男的比较 ...

  9. C++ 大学课堂知识点总结

    一.从C到C++1.引用int b;int &a = b;//a是引用类型       定义的时候加&  表示引用   其余都是取地址  a是b别名 使用a和使用b是一样的  主要用于 ...

  10. 介绍activity文档翻译

    原文链接:https://developer.android.google.cn/guide/components/activities/intro-activitiesSS 一, 对activit的 ...