如何让ASP.NET Web API的Action方法在希望的Culture下执行
在今天编辑推荐的《Hello Web API系列教程——Web API与国际化》一文中,作者通过自定义的HttpMessageHandler的方式根据请求的Accep-Language报头设置当前线程UI Culture的方式来解决Localization的问题。如果你对ASP.NET Web API的执行机制有足够了解的话,你会发现实际上有很多种解决方案。不过这些解决方案都不够完美,原因很简单:ASP.NET Web API的整个框架均采用基于Task的并行编程模式,所以每个可扩展组件均可以在不同的线程中执行,这样会导致我们没有办法100%控制目标方法真正执行的线程的UI Culture。不过在默认情况下,大部分组件是按照同步的方式执行的,所以我们之需要在目标Action方法执行之前设置当前线程的UI Culture即可。
一、两个辅助的扩展方法我们针对HttpRequestMessage定义了如下两个扩展方法。SetCurrentUICulture从请求的Accpet-Language报头提取客户端接受的语言并据此设置当前线程的UI Culture。在这之前,它会将当前线程的UI Culture保存到HttpRequestMessage对象中。ResetCurrentUICulture方法将这个CultureInfo对象从HttpRequestMessage其中提取出来,将当前线程的UI Cuilture回复到之前的状态。 1: public static class HttpRequestMessageExtensions 2: {
3: void SetCurrentUICulture(this HttpRequestMessage request) 4: {
5: StringWithQualityHeaderValue acceptCultureHeader = request.Headers.AcceptLanguage.OrderByDescending(header => header.Quality).FirstOrDefault();
6: if (null != acceptCultureHeader) 7: {
8: request.Properties["__CurrentCulture"] = Thread.CurrentThread.CurrentUICulture; 9: Thread.CurrentThread.CurrentUICulture = new CultureInfo(acceptCultureHeader.Value); 10: }
11: }
12:?
13: void ResetCurrentUICulture(this HttpRequestMessage request) 14: {
15: object culture; 16: if (request.Properties.TryGetValue("__CurrentCulture",out culture)) 17: {
18: Thread.CurrentThread.CurrentUICulture = (CultureInfo)culture;
19: }
20: }
21: }
二、第1种方案:自定义ActionFilter我想这应该是大家最容易想到的解决方案,因为ActionFilter可以注册一些回调操作在目标Action方法执行前后被自动调用。为此我们定义了如下一个继承自ActionFilterAttribute的UseAcceptCultureAttribute类型。我们分别在重写的OnActionExecuting和OnActionExecuted方法中利用上面定义的两个扩展方法对当前线程的UI Culture进行设置和恢复。 class UseAcceptCultureAttribute: ActionFilterAttribute
override void OnActionExecuting(HttpActionContext actionContext) 5: actionContext.Request.SetCurrentUICulture();
6: }
7:?
8: void OnActionExecuted(HttpActionExecutedContext actionExecutedContext) 9: {
10: actionExecutedContext.Request.ResetCurrentUICulture();
12: }
为了验证这个ActionFilterAttribute特性,我们定义了如下一个继承自ApiController的HelloController。唯一的Action方法返回的字符串是从资源文件中提取的(类型Resources为资源文件自动生成的类型),而ActionFilterAttribute就应用在这个Get方法上。 class ExtendedReflectedHttpActionDescriptor : ReflectedHttpActionDescriptor
public ExtendedReflectedHttpActionDescriptor(ReflectedHttpActionDescriptor actionDescriptor)
4: : base(actionDescriptor.ControllerDescriptor,actionDescriptor.MethodInfo) 5: { }
6: override Task<object> ExecuteAsync(HttpControllerContext controllerContext,IDictionary<string,1)">object> arguments,CancellationToken cancellationToken) 7: {
8: controllerContext.Request.SetCurrentUICulture();
9: Task<object> task = base.ExecuteAsync(controllerContext,arguments,cancellationToken); 10: controllerContext.Request.ResetCurrentUICulture();
11: return task; 12: }
13: }
ASP.NET Web API利用一个名为HttpActionSelector的对象来选择与当前请求匹配的HttpActionDescriptor,要让我们自定义的ExtendedReflectedHttpActionDescriptor被使用,我们得对应的HttpActionSelector。ASP.NET Web API默认使用的HttpActionSelector类型为ApiControllerActionSelector,我们自定义的ExtentedApiControllerActionSelector就继承于它。如下面的代码片断所示,在重写的SelectAction方法中,我们调用基类的同名方法得到一个ReflectedHttpActionDescriptor 对象,并根据它创建一个ExtendedReflectedHttpActionDescriptor 对象并返回。 class WebApiApplication : System.Web.HttpApplication
protected void Application_Start() 5: GlobalConfiguration.Configuration.Services.Replace(typeof(IHttpActionSelector),1)">new ExtentedApiControllerActionSelector()); 6: //... 7: }
8: }
四、第3种方案:自定义HttpActionInvoker目标Action的执行是通过一个名为HttpActionInvoker驱动执行的(它调用HttpActionDescriptor的ExecuteAsync方法),默认的HttpActionInvoker类型为ApiControllerActionInvoker。我们可以继承它,并在执行目标Action方法前后设置和恢复当前线程的UI Culture。为此我定义了如下一个ExtendedApiControllerActionInvoker,在重写的InvokeActionAsync方法中,我们调用基类的同名方法执行目标Action方法,并在这前后分别调用当前HttpRequestMessage的两个扩展方法设置和恢复当前线程的UI Culture。 typeof(IHttpActionInvoker),1)">new ExtendedApiControllerActionInvoker());
6: //... 8: }
五、第4种方案:为HttpController创建一个基类HttpActionInvoker的最终又是在执行HttpController时被调用的,所以我们可以在执行HttpController上作文章。所以我们定义了如下一个继承自ApiController的ExtendedApiController 类型。在重写的ExecuteAsync方法中,我们调用基类同名方法前后对当前线程的UI Culture进行了设置和恢复。 相关内容
推荐文章
站长推荐
热点阅读
|