百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

ABP vNext框架文档解读1-缓存(abp框架实现前后端分离了吗)

ccwgpt 2024-09-20 13:35 41 浏览 0 评论

Volo.Abp.Caching Package

Volo.Abp.Caching是缓存系统的核心包, 可以通过Nuget PMC安装到项目:

Install-Package Volo.Abp.Caching

IDistributedCache

ASP.NET Core 定义了 IDistributedCache 接口用于 get/set 缓存值 . 但是会有以下问题:

  • 它适用于 byte 数组 而不是 .NET 对象. 因此你需要对缓存的对象进行序列化/反序列化.
  • 它为所有的缓存项提供了单个 key 池 , 因此你需要注意区分不同类型的对象.

IDistributedCache 定义在 Microsoft.Extensions.Caching.Abstractions 包中.

接口的默认实现是 MemoryDistributedCache,它使用内存工作.

IDistributedCache<TCacheItem>

ABP在Volo.Abp.Caching包中定义了通用的泛型接口IDistributedCache<TCacheItem>, TCacheItem是存储在缓存中的对象类型, 其使用方法与ASP.NET Core的IDistributedCache 接口相同.

IDistributedCache<TCacheItem, TCacheKey>

IDistributedCache<TCacheItem> 中的Key默认是 string 类型, 如果你的Key不是string类型, 需要进行手动类型转换, IDistributedCache<TCacheItem, TCacheKey> 将Key的类型泛型化, 以简化手动转换的操作, 其使用方法与ASP.NET Core的IDistributedCache 接口相同.

解决的痛点

IDistributedCache<TCacheItem>和IDistributedCache<TCacheItem, TCacheKey> 解决了IDistributedCache中存在的问题;

  • 它在内部 序列化/反序列化 缓存对象. 默认使用 JSON 序列化, 但可以替换依赖注入系统中 IDistributedCacheSerializer 服务的实现来覆盖默认的处理.
  • 它根据缓存中对象类型自动向缓存key添加 缓存名称 前缀. 默认缓存名是缓存对象类的全名(如果你的类名以CacheItem 结尾, 那么CacheItem 会被忽略,不应用到缓存名称上). 你也可以在缓存类上使用 CacheName 设置换缓存的名称.
  • 它自动将当前的租户id添加到缓存键中, 以区分不同租户的缓存项 (只有在你的应用程序是多租户的情况下生效). 在缓存类上应用 IgnoreMultiTenancy attribute, 可以在所有的租户间共享缓存.
  • 允许为每个应用程序定义 全局缓存键前缀 ,不同的应用程序可以在共享的分布式缓存中拥有自己的隔离池.

使用方法

定义缓存中存储项的示例类, 通常根据自身业务定义类中的各个属性.

public class BookCacheItem
{
        public string Name { get; set; }

        public decimal Price { get; set; }
}

将IDistributedCache<BookCacheItem>注入到需要的Service中

public class BookService:ITransientDependency
{
        private readonly IDistributedCache<BookCacheItem> _cache;

        public BookService(IDistributedCache<BookCacheItem> cache)
        {
            _cache = cache;
        }

        public async Task<BookCacheItem> GetAsycn(Guid bookId)
        {
            return await _cache.GetOrAddAsync(
                bookId.ToString(),
                async ()=>await GetBookFromDatabaseAsync(bookId),
                ()=>new DistributedCacheEntryOptions
                {
                    AbsoluteExpiration = DateTimeOffset.Now.AddHours(1)
                });
        }

        public async Task<BookCacheItem> GetBookFromDatabaseAsync(Guid bookId)
        {
            //TODO: get from database
        }
}
  • 示例Service中的 GetOrAddAsync() 方法从缓存中获取数据.
  • 如果没有在缓存中找到数据,它会调用工厂方法 (本示例中是 GetBookFromDatabaseAsync)从原始数据源中获取数据.
  • GetOrAddAsync 有一个可选参数 DistributedCacheEntryOptions , 可用于设置缓存的生命周期.

