场景
刚写完一个干净利落的方法,比如保存数据到数据库,逻辑清晰、结构优雅,
第二天,“嘿,保存完数据,记得给客户发个邮件哦~”
第三天,“能不能再发个消息通知其他系统?”
第四天,“能不能记录一下操作日志?”
第五天,“再加个短信提醒吧。”
……
就这样,原本清清爽爽的 SaveData 方法,变成了一个臃肿不堪的函数:
我们管这种代码叫 “脚本代码”或“面条代码” —— 逻辑缠在一起,改一处,处处提心吊胆。

C# 提供了更灵活的方式来处理这种场景,那就是利用 Attribute 来对业务进行解耦,从而避免这种脚本式的代码,提高代码的可扩展性

1. 定义特性

namespace WebApplication2.Attributes
{/// <summary>/// https://mp.weixin.qq.com/s/Sd9q7FOTlk29wBknNQh87w/// 后置操作特性基类/// 所有继承它的特性都可以用在方法上,允许多个,不继承到子类/// </summary>[AttributeUsage(AttributeTargets.Method, Inherited = false, AllowMultiple = true)]public abstract class PostOperationAttribute : Attribute{// 每个后置操作都必须实现 Execute 方法public abstract void Execute(object returnValue);}/// <summary>/// 发送邮件特性/// </summary>public class SendEmailAttribute : PostOperationAttribute{private readonly string _emailTemplate; // 邮件模板名称// 构造函数接收模板名public SendEmailAttribute(string emailTemplate){_emailTemplate = emailTemplate;}// 实现具体的发送邮件逻辑public override void Execute(object returnValue){// 实际项目中这里应该调用邮件服务Console.WriteLine($"发送邮件 - 使用模板: {_emailTemplate}");Console.WriteLine($"邮件内容包含数据: {returnValue}");}}/// <summary>/// 发送消息特性/// </summary>public class SendMessageAttribute : PostOperationAttribute{private readonly string _messageType; // 消息类型public SendMessageAttribute(string messageType){_messageType = messageType;}public override void Execute(object returnValue){Console.WriteLine($"发送 {_messageType} 消息");Console.WriteLine($"消息内容包含数据: {returnValue}");}}
}

2. 编写业务方法

