RESTful的版本化有四种基本方法:
URI路径方式,形式如下:
http://api/v2/Tasks/{TaskId}
URI参数方式,形式如下:
http://api/Tasks/{TaskId}?v=2
内容协商方式,通过HTTP头进行,例如:
Content Type: application/vnd.taskManagerApp.v2.param.json
请求头方式,也是通过HTTP头进行,例如:
x-taskManagerApp-version: 2
我个人比较喜欢第一种方式。你可以阅读 Mike Wasson的ASP.NET Web API: Using Namespaces to Version Web APIs。
许多人对Mike Wasson的原始代码进行了修改,我喜欢Jamie Kurtz和Brian Wortman所著的ASP.NET Web API 2书中使用的那个版本。
由于它涉及到太多的组件,我在GitHub上创建了一个示例项目。
config.Routes.MapHttpRoute(
name: "DefaultApi",
routeTemplate: "api/{version}/{controller}",
defaults: new { version = "v2" }
);
config.Routes.MapHttpRoute(
name: "DefaultApiWithId",
routeTemplate: "api/{version}/{controller}/{id}",
defaults: new { id = RouteParameter.Optional }
);
然后,您需要添加 ApiVersionConstraint
public class ApiVersionConstraint : IHttpRouteConstraint
{
public ApiVersionConstraint(string allowedVersion)
{
AllowedVersion = allowedVersion.ToLowerInvariant();
}
public string AllowedVersion { get; private set; }
public bool Match(HttpRequestMessage request, IHttpRoute route, string parameterName,
IDictionary<string, object> values, HttpRouteDirection routeDirection)
{
object value;
if (values.TryGetValue(parameterName, out value) && value != null)
{
return AllowedVersion.Equals(value.ToString().ToLowerInvariant());
}
return false;
}
}
使用方法
只需将RoutePrefix放在控制器上即可完成。
[RoutePrefix("api/{apiVersion:apiVersionConstraint(v1)}/values")]
public class ValuesController : ApiController
{
[Route("")]
public IEnumerable<string> Get()
{
return new string[] { "v1-value1", "v1-value2" };
}
[Route("{id}")]
public string Get(int id)
{
return "v1-value-" + id;
}
}