拦截器Interceptors是一种可以在编译时以声明方式替换原有应用的方法。
这种替换是通过让Interceptors声明它拦截的调用的源位置来实现的。
您可以使用拦截器作为源生成器的一部分进行修改,而不是向现有源编译添加代码。
演示
使用 .NET 8 创建一个控制台应用程序。并在PropertyGroup中添加以下配置.。需要将其中WebApplication6替换为自己的命名空间。
<InterceptorsPreviewNamespaces>$(InterceptorsPreviewNamespaces);WebApplication6</InterceptorsPreviewNamespaces>
然后在单独的文件中创建InterceptsLocationAttribute。其命名空间必须是System.Runtime.CompilerServices,而不是应用程序的命名空间。
namespace System.Runtime.CompilerServices
{
[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public sealed class InterceptsLocationAttribute(string filePath, int line, int character) : Attribute
{
}
}
该属性包含三个参数。
- filePath是您要拦截的文件的路径。
- line是您要拦截的代码行。
- character是您要拦截的代码字符位置。
接着来创建一个具有三种方法的类,模拟新增/查询用户作为示例:
public class GetUserService
{
// This method will not be intercepted;
public void GetUserName()
{
Console.WriteLine("GetUserName");
}
// This method will be intercepted;
public void AddUser()
{
Console.WriteLine("AddUser");
}
// This method will not be intercepted;
public void DeleteUser()
{
Console.WriteLine("DeleteUser");
}
}
在 Program.cs 文件中,我创建了此类的一个实例,并创建了对这三个方法中每一个的调用。输出如下所示:
var userService = new GetUserService();
userService.GetUserName();
userService.AddUser();
userService.DeleteUser();
现在让我们创建拦截类。该类必须遵循以下规则:
- 一定是一个static类。
- 必须是我们要拦截的类的扩展方法。
- 必须具有该InterceptsLocation属性,其中包含我们要拦截的文件路径的值以及行号和字符号。
using System.Runtime.CompilerServices;
namespace WebApplication6
{
public static class InterceptUserService
{
[InterceptsLocation(
filePath: @"D:\demo\test\ConsoleApp1\WebApplication6\Program.cs",
line: 14,
character: 25)]
public static void InterceptMethodAddUser(this GetUserService example)
{
Console.WriteLine("Interceptor is here!");
}
}
}
在此示例中,将拦截AddUser方法,并且将执行InterceptMethodAddUser方法,而不是执行方法AddUser。
filePath可以按以下方式获取
行号和字符号可以按以下方式获取
现在运行代码,方法AddUser将被拦截,并且不会被执行,而是实际执行拦截器方法,以下是输出: