// 在program.cs中加入这句,加入缓存支持
builder.Services.AddMemoryCache();
public class Book
{
public int Id { get; init; }
public string Title { get; init; }
public Book(int id, string title)
{
Id = id;
Title = title;
}
public override string ToString()
{
return $"book:{Id}, {Title}";
}
}
public class BookDbContext
{
public Task<Book?> GetBookByIdAsync(int id)
{
Book? book = null;
switch (id)
{
case 1:
book = new Book(1, "book1");
break;
case 2:
book = new Book(2, "book2");
break;
case 3:
book = new Book(3, "book3");
break;
}
return Task.Fromresult(book);
}
}
[ApiController]
[Route("api/[controller]")]
public class CacheTestController : ControllerBase
{
private IMemoryCache MemoryCache { get; init; }
private ILogger<CacheTestController> Logger { get; set; }
public CacheTestController(IMemoryCache memoryCache, ILogger<CacheTestController> logger)
{
MemoryCache = memoryCache;
Logger = logger;
}
[HttpGet("book")]
public async Task<ActionResult<Book?>> GetBookById(int id)
{
Logger.Log@R_723_4045@ion($"enter GetBookById, id={id}");
// 尽量使用GetorCreateAsync,而不是Get/Set。
// 因为GetorCreateAsync可以把null也作为缓存值,这样就可以避免缓存穿透。
Book? book = await MemoryCache.GetorCreateAsync($"book_{id}", (e) =>
{
Logger.Log@R_723_4045@ion($"not found in cache, search in BookDbContext.");
// 混用SlidingExpiration和AbsoluteExpirationRelativetoNow,这样滑动超时或绝对超时后,缓存都会失效,
e.SlidingExpiration = TimeSpan.FromSeconds(5);
// 使用随机数作为超时值,避免缓存雪崩
e.AbsoluteExpirationRelativetoNow = TimeSpan.FromSeconds(Random.Shared.Next(10, 50));
return new BookDbContext().GetBookByIdAsync(id);
});
Logger.Log@R_723_4045@ion($"GetorCreateAsync returns {book}");
if (book == null)
return NotFound($"cannot find bood with id:{id}");
return book;
}
[responsecache(Duration = 10)] // 浏览器缓存支持,在响应头里加入cache-control: public,max-age=10
[HttpGet("Now")]
public DateTime Now()
{
return DateTime.Now;
}
}
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。