Java 中装饰者模式与策略模式在埋点系统中的应用

前言

在软件开发中,装饰者模式和策略模式是两种常用的设计模式,它们在特定的业务场景下能够发挥巨大的作用。本文将通过一个实际的埋点系统案例,探讨如何在 Java 中运用装饰者模式和策略模式,以及如何结合工厂方法模式来优化代码结构。

业务场景分析

随着互联网的发展,用户行为分析变得越来越重要,而埋点技术是实现用户行为分析的关键手段之一。埋点系统需要记录用户在应用中的各种操作行为,如点击、浏览、提交等,以便后续进行数据分析和业务决策。

假设我们正在开发一个在线教育平台,需要实现以下埋点功能:

  1. 点击埋点:记录用户点击的位置。

  2. 课程埋点:记录用户点击的课程信息。

  3. 任务埋点:记录用户点击的任务信息。

这些埋点功能需要根据不同的业务场景进行动态组合,例如在课程页面的点击操作需要记录点击位置和课程信息,而在任务页面的点击操作需要记录点击位置和任务信息。

装饰者模式的应用

装饰者模式允许我们在不修改原有代码的基础上,动态地给对象添加职责。它由以下几部分组成:

  • Component:定义对象的接口。

  • Concrete Component:实现 Component 接口的具体对象。

  • Decorator:维护一个对 Component 对象的引用,并定义与 Component 接口相同的接口。

  • Concrete Decorator:实现 Decorator 接口,负责给 Component 对象添加特定的职责。

实现埋点功能