如果缓存项中Key的类型是Guid, 则将IDistributedCache<BookCacheItem,Guid>注入到需要的Service中

public class BookService : ITransientDependency
{
    private readonly IDistributedCache<BookCacheItem, Guid> _cache;

    public BookService(IDistributedCache<BookCacheItem, Guid> cache)
    {
        _cache = cache;
    }

    public async Task<BookCacheItem> GetAsync(Guid bookId)
    {
        return await _cache.GetOrAddAsync(
            bookId, //Guid type used as the cache key
            async () => await GetBookFromDatabaseAsync(bookId),
            () => new DistributedCacheEntryOptions
            {
                AbsoluteExpiration = DateTimeOffset.Now.AddHours(1)
            }
        );
    }
    private Task<BookCacheItem> GetBookFromDatabaseAsync(Guid bookId)
    {
        //TODO: get from database
    }
}
  • 示例服务中 GetOrAddAsync() 方法获取缓存的数据.
  • 我们采用了 Guid 作为Key, 在 _cache.GetOrAddAsync() 方法中传入 Guid 类型的bookid

IDistributedCache<TCacheItem, TCacheKey> 在内部使用Key对象的 ToString() 方法转换类型为string. 如果你的将复杂对象作为Key,那么需要重写类的 ToString 方法.

public class UserInOrganizationCacheKey
{
    public Guid UserId { get; set; }
 
    public Guid OrganizationId { get; set; }

    //构建缓存key
    public override string ToString()
    {
        return #34;{UserId}_{OrganizationId}";
    }
}
public class BookService : ITransientDependency
{
    private readonly IDistributedCache<UserCacheItem, UserInOrganizationCacheKey> _cache;

    public BookService(
        IDistributedCache<UserCacheItem, UserInOrganizationCacheKey> cache)
    {
        _cache = cache;
    }

    ...
}

写在结尾

每次看到这样的文章, 总想自己着手建个项目(Web或者Console)把代码run一遍, 但是文章中的代码总是松散不完整的. 等到自己去解读一些框架时,才发现原因可能有以下几点:

  1. 文章需要建立关注点, 如果写得大而全, 往往会让读者失去关注点,导致容易忽视作者要透露的关键信息.
  2. 知识点的牵扯面较多, 比如我想写个控制台项目去run一遍代码的时候, 发现如何注入IDistributedCache<TCacheItem>时, 就牵扯到了module, 咱们下一篇继续解读.
class Program
{
        static async Task Main(string[] args)
        {
            using var host = CreateHostBuilder(args).Build();
            
            var bookService = host.Services.GetRequiredService<BookService>();
            var book = await bookService.GetAsycn(Guid.Parse("7a56d60b-3c2b-46de-b280-ba3ceae51d3d"));

            await host.RunAsync();
        }

        static IHostBuilder CreateHostBuilder(string[] args) =>
            Host.CreateDefaultBuilder(args)
                .ConfigureServices((_, services)=>
                    services.AddTransient<BookService>()
                       .AddApplication<AbpCachingModule>());//引入缓存模块
 }
  1. 当从数据库中读取出测试数据时, 需要把Entity的属性值赋值给EntityCacheItem时, 又牵扯到了AutoMapper这类对象到对象的映射工具, 如果都按照我下面这样写代码的话, 就太累了...
public class BookCacheItem
{
        public string Name { get; set; }
        public decimal Price { get; set; }

        public static BookCacheItem FromBook(Book item)
        {
            return new BookCacheItem
            {
                Name = item.Name,
                Price = item.Price
            };
        }
}
  1. 如果不run一遍代码的话, 也可以通过单元测试的方式测试.

相关推荐

Spring WebFlux vs. Spring MVC(springboot是什么)

背景随着异步I/O和Netty等框架的流行,响应式编程逐渐走入大众的视野。但是,响应式编程本身并不是太新的概念,这个术语最早出现在1985年DavidHarel和AmirPnue...

深度解析微服务高并发:适配SpringMVC框架适配模块及实现原理

适配主流框架如果不借助Sentinel提供的适配主流框架的模块,则在使用Sentinel时需要借助try-catchfinally将要保护的资源(方法或代码块)包起来,在目标方法或代码块执行之前,调...

Spring MVC 底层原理深度解析:从请求到响应的全链路拆解

一、Servlet容器与DispatcherServlet的启动博弈1.Tomcat初始化阶段java//Tomcat初始化流程StandardContext#startInterna...

改造总结之传统SpringMVC架构转换为SpringBoot再到集群

改造出发点,是基于现在服务都在向上云的目标前进,传统SpringMVC难以满足项目持续构建、服务节点任意扩展的需求,所以开始了历史项目的改造。项目改造考虑的主要是兼容以前的业务代码,以及session...

SpringBoot3 整合 Spring MVC 全解析:开启高效 Web 开发之旅

在当今的JavaWeb开发领域,Spring框架家族无疑占据着重要的地位。其中,SpringBoot3和SpringMVC更是开发者们构建强大、高效Web应用的得力工具。今天,...

一文读懂SpringMVC(一文读懂!残疾人低保边缘家庭能领的超实用福利政策)

1.SpringMVC定义1.1.MVC定义Model(模型):是应用程序中用于处理应用程序数据逻辑的部分。通常模型对象负责在数据库中存取数据View(视图):是应用程序中处理数据显示的部分。通常...

69 个Spring mvc 全部注解:真实业务使用案例说明(必须收藏)

SpringMVC框架的注解为Web开发提供了一种简洁而强大的声明式方法。从控制器的定义、请求映射、参数绑定到异常处理和响应构建,这些注解涵盖了Web应用程序开发的各个方面。它们不仅简化了编码工作,...

Spring MVC工作原理:像拼积木一样构建Web应用

SpringMVC工作原理:像拼积木一样构建Web应用在Java的Web开发领域,SpringMVC无疑是一个让人又爱又恨的存在。它像一位神通广大的积木搭建大师,将一个个分散的功能模块巧妙地拼接在...

5千字的SpringMVC总结,我觉得你会需要

思维导图文章已收录到我的Github精选,欢迎Star:https://github.com/yehongzhi/learningSummary概述SpringMVC再熟悉不过的框架了,因为现在最火的...

SpringMVC工作原理与优化指南(springmvc工作原理和工作流程)

SpringMVC工作原理与优化指南在现代Java开发中,SpringMVC无疑是构建Web应用程序的首选框架之一。它以其优雅的设计和强大的功能吸引了无数开发者。那么,SpringMVC究竟是如何工作...

Spring MVC框架源码深度剖析:从入门到精通

SpringMVC框架源码深度剖析:从入门到精通SpringMVC框架简介SpringMVC作为Spring框架的一部分,为构建Web应用程序提供了强大且灵活的支持。它遵循MVC(Model-V...

3000字搞明白SpringMVC工作流程、DispatcherServlet类、拦截器!

SpringMVC基础虽然SpringBoot近几年发展迅猛,但是SpringMVC在Web开发领域仍然占有重要的地位。本章主要讲解SpringMVC的核心:DispatcherServlet类...

多年经验大佬用2000字透彻解析SpringMVC的常用注解及相关示例

SpringMVC注解SpringMVC框架提供了大量的注解,如请求注解、参数注解、响应注解及跨域注解等。这些注解提供了解决HTTP请求的方案。本节主要讲解SpringMVC的常用注解及相关示例...

知乎热议:如何成为前端架构师,赚百万年薪?

作者|慕课网精英讲师双越最近有一条知乎热议:从一个前端工程师,如何根据目标,制定计划,才能快速进阶成为前端架构师?不久之前我参与了一次直播,讲到了自己对于Web前端架构师的理解。架构师这个角色...

学习笔记-前端开发架构设计(前端架构设计方案)

前端开发的技术选项主要包含以下几点,下面对一些名词概念的解释做了笔记:1、分层架构:把功能相似,抽象级别相近的实现进行分层隔离优势:松散耦合(易维护,易复用,易扩展)常见分层方式:MVC,MVVM2、...

取消回复欢迎 发表评论: