什么是AspectCore Project ?

AspectCore Project 是适用于Asp.Net Core 平台的轻量级 Aop(Aspect-oriented programming) 解决方案,它更好的遵循Asp.Net Core的模块化开发理念,使用AspectCore可以更容易构建低耦合、易扩展的Web应用程序。AspectCore使用Emit实现高效的动态代理从而不依赖任何第三方Aop库。

开使使用AspectCore

  • 启动 Visual Studio。从 File 菜单, 选择 New > Project。选择 ASP.NET Core Web Application 项目模版,创建新的 ASP.NET Core Web Application 项目。

  • 从 Nuget 安装 AspectCore.Extensions.DependencyInjection package:

    PM>   Install-Package AspectCore.Extensions.DependencyInjection
  • 在一般情况下可以使用抽象的AbstractInterceptorAttribute 自定义特性类,它实现IInterceptor接口。AspectCore默认实现了基于Attribute的拦截器配置。我们的自定义拦截器看起来像下面这样:

    public class CustomInterceptorAttribute : AbstractInterceptorAttribute
    {
    public async override Task Invoke(AspectContext context, AspectDelegate next)
    {
    try
    {
    Console.WriteLine("Before service call");
    await next(context);
    }
    catch (Exception)
    {
    Console.WriteLine("Service threw an exception!");
    throw;
    }
    finally
    {
    Console.WriteLine("After service call");
    }
    }
    }
  • 定义ICustomService接口和它的实现类CustomService:

    public interface ICustomService
    {
    [CustomInterceptor]
    void Call();
    } public class CustomService : ICustomService
    {
    public void Call()
    {
    Console.WriteLine("service calling...");
    }
    }
  • HomeController中注入ICustomService:

    public class HomeController : Controller
    {
    private readonly ICustomService _service;
    public HomeController(ICustomService service)
    {
    _service = service;
    } public IActionResult Index()
    {
    _service.Call();
    return View();
    }
    }
  • 注册ICustomService,接着,在ConfigureServices中配置创建代理类型的容器:

    public IServiceProvider ConfigureServices(IServiceCollection services)
    {
    services.AddTransient<ICustomService, CustomService>();
    services.AddMvc();
    services.AddDynamicProxy();
    return services.BuildAspectCoreServiceProvider();
    }
  • 拦截器配置。

    全局拦截器。使用AddDynamicProxy(Action<IAspectConfiguration>)的重载方法,其中IAspectConfiguration提供Interceptors注册全局拦截器:

     services.AddDynamicProxy(config =>
    {
    config.Interceptors.AddTyped<CustomInterceptorAttribute>();
    });

    带构造器参数的全局拦截器,在CustomInterceptorAttribute中添加带参数的构造器:

    public class CustomInterceptorAttribute : AbstractInterceptorAttribute
    {
    private readonly string _name;
    public CustomInterceptorAttribute(string name)
    {
    _name = name;
    }
    public async override Task Invoke(AspectContext context, AspectDelegate next)
    {
    try
    {
    Console.WriteLine("Before service call");
    await next(context);
    }
    catch (Exception)
    {
    Console.WriteLine("Service threw an exception!");
    throw;
    }
    finally
    {
    Console.WriteLine("After service call");
    }
    }
    }

    修改全局拦截器注册:

    services.AddDynamicProxy(config =>
    {
    config.Interceptors.AddTyped<CustomInterceptorAttribute>(args: new object[] { "custom" });
    });

    作为服务的全局拦截器。在ConfigureServices中添加:

    services.AddTransient<CustomInterceptorAttribute>(provider => new CustomInterceptorAttribute("service"));

    修改全局拦截器注册:

    services.AddDynamicProxy(config =>
    {
    config.Interceptors.AddServiced<CustomInterceptorAttribute>();
    });

    作用于特定ServiceMethod的全局拦截器,下面的代码演示了作用于带有Service后缀的类的全局拦截器:

    services.AddDynamicProxy(config =>
    {
    config.Interceptors.AddTyped<CustomInterceptorAttribute>(method => method.DeclaringType.Name.EndsWith("Service"));
    });

    使用通配符的特定全局拦截器:

    services.AddDynamicProxy(config =>
    {
    config.Interceptors.AddTyped<CustomInterceptorAttribute>(Predicates.ForService("*Service"));
    });
  • 在AspectCore中提供NonAspectAttribute来使得ServiceMethod不被代理:

    [NonAspect]
    public interface ICustomService
    {
    void Call();
    }

    同时支持全局忽略配置,亦支持通配符:

     services.AddDynamicProxy(config =>
    {
    //App1命名空间下的Service不会被代理
    config.NonAspectPredicates.AddNamespace("App1"); //最后一级为App1的命名空间下的Service不会被代理
    config.NonAspectPredicates.AddNamespace("*.App1"); //ICustomService接口不会被代理
    config.NonAspectPredicates.AddService("ICustomService"); //后缀为Service的接口和类不会被代理
    config.NonAspectPredicates.AddService("*Service"); //命名为Query的方法不会被代理
    config.NonAspectPredicates.AddMethod("Query"); //后缀为Query的方法不会被代理
    config.NonAspectPredicates.AddMethod("*Query");
    });
  • 拦截器中的依赖注入。在拦截器中支持属性注入,构造器注入和服务定位器模式。

    属性注入,在拦截器中拥有public get and set权限的属性标记[AspectCore.Injector.FromContainerAttribute]特性,即可自动注入该属性,如:

    public class CustomInterceptorAttribute : AbstractInterceptorAttribute
    {
    [FromContainer]
    public ILogger<CustomInterceptorAttribute> Logger { get; set; } public override Task Invoke(AspectContext context, AspectDelegate next)
    {
    Logger.LogInformation("call interceptor");
    return next(context);
    }
    }

    构造器注入需要使拦截器作为Service,除全局拦截器外,仍可使用ServiceInterceptor使拦截器从DI中激活:

    public interface ICustomService
    {
    [ServiceInterceptor(typeof(CustomInterceptorAttribute))]
    void Call();
    }

    服务定位器模式。拦截器上下文AspectContext可以获取当前Scoped的ServiceProvider

    public class CustomInterceptorAttribute : AbstractInterceptorAttribute
    {
    public override Task Invoke(AspectContext context, AspectDelegate next)
    {
    var logger = context.ServiceProvider.GetService<ILogger<CustomInterceptorAttribute>>();
    logger.LogInformation("call interceptor");
    return next(context);
    }
    }

有问题反馈

如果您有任何问题,请提交 Issue 给我们。

AspectCore Project 项目地址: https://github.com/dotnetcore/AspectCore-Framework

AspectCore Group QQ群: 306531723

Asp.Net Core轻量级Aop解决方案:AspectCore的更多相关文章

  1. [Asp.Net Core轻量级Aop解决方案]AspectCore Project 介绍

    AspectCore Project 介绍 什么是AspectCore Project ? AspectCore Project 是适用于Asp.Net Core 平台的轻量级 Aop(Aspect- ...

  2. asp.net core的AOP记录

    序曲:学习编程最好的方式就是敲代码,没有比这个更好的方法,哪怕你看了上百G的视频,都不如你自己敲几行代码更为有效.还有要记得敲完代码然后写一篇随笔来记录一下你所学所想. 什么叫AOP? AOP面向切面 ...

  3. 打造静态分析器(二)基于Asp.Net Core 3.0的AspectCore组件检测

    上一篇,我们打造了一个简单的分析器,但是我们实际使用分析器就是为了对项目做分析检测,增加一些非语法的自检的 比如Asp.Net Core 3.0的替换依赖注入检测 设计分析 我们创建一个默认的Asp. ...

  4. 一个遵循CleanArchitecture原则的Asp.net core轻量级开源项目

    这是一个基于最新的ASP.net core 5.0创建Razor Page应用程序解决方案模板.遵循Clean Architecture的原则,以最求简洁的代码风格和实现快速开发小型的web业务系统的 ...

  5. Asp.net Core 3.1基于AspectCore实现AOP,实现事务、缓存拦截器

    最近想给我的框架加一种功能,就是比如给一个方法加一个事务的特性Attribute,那这个方法就会启用事务处理.给一个方法加一个缓存特性,那这个方法就会进行缓存. 这个也是网上说的面向切面编程AOP. ...

  6. .Net Core 学习 (1) - ASP.NET Core 总览

    什么是ASP.NET 1.0 开源 - GitHub 跨平台 - 支持Windows, Mac, Linux 从底层进行了优化 - 使用最小开销的模块化组件 - 给与了开发人员很大的灵活性 为什么要使 ...

  7. 适合 ASP.NET Core 的超级-DRY开发

    作者 Thomas Hansen DRY 是那些非常重要的软件体系结构缩写之一.它的意思是“不要自我重复”,并向维护旧源代码项目的任何用户阐明了一个重要原则.也就是说,如果你在代码中自我重复,会发现每 ...

  8. ASP.NET Core应用程序容器化、持续集成与Kubernetes集群部署(一)(转载)

    本文结构 ASP.NET Core应用程序的构建 ASP.NET Core应用程序容器化所需注意的问题 应用程序的配置信息 端口侦听 ASP.NET Core的容器版本 docker镜像构建上下文(B ...

  9. Workflow Core + asp.net core 5.0 实现简单审批工作流

    我们知道企业业务系统到处都可以审批工作流的,但也很少有像OA系统一样复杂多级多条件的审批工作流需要设计,所以我们需要一个轻量级的容易上手的workflow框架,通过GitHub,我发现danielge ...

随机推荐

  1. 【one day one linux】grep工具

    grep  筛选功能 学习这些命令采用20/80原则,这样,可以达到使用%20的命令选项,处理80%的情况. #grep 的使用格式 grep [option] pattern file 那么接下来看 ...

  2. GitHub上非常受开发者欢迎的iOS开源项目(二)

    "每一次的改变总意味着新的开始."这句话用在iOS上可谓是再合适不过的了.GitHub上的iOS开源项目数不胜数,iOS每一次的改变,总会引发iOS开源项目的演变,从iOS 1.x ...

  3. 初遇stm32

    刚开始接触32,建一个工程都这么费劲,可能是keil安装时一些文件和库没有安装完整,真是坑啊. 回头可能还要从新安装,然后开始新的学习,争取十天之内入门32,在博客园这个强大的技术支持下, 想不入门都 ...

  4. 实现TOLock过程中的一处多线程bug

    背景 最近在啃<多处理器编程的艺术>,书中的7.6节介绍了时限锁--实现了tryLock方法的队列锁. 书中重点讲解了tryLock的实现,也就是如何实现在等待超时后退出队列,放弃锁请求, ...

  5. Elasticserach学习笔记-01基础概念

    本文系本人根据官方文档的翻译,能力有限.水平一般,如果对想学习Elasticsearch的朋友有帮助,将是本人的莫大荣幸. 原文出处:https://www.elastic.co/guide/en/e ...

  6. # Android动画笔记

    标签: Android开发艺术探索笔记 View动画 帧动画 属性动画 View动画 View动画的作用对象时View,有4种动画效果,分别是平移动画.缩放动画.旋转动画.和透明度动画. 此类动画通常 ...

  7. 内存管理之slab分配器

    基本思想 与传统的内存管理模式相比, slab 缓存分配器提供了很多优点.首先,内核通常依赖于对小对象的分配,它们会在系统生命周期内进行无数次分配.slab 缓存分配器通过对类似大小的对象进行缓存而提 ...

  8. 详解Java反射机制

    反射是程序在运行状态下,动态的获取某个类的内部信息的一种操作.例如:类名,包名,所有属性的集合,所有方法的集合,构造方法的集合等.该操作发生在程序的运行时状态,所以编译器管不着有关反射的一些代码,通常 ...

  9. SQL SERVER大话存储结构(1)_数据页类型及页面指令分析

                如果转载,请注明博文来源: www.cnblogs.com/xinysu/   ,版权归 博客园 苏家小萝卜 所有.望各位支持!          SQLServer的数据页大 ...

  10. TypeScript入门-高级类型

    高级类型 交叉类型 交叉类型,就是将多个类型合并为一个新的类型,这个新的类型具有这多个类型的成员,含有这几个类型的所有特性,是他们的综合体,像是集合的并集 例子: function extend< ...