// Component 接口
public interface SaveMessage {void saveMessage(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest);
}// Concrete Component:基础点击埋点
public class CommonClickPoint implements SaveMessage {@Overridepublic void saveMessage(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {pointSaveBean.setClickLocation(pointSaveRequest.getClickLocation());}
}// Decorator 抽象类
public abstract class AddPointMessageService implements SaveMessage {protected SaveMessage saveMessage;public AddPointMessageService(SaveMessage saveMessage) {this.saveMessage = saveMessage;}@Overridepublic void saveMessage(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {saveMessage.saveMessage(pointSaveBean, pointSaveRequest);}
}// Concrete Decorator:课程埋点
public class CourseClickPoint extends AddPointMessageService {public CourseClickPoint(SaveMessage saveMessage) {super(saveMessage);}@Overridepublic void saveMessage(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {super.saveMessage(pointSaveBean, pointSaveRequest);pointSaveBean.setCourseId(pointSaveRequest.getCourseId());}
}// Concrete Decorator:任务埋点
public class TaskClickPoint extends AddPointMessageService {public TaskClickPoint(SaveMessage saveMessage) {super(saveMessage);}@Overridepublic void saveMessage(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {super.saveMessage(pointSaveBean, pointSaveRequest);pointSaveBean.setTaskId(pointSaveRequest.getTaskId());}
}

客户端代码

public class CommonMain {public static void main(String[] args) {// 初始化埋点类型列表List<PointSaveType> types = Arrays.asList(PointSaveType.TASK, PointSaveType.COURSE);// 初始化埋点保存对象PointSaveBean pointSaveBean = new PointSaveBean();// 初始化埋点请求对象PointSaveRequest pointSaveRequest = PointSaveRequest.builder().pointSaveTypeList(types).clickLocation("右上角落").courseId("英语").taskId("任务1").build();// 初始化基础保存逻辑SaveMessage saveMessage = new CommonClickPoint();saveMessage.saveMessage(pointSaveBean, pointSaveRequest);System.out.println("基础埋点保存数据: " + pointSaveBean);// 根据埋点类型动态添加保存逻辑for (PointSaveType type : types) {if (type == PointSaveType.COURSE) {saveMessage = new CourseClickPoint(saveMessage);} else if (type == PointSaveType.TASK) {saveMessage = new TaskClickPoint(saveMessage);}saveMessage.saveMessage(pointSaveBean, pointSaveRequest);}// 打印最终埋点数据System.out.println("最终埋点保存数据: " + pointSaveBean);}
}

策略模式的应用

策略模式定义了一系列算法,并将每个算法封装到具有共同接口的独立类中,使它们可以互相替换。当埋点逻辑之间存在复杂的组合关系时,结合策略模式可以更好地管理这些组合逻辑。

实现埋点功能

// 策略接口
public interface PointSaveStrategy {void save(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest);
}// 具体策略:点击埋点
public class ClickPointSaveStrategy implements PointSaveStrategy {@Overridepublic void save(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {pointSaveBean.setClickLocation(pointSaveRequest.getClickLocation());}
}// 具体策略:课程埋点
public class CoursePointSaveStrategy implements PointSaveStrategy {@Overridepublic void save(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {pointSaveBean.setCourseId(pointSaveRequest.getCourseId());}
}// 具体策略:任务埋点
public class TaskPointSaveStrategy implements PointSaveStrategy {@Overridepublic void save(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {pointSaveBean.setTaskId(pointSaveRequest.getTaskId());}
}// 策略上下文
public class PointSaveStrategyContext {private List<PointSaveStrategy> strategies = new ArrayList<>();public void addStrategy(PointSaveStrategy strategy) {strategies.add(strategy);}public void execute(PointSaveBean pointSaveBean, PointSaveRequest pointSaveRequest) {for (PointSaveStrategy strategy : strategies) {strategy.save(pointSaveBean, pointSaveRequest);}}
}// 策略配置
public class PointSaveStrategyConfig {private static final Map<PointSaveType, PointSaveStrategy> STRATEGY_MAP = new HashMap<>();static {STRATEGY_MAP.put(PointSaveType.CLICK, new ClickPointSaveStrategy());STRATEGY_MAP.put(PointSaveType.COURSE, new CoursePointSaveStrategy());STRATEGY_MAP.put(PointSaveType.TASK, new TaskPointSaveStrategy());}public static PointSaveStrategy getStrategy(PointSaveType type) {return STRATEGY_MAP.getOrDefault(type, null);}
}

客户端代码

public class CommonMain {public static void main(String[] args) {// 初始化埋点类型列表List<PointSaveType> types = Arrays.asList(PointSaveType.TASK, PointSaveType.COURSE);// 初始化埋点保存对象PointSaveBean pointSaveBean = new PointSaveBean();// 初始化埋点请求对象PointSaveRequest pointSaveRequest = PointSaveRequest.builder().pointSaveTypeList(types).clickLocation("右上角落").courseId("英语").taskId("任务1").build();// 创建策略上下文PointSaveStrategyContext context = new PointSaveStrategyContext();// 添加基础策略context.addStrategy(new ClickPointSaveStrategy());// 根据埋点类型动态添加策略for (PointSaveType type : types) {PointSaveStrategy strategy = PointSaveStrategyConfig.getStrategy(type);if (strategy != null) {context.addStrategy(strategy);}}// 执行所有策略context.execute(pointSaveBean, pointSaveRequest);// 打印最终埋点数据System.out.println("最终埋点保存数据: " + pointSaveBean);}
}

工厂方法模式的结合

为了进一步简化客户端代码,我们可以引入工厂方法模式来创建装饰者对象。

// 工厂类
public class PointSaveDecoratorFactory {public static SaveMessage getDecorator(PointSaveType type, SaveMessage saveMessage) {switch (type) {case COURSE:return new CourseClickPoint(saveMessage);case TASK:return new TaskClickPoint(saveMessage);default:System.out.println("未知的埋点类型: " + type);return saveMessage;}}
}

客户端代码优化

public class CommonMain {public static void main(String[] args) {// 初始化埋点类型列表List<PointSaveType> types = Arrays.asList(PointSaveType.TASK, PointSaveType.COURSE);// 初始化埋点保存对象PointSaveBean pointSaveBean = new PointSaveBean();// 初始化埋点请求对象PointSaveRequest pointSaveRequest = PointSaveRequest.builder().pointSaveTypeList(types).clickLocation("右上角落").courseId("英语").taskId("任务1").build();// 初始化基础保存逻辑SaveMessage saveMessage = new CommonClickPoint();saveMessage.saveMessage(pointSaveBean, pointSaveRequest);System.out.println("基础埋点保存数据: " + pointSaveBean);// 根据埋点类型动态添加保存逻辑for (PointSaveType type : types) {saveMessage = PointSaveDecoratorFactory.getDecorator(type, saveMessage);saveMessage.saveMessage(pointSaveBean, pointSaveRequest);}// 打印最终埋点数据System.out.println("最终埋点保存数据: " + pointSaveBean);}
}

总结

在实际的开发过程中,合理地运用设计模式能够使我们的代码更加灵活、可维护和可扩展。装饰者模式适合用于在运行时动态地给对象添加职责,而策略模式则适合用于管理多种算法或行为的组合。通过结合工厂方法模式,我们可以进一步简化客户端代码,使系统更加模块化和易于使用。

通过本文的示例,我们看到了装饰者模式和策略模式在埋点系统中的有效应用。这些设计模式不仅解决了实际的业务问题,还为我们提供了应对复杂需求变化的优雅解决方案。在未来的开发中,我们可以根据具体的需求场景,灵活地选择和结合不同的设计模式,以构建高质量的软件系统。

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

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

相关文章

【3-22 list 详解STL C++ 】

先看代码&#xff0c;常用的就是代码中有的那些 #include <bits/stdc.h> using namespace std; int main() {list<int> mylist;for(int i0;i<5;i){mylist.push_back(i);//TODO}for(const auto&i:mylist)cout<<i<<\n;//fanzhuanreverse(mylist.…

田间机器人幼苗视觉检测与护苗施肥装置研究(大纲)

田间机器人幼苗视觉检测与护苗施肥装置研究 基于多光谱视觉与精准施肥的农业机器人系统设计 第一章 绪论 1.1 研究背景与意义 农业智能化需求&#xff1a; 传统幼苗检测依赖人工&#xff0c;效率低且易遗漏弱苗/病苗施肥不精准导致资源浪费和环境污染 技术挑战&#xff1a;…

如何在Linux CentOS上安装和配置Redis

如何在Linux CentOS上安装和配置Redis 大家好&#xff0c;我是曾续缘。欢迎来到本教程&#xff01;今天我将向您介绍在Linux CentOS上安装和配置Redis的详细步骤。Redis是一个高性能的键值存储系统&#xff0c;常用于缓存、消息队列和数据持久化等应用场景。让我们一起开始吧&…

requests库post方法怎么传params类型的参数

在使用 requests 库的 post 方法时&#xff0c;params 类型的参数通常用于在 URL 中作为查询字符串传递。这与 data 或 json 参数不同&#xff0c;后者是放在请求体中的。下面详细介绍如何在使用 post 方法时传递 params 参数。 使用 params 参数 params 参数接受一个字典或包…

C++常见问题与思考

TLS&#xff08;线程本地存储&#xff09;原理 线程本地存储&#xff08;Thread Local Storage&#xff0c;TLS&#xff09;是一种机制&#xff0c;它允许每个线程拥有自己独立的变量实例&#xff0c;这些变量的生命周期与线程相同。也就是说&#xff0c;不同线程对同一个 TLS…

如何快速下载并安装 Postman?

从下载、安装、启动 Postman 这三个方面为大家详细讲解下载安装 Postman 每一步操作&#xff0c;帮助初学者快速上手。 Postman 下载及安装教程(2025最新)

使用Gitee Go流水线部署个人项目到服务器指南

使用Gitee Go流水线部署个人项目到服务器指南 前言&#xff01;&#xff01;&#xff01; 本文解决的问题&#xff1a; 你有一台ECS服务器&#xff0c;你在上面部署了一个Java服务也就是一个jar&#xff0c;你觉着你每次手动本地打包&#xff0c;上传&#xff0c;在通过命令去…

Linux第一节:Linux系统编程入门指南

摘要 本文面向Linux初学者&#xff0c;系统讲解操作系统核心概念、Shell命令实战、权限管理精髓及目录结构解析。通过思维导图命令示例原理解析的方法&#xff0c;帮助开发者快速构建Linux知识体系&#xff0c;掌握生产环境必备技能。 一、Linux的前世今生&#xff1a;从实验室…

【Linux 维测专栏 5 -- linux pstore 使用介绍】

文章目录 Linux pstore 功能简介1. pstore 概述2. pstore 的核心功能3. pstore 的工作原理4. pstore 的使用示例5. pstore 的优势6. 典型应用场景配置示例1)DTS配置2)config配置运行测试及log问题小结Linux pstore 功能简介 1. pstore 概述 pstore(Persistent Storage)是…

在 ASP .NET Core 9.0 中使用 Scalar 创建漂亮的 API 文档

示例代码&#xff1a;https://download.csdn.net/download/hefeng_aspnet/90407900 Scalar 是一款可帮助我们为 API 创建精美文档的工具。与感觉有些过时的默认 Swagger 文档不同&#xff0c;Scalar 为 API 文档提供了全新而现代的 UI。其简洁的设计让开发人员可以轻松找到测试…

Rabbitmq消息被消费时抛异常,进入Unacked 状态,进而导致消费者不断尝试消费(下)

一、消费流程图 消息在消费出现异常的时候&#xff0c;将一直保留在消息队列&#xff0c;所以你会看到以下奇怪的现象&#xff1a; 消息队列仅有5个消息&#xff0c; 投递速度也非常快&#xff0c;结果却一直无法消费掉。 二、重试策略 重试机制的使用场景&#xff1a;重试机制…

【STM32】知识点介绍二:GPIO引脚介绍

文章目录 一、概述二、GPIO的工作模式三、寄存器编程 一、概述 GPIO&#xff08;英语&#xff1a;General-purpose input/output&#xff09;,即通用I/O(输入/输出)端口&#xff0c;是STM32可控制的引脚。STM32芯片的GPIO引脚与外部设备连接起来&#xff0c;可实现与外部通讯、…

JavaScript流程控制精讲(二)运算符与循环实战

JavaScript流程控制精讲&#xff08;二&#xff09;运算符与循环实战 学习目标&#xff1a;掌握条件判断与循环控制&#xff0c;实现基础业务逻辑 核心要点&#xff1a;运算符优先级 | 短路运算 | 循环优化 | 项目实战 一、运算符进阶技巧 1.1 算术运算符 console.log(5 % 3)…

如何在IPhone 16Pro上运行python文件?

在 iPhone 16 Pro 上运行 Python 文件需要借助第三方工具或远程服务&#xff0c;以下是具体实现方法和步骤&#xff1a; 一、本地运行方案&#xff08;无需越狱&#xff09; 使用 Python 编程类 App 以下应用可在 App Store 下载&#xff0c;支持直接在 iPhone 上编写并运行 …

【赵渝强老师】达梦数据库的数据库对象

达梦数据库中包含各种数据库对象&#xff0c;主要分为两大类型&#xff1a;基本数据库对象和复杂数据库对象。下面分别进行介绍。 视频讲解如下 【赵渝强老师】达梦数据库的数据库对象 一、 基本数据库对象 常见的基本数据库对象有&#xff1a;表、索引、视图、序列、同义词等…

【每日算法】Day 6-1:哈希表从入门到实战——高频算法题(C++实现)

摘要 &#xff1a;掌握高频数据结构&#xff01;今日深入解析哈希表的核心原理与设计实现&#xff0c;结合冲突解决策略与大厂高频真题&#xff0c;彻底掌握O(1)时间复杂度的数据访问技术。 一、哈希表核心思想 哈希表&#xff08;Hash Table&#xff09; 是一种基于键值对的…

LeetCode 第29题、30题

LeetCode 第29题&#xff1a;两数相除 题目描述 给你两个整数&#xff0c;被除数dividend和除数divisor。将两数相除&#xff0c;要求不使用乘法、除法和取余运算。整数除法应该向零截断&#xff0c;也就是截去其小数部分。例如&#xff0c;8.345将被截断为8&#xff0c;-2.733…

26考研——树与二叉树_树、森林(5)

408答疑 文章目录 二、树、森林树的基本概念树的定义和特性树的定义树的特性 基本术语树的基本术语和概念祖先、子孙、双亲、孩子、兄弟和堂兄弟结点的层次、度、深度和高度树的度和高度分支结点和叶结点有序树和无序树路径和路径长度 森林的基本术语和概念森林的定义森林与树的…

【HarmonyOS Next之旅】DevEco Studio使用指南(六)

目录 1 -> 在模块中添加Ability 1.1 -> Stage模型添加UIAbility 1.1.1 -> 在模块中添加UIAbility 1.1.2 -> 在模块中添加Extension Ability 2 -> 创建服务卡片 2.1 -> 概述 2.2 -> 使用约束 2.3 -> 创建服务卡片 2.4 -> 创建动态/静态卡片…

Langchain 多模态输入和格式化输出

多模态输入 图片处理&#xff08;最高频&#xff09; 1.1 URL形式&#xff08;推荐大文件&#xff09; from langchain.schema import HumanMessage from langchain.chat_models import ChatOpenAIchat ChatOpenAI(model"gpt-4-vision-preview")message HumanMes…