前言
当我们使用DI方式写了很多的Service后, 可能会发现我们的有些做法并不是最优的.
获取注入的对象, 大家经常在构造函数中获取, 这样也是官方推荐的方式, 但有时不是效率最高的方法.
如果在构造函数中获取对象,那么每次对象的初始化都会把构造函数中的对象初始化一遍, 如果某个方法只用到其中一个注入对象, 那么其他的注入对象就白注入了
注入方法:
分别为构造函数、方法特性(FromServices)使用注入, 还有今天的主角 手动获取服务注入
构造函数注入
/// <summary>
/// xxx服务
/// </summary>
[Route("api/[controller]")]
[ApiController]
public class ValueController : ControllerBase
{
private readonly ILogger<ValueController> _logger;
private readonly IFreeSql _freeSql;
private readonly IOptions<ValueOptions> _valueOptions;
private readonly RoadService _roadService;
/// <summary>
/// 构造函数
/// </summary>
/// <returns></returns>
public RollerController(ILogger<ValueController> logger, IFreeSql freeSql, IOptions<ValueOptions> valueOptions, RoadService roadService)
{
_logger = logger;
_freeSql = freeSql;
_baiduOptions = baiduOptions;
_roadService = roadService;
}
/// <summary>
/// 获取所有项目
/// </summary>
/// <returns></returns>
[HttpGet("project/all")]
public Task<List<ProjectDto>> GetProjects() => _freeSql.Select<Project>().ToListAsync<ProjectDto>();
/// <summary>
/// 获取工程名下的道路
/// </summary>
/// <param name="projectId"></param>
/// <returns></returns>
[HttpGet("project/{projectId}/road")]
public Task<List<RoadDto>> GetRoads([FromRoute] Guid projectId) => _roadService.GetRoads(projectId);
}
方法特性(FromServices)注入
/// <summary>
/// 上传数据
/// </summary>
/// <param name="gridDataService"></param>
/// <param name="dataList"></param>
/// <param name="roadId"></param>
/// <returns></returns>
[HttpPost("road/upload-grid-data/{roadId}")]
public async Task<IActionResult> UpdateRoadGridData([FromServices] GridDataService gridDataService, [FromBody] List<BizRoadGridInput> dataList, Guid roadId)
{
var data = await gridDataService.UploadGridData(dataList, roadId);
if (!data.ok) return BadRequest(data.message);
return Ok();
}
如上: 每个方法并不是都用到了构造函数中的服务, 所以我们这里就有性能损失, 毕竟创建对象也是有代价的, 而且还会伴有GC.
手动获取服务注入
在不是Controller中就不能使用[FromServices]特性了
为了能在个别的方法中注入对象就要用到手动获取注册对象的方式
如下:
在能直接拿到HttpContext时
/// <summary>
/// 处理道路信息
/// </summary>
/// <param name="roadId"></param>
/// <exception cref="ArgumentNullException"></exception>
[HttpPost("road/{roadid}/handle-road")]
public async Task<ActionResult> HandleRoadInfo([FromRoute] Guid roadId)
{
var imageService = HttpContext.GetRequiredService.GetRequiredService<ImageService>();
await imageService.HandleRoadInfo(roadId);
return Ok();
}
在不能直接拿到HttpContext时,需要先获取IHttpContextAccessor
, 然后再获取 HttpContext
上下文对象
/// <summary>
/// 图片服务
/// </summary>
public class ImageService
{
private readonly IHttpContextAccessor _httpContextAccessor;
/// <summary>
/// 构造函数
/// </summary>
public ImageService(IHttpContextAccessor httpContextAccessor)
{
_httpContextAccessor = httpContextAccessor;
}
/// <summary>
/// 处理道路信息
/// </summary>
/// <param name="roadId"></param>
/// <exception cref="ArgumentNullException"></exception>
public async Task HandleRoadInfo(Guid roadId)
{
var roadService = _httpContextAccessor.HttpContext?.RequestServices.GetRequiredService<RoadService>();
var data = await roadService.GetRoadData(roadId);
// 省略其他
}
}
注意这里面我们使用了GetRequiredService
, 而没有使用GetService
, 因为使用GetRequiredService
不需要我们自己再去做空值检查, 如果为空再很快就会失败.
关于GetService 和 GetRequiredService
GetService()是IServiceProvider上的唯一方法,ISeviceProvider是ASP.NET核心DI抽象中的中央接口。第三方容器还可以实现可选接口ISupportRequiredService,该接口提供GetRequiredService()方法。当请求的类型serviceType可用时,这些方法的行为相同。如果服务不可用(即它没有注册),则GetService()返回null,而GetRequiredService()抛出一个InvalidOperationException。
GetRequiredService()相对于GetService()的主要好处是当服务不可用时,它允许第三方容器提供额外的诊断信息。因此,在使用第三方容器时最好使用GetRequiredService()。就个人而言,我会在任何地方使用它,即使我只使用内置的DI容器。
总结
将所有方法都用的service中使用构造函数注入是个优选方案, 将个别方法使用到的service使用手动获取的方式代码执行会更有效率
作者:wwmin
联系:wwei.min@163.com
欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权利。如有问题或建议,请多多赐教,非常感谢。
网友评论