using System.Reflection;namespace WebApplication2.Attributes
{public class DataService{/// <summary>/// 核心逻辑只负责保存数据/// 使用特性标记需要后置处理的方法/// </summary>/// <param name="data"></param>/// <returns></returns>[SendEmail("DataSavedTemplate")][SendMessage("Notification")]public virtual int SaveData(string data){// 这里只关注保存数据的核心业务逻辑Console.WriteLine($"保存数据: {data}");// 模拟返回保存后的IDreturn new Random().Next(1000);}}
}

3. 创建拦截类(代理)

using System.Reflection;namespace WebApplication2.Attributes
{/// <summary>/// 自动处理 Attribute 的代理类/// </summary>public class DataServiceProxy : DataService{public override int SaveData(string data){// 调用基类方法var result = base.SaveData(data);// 获取方法信息MethodInfo methodInfo = typeof(DataService).GetMethod("SaveData");// 获取该方法上所有的 PostOperationAttribute 特性实例var postOps = methodInfo.GetCustomAttributes<PostOperationAttribute>(true);// 遍历并执行每一个后置操作foreach (var op in postOps){op.Execute(result);}return result;}}
}

4. 使用

using Microsoft.AspNetCore.Mvc;
using WebApplication2.Attributes;namespace WebApplication2.Controllers
{[Route("api/Attributes/[action]")][ApiController]public class AttributesController : ControllerBase{[HttpGet]public string Test(){// 使用代理类而不是直接使用DataServicevar dataService = new DataServiceProxy();// 直接调用方法,后置操作会自动执行int savedId = dataService.SaveData("测试数据");Console.WriteLine($"保存成功,ID: {savedId}");return "";}}
}

5. 运行和测试

6.总结
● 核心业务方法不再被新增加的业务需求污染
● 扩展功能就像搭积木一样快捷方便
● 新增功能无需修改原有代码,维护成本大大降低
● 一眼就能看出某个方法执行后会触发哪些操作,代码可读性更强

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/93925.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/93925.shtml
英文地址,请注明出处:http://en.pswp.cn/bicheng/93925.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【URP】[法线贴图]为什么主要是蓝色的?

【从UnityURP开始探索游戏渲染】专栏-直达 法线贴图呈现蓝紫色调&#xff08;尤其以蓝色为主&#xff09;是由其‌存储原理、切线空间坐标系设计及颜色编码规则共同决定的‌。 核心原因&#xff1a;法线向量的存储规则‌ ‌法线向量的物理范围‌ 法线是单位向量&#xff0c;…

驱动开发系列63 - NVIDIA 开源GPU驱动open-gpu-kernel-modules编译调试

目录 一:通过apt方式安装nvidia 驱动 二:通过 .run 方式安装nvidia驱动 三:编译安装nvidia开源内核驱动 四:验证和调试 五:卸载驱动 1. 以apt方式安装nvidia 驱动的卸载方法 2. 以.run方式安装nvidia驱动的卸载方法 六:安装CUDA环境 一:通过apt方式安装nvidia 驱动…

对KingbaseES架构的解析:从读写分离到异地灾备的技术实现与保障机制

声明&#xff1a;文章为本人真实测评博客&#xff0c;非广告&#xff0c;并没有推广该平台 &#xff0c;为用户体验文章 本人旨在分享最真实的用户体验&#xff0c;为关注此类产品的朋友们提供一个客观的参考。 文章目录一、架构全景&#xff1a;四级高可用构建数字基础1.1 物…

Visual Studio中的常用调试功能(上)

1、利用断点进行调试添加断点的方式有以下几种1.键盘快捷键F92.通过菜单【Debug&#xff08;调试&#xff09;】-》【Toggle BreakPoint&#xff08;切换断点&#xff09;】3.点击代码行左边的空白处&#xff08;推荐&#xff09;设置断点后&#xff0c;按F5运行程序&#xff0…

Linux -- 线程同步

1.1条件变量 (1)当⼀个线程互斥地访问某个变量时&#xff0c;它可能发现在其它线程改变状态之前&#xff0c;它什么也做不了。 (2)例如⼀个线程访问队列时&#xff0c;发现队列为空&#xff0c;它只能等待&#xff0c;只到其它线程将⼀个节点添加到队列 中。这种情况就需要⽤到…

前端进阶指南:JavaScript性能优化实战全解析

深入剖析 JavaScript 性能瓶颈&#xff0c;分享优化技巧与最佳实践&#xff0c;让你的前端应用更快、更稳、更流畅。 &#x1f4d1; 目录 一、前言 二、性能瓶颈的常见来源 三、JavaScript代码优化技巧 1. 避免重复计算 2. 合理使用防抖与节流 3. 使用事件委托 四、渲染…

RabbitMQ:SpringAMQP Direct Exchange(直连型交换机)

目录一、案例需求二、基础配置三、代码实现直连型交换机也叫做定向交换机&#xff0c;通过RoutingKey绑定交换机与队列直接的关系。 生产者源码 消费者源码 一、案例需求 在RabbitMQ控制台中&#xff0c;声明队列direct.queue1和direct.queue2。在RabbitMQ控制台中&#xff…

implement libtime on Windows

因为Windows的time命令和Linux的time命令不一样&#xff0c;尝试实现libtime libtime.h /** libtime.h - 跨平台时间测量库* 功能&#xff1a;执行外部命令并测量其运行时间和资源使用*/#ifndef LIBTIME_H #define LIBTIME_H#include <stdio.h> #include <stdlib.h>…

Unity进阶--C#补充知识点--【C#各版本的新功能新语法】C#1~4与C#5

来源于唐老狮的视频教学&#xff0c;仅作记录和感悟记录&#xff0c;方便日后复习或者查找 一.C#版本与Unity的关系 1.各Unity版本支持的C#版本 更多信息可以在Untiy官网说明查看 https://docs.unity3d.com/2020.3/Documentation/Manual/CSharpCompiler.html&#xff08;这个好…

水闸安全综合监测系统解决方案

一、方案概述 水闸作为重要的水利工程设施&#xff0c;承担着防洪、排涝、供水和灌溉等关键功能。其安全性直接关系到下游人民群众的生命财产安全以及区域经济的稳定发展。近年来&#xff0c;随着极端天气频发和工程老化问题日益突出&#xff0c;水闸安全监测工作显得尤为重要。…

基于单片机智能点滴输液系统

传送门 &#x1f449;&#x1f449;&#x1f449;&#x1f449;其他作品题目速选一览表 &#x1f449;&#x1f449;&#x1f449;&#x1f449;其他作品题目功能速览 概述 该系统基于单片机控制技术&#xff0c;结合传感器和无线通信模块&#xff0c;实现对输液过程的实…

AI数据仓库管理提升效率

内容概要在数字化转型浪潮中&#xff0c;AI数据仓库管理正重塑企业数据处理格局。本部分简要介绍其核心机制&#xff0c;即通过智能API接入外部数据源实现多平台数据无缝整合&#xff0c;随后应用数据清洗技术去除冗余信息&#xff0c;确保数据质量。同时&#xff0c;加密存储机…

使用 Docker 安装长安链管理平台 + 部署区块链与示例合约

文章目录简介登录官网GithubPodman 配置&#xff08;Docker 配置 registry 地址&#xff09;安装长安链管理平台下载源码docker-compose.yml登录管理平台部署区块链Dockerfile构建镜像部署长安链订阅区块链部署合约下载示例合约部署示例合约投票管理文件哈希存证查找存证信息区…

Python训练营打卡 DAY 41 简单CNN

知识回顾 数据增强卷积神经网络定义的写法batch归一化&#xff1a;调整一个批次的分布&#xff0c;常用与图像数据特征图&#xff1a;只有卷积操作输出的才叫特征图调度器&#xff1a;直接修改基础学习率 卷积操作常见流程如下&#xff1a; 1. 输入 → 卷积层 → Batch归一化层…

云端赋能,智慧运维:分布式光伏电站一体化监控平台研究

摘要 本文针对分布式光伏电站存在的监管困难、火灾隐患、系统繁杂及运维不规范等行业痛点&#xff0c;提出AcrelCloud-1200光伏运维云平台解决方案。平台通过ANet-1E2S-4G网关集成多品牌逆变器数据&#xff0c;结合视频监控与气象站&#xff0c;实现电站全域监测&#xff1b;开…

CVPR 2025 | 具身智能 | HOLODECK:一句话召唤3D世界,智能体的“元宇宙练功房”来了

关注gongzhonghao【CVPR顶会精选】1.导读1.1 论文基本信息论文标题&#xff1a;《HOLODECK: Language Guided Generation of 3D Embodied AI Environments》作者&#xff1a;Yue Yang*1, Fan-Yun Sun*2, Luca Weihs*4, Eli Vanderbilt4, Alvaro Herrasti4,Winson Han4, Jiajun …

迅为RK3568开发板搭建Ubuntu环境

本小节介绍开发所需 Ubuntu 环境的搭建方法。系统要求:Ubuntu 系统要求&#xff1a;Ubuntu18.04~21.10 版本。推荐使用 20.04 版本&#xff0c;内存 16GB 及以上&#xff0c;硬盘 100GB 及以上。Ubuntu 系统的用户名不能包含中文字符。建议 Ubuntu 和 Windows 系统上安装的 Dev…

【数据结构】用堆解决TOPK问题

设计一个算法&#xff0c;找出数组中最小的k个数。以任意顺序返回这k个数均可。示例&#xff1a;输入&#xff1a; arr [1,3,5,7,2,4,6,8], k 4 输出&#xff1a; [1,2,3,4]比较替换堆顶的数时&#xff0c;不需要让堆顶与数组的每一个数再进行比较&#xff0c;比较数组减去k个…

【深度长文】Anthropic发布Prompt Engineering全新指南

目录 1.什么时候适合用提示工程? 2.如何进行提示工程 2.1 使用提示模板 2.1.1 使用提示模板和变量 2.1.2 何时使用提示模板和变量 2.1.3 提示模板示例 2.2 保持清晰和直接 2.2.1 如何保持清晰、具有上下文和具体 2.2.2 示例 ​2.3 使用示例&#xff08;多示例提示…

【基础-判断】HarmonyOS提供了基础的应用加固安全能力,包括混淆、加密和代码签名能力

正确 解释如下: 应用加固: 这是指对应用程序进行保护,使其更难被逆向工程、篡改或盗版。HarmonyOS 作为现代操作系统,确实提供了这样的基础安全能力。 混淆: HarmonyOS 的 SDK 提供了代码混淆工具(通常基于 ProGuard 或类似技术)。开发者在构建应用时启用混淆,可以将类…