对我的WebAPI的每个调用都可能(也可能不)包含GET参数
/api/SomeControllerFunction?loglevel=(someint)
通过控制器内部的功能,我可以初始化LogCollector:
[HttpGet]
SomeControllerFunction(int loglevel = 0)
{
LogCollector logger = new LogCollector(loglevel)
}
为了不经常重复,我想通过将其添加到BaseController的构造函数中来将其隐藏在类层次结构中,我的所有控制器都将从中继承:
public class BaseController: ApiController
{
internal LogCollector Logger
BaseController()
{
Logger = new LogCollector(loglevel);
}
但是,我如何从构造函数访问GET参数呢?
您可以将LogCollector
直接注入到方法中,而不是使用构造函数。如果你确实想使用构造函数,你应该使用Di/IoC框架,因为这更合适。
在下面的示例中,您可以使用自定义ActionFilterAttribute
实例,该实例基于传入(可选)日志级别注入Logger。然后使用动作上的RouteAttribute
在路由中定义日志级别。RouteAttribute
还定义了日志级别的默认值,因此在调用该操作时不需要该值。
LogInjectorFilterAttribute.cs
public class LogInjectorFilterAttribute : ActionFilterAttribute
{
public override void OnActionExecuting(HttpActionContext actionContext)
{
const string key = "loglevel";
if(actionContext.ControllerContext.RouteData.Values.ContainsKey(key))
{
var loglevel = int.Parse(actionContext.ControllerContext.RouteData.Values[key].ToString());
LogCollector logger = new LogCollector(loglevel);
actionContext.ActionArguments["logger"] = logger;
}
base.OnActionExecuting(actionContext);
}
}
HomeController.cs
[HttpGet]
[Route("api/Home/Get/{loglevel:int=1}")]
[LogInjectorFilter]
public IHttpActionResult Get(LogCollector logger)
{
}
构造函数调用得太早,您无法从那里访问参数。但是,您可以覆盖Initialize
方法,并从上下文中检索GET参数:
protected override void Initialize(HttpControllerContext controllerContext)
{
foreach (var parameter in controllerContext.Request.GetQueryNameValuePairs())
{
Debug.WriteLine(string.Format("{0} = {1}", parameter.Key, parameter.Value));
}
base.Initialize(controllerContext);
}