【设计模式】深入理解Python中的过滤器模式(Filter Pattern)
深入理解Python中的过滤器模式(Filter Pattern)
在软件设计中,面对复杂的数据处理需求时,我们常常需要从一组数据中筛选出符合特定条件的子集。**过滤器模式(Filter Pattern)**是一种能够简化这种操作的设计模式。它通过将过滤逻辑与对象的业务逻辑分离,使得数据过滤过程更加灵活、可扩展。
本文将详细探讨过滤器模式的概念、应用场景、实现步骤,并通过代码示例演示如何在Python中实现过滤器模式。
1. 什么是过滤器模式?
过滤器模式是一种行为型设计模式,允许用户通过一组标准对对象集合进行筛选。它使用多个标准进行过滤,并提供了一个组合的方式来对数据进行选择。
过滤器模式的核心要点
- 目标对象:需要过滤的对象集合。
- 过滤器接口:定义过滤操作的接口,通常有一个或多个方法,用于执行具体的过滤逻辑。
- 具体过滤器:实现过滤器接口的具体过滤器,包含过滤条件。
- 上下文(Context):用于管理目标对象和过滤器的类,负责应用过滤器以获取过滤结果。
UML 类图表示
+---------------------+
| Client |
+---------------------+||V
+---------------------+
| Context |
+---------------------+
| +add_filter() |
| +filter() |
+---------------------+||V
+---------------------+
| Filter Interface |
+---------------------+
| +filter() |
+---------------------+||
+---------------------+
| Concrete Filter |
+---------------------+
| +filter() |
+---------------------+
2. 过滤器模式的应用场景
过滤器模式适用于以下场景:
- 需要动态筛选对象集合:当对象集合中有多种属性需要筛选时,过滤器模式提供了一种灵活的方式来组合多个过滤条件。
- 系统需要支持多种过滤标准:在数据处理过程中,可能需要根据不同的条件进行过滤,使用过滤器模式可以轻松实现这一点。
- 复杂业务逻辑与过滤逻辑分离:过滤器模式能够将过滤逻辑与具体的业务逻辑分开,使得代码更易于维护和扩展。
典型应用场景
- 用户过滤:在用户管理系统中,根据用户角色、状态等多个属性过滤用户列表。
- 商品筛选:在电商平台中,根据价格、类别、品牌等条件筛选商品。
- 数据处理:在数据分析和处理过程中,根据多个条件对数据集进行筛选。
3. Python 实现过滤器模式
接下来,我们将通过一个具体的例子来实现过滤器模式。假设我们需要处理一组用户对象,并根据不同的条件进行过滤(如按年龄和性别过滤)。
3.1 定义用户类
首先,我们定义一个 User
类,表示用户对象。
class User:def __init__(self, name, gender, age):self.name = nameself.gender = genderself.age = agedef __repr__(self):return f"User(name={self.name}, gender={self.gender}, age={self.age})"
3.2 定义过滤器接口
接下来,定义一个过滤器接口,声明过滤方法。
from abc import ABC, abstractmethod# 过滤器接口
class Filter(ABC):@abstractmethoddef filter(self, users):pass
3.3 实现具体过滤器
我们实现具体的过滤器来根据年龄和性别过滤用户。
# 按年龄过滤用户
class AgeFilter(Filter):def __init__(self, age):self.age = agedef filter(self, users):return [user for user in users if user.age >= self.age]# 按性别过滤用户
class GenderFilter(Filter):def __init__(self, gender):self.gender = genderdef filter(self, users):return [user for user in users if user.gender == self.gender]
3.4 定义上下文类
定义一个上下文类,用于管理用户集合和应用过滤器。
class UserFilter:def __init__(self):self.filters = []def add_filter(self, filter):self.filters.append(filter)def filter(self, users):filtered_users = usersfor filter in self.filters:filtered_users = filter.filter(filtered_users)return filtered_users
3.5 测试过滤器模式
最后,我们创建用户列表,并使用过滤器模式来筛选用户。
# 创建用户列表
users = [User("Alice", "Female", 30),User("Bob", "Male", 24),User("Charlie", "Male", 35),User("Diana", "Female", 28),User("Ethan", "Male", 22)
]# 使用过滤器模式
user_filter = UserFilter()# 添加过滤器
user_filter.add_filter(AgeFilter(25)) # 年龄过滤
user_filter.add_filter(GenderFilter("Male")) # 性别过滤# 过滤用户
filtered_users = user_filter.filter(users)
print("Filtered Users:", filtered_users)
输出结果:
Filtered Users: [User(name=Charlie, gender=Male, age=35)]
在这个例子中,我们通过组合不同的过滤器实现了对用户列表的灵活筛选。用户过滤器先应用年龄过滤,再应用性别过滤,最终得到了符合条件的用户列表。
4. 过滤器模式的优缺点
优点
- 灵活性:通过组合不同的过滤器,可以轻松实现复杂的过滤逻辑。
- 代码重用:过滤器可以被复用,在不同的场合下进行组合使用。
- 易于扩展:如果需要增加新的过滤条件,只需添加新的过滤器类,无需修改已有代码。
缺点
- 性能开销:如果过滤器较多,过滤链条较长,可能会增加性能开销。
- 复杂性:在某些情况下,过多的过滤器可能会使代码变得复杂,增加理解和维护的难度。
5. 改进过滤器模式:使用链式调用
在Python中,我们可以对过滤器模式进行改进,支持链式调用,使得代码更加简洁和流畅。
class ChainedFilter(Filter):def __init__(self):self.filters = []def add_filter(self, filter):self.filters.append(filter)return self # 支持链式调用def filter(self, users):filtered_users = usersfor filter in self.filters:filtered_users = filter.filter(filtered_users)return filtered_users
使用链式调用的方式构建过滤器:
# 创建链式过滤器
chained_filter = ChainedFilter()
chained_filter.add_filter(AgeFilter(25)).add_filter(GenderFilter("Female"))# 过滤用户
filtered_users = chained_filter.filter(users)
print("Filtered Users (Chained):", filtered_users)
这种改进方式使得过滤器的使用更加灵活,便于构建复杂的过滤条件。
6. 结论
过滤器模式是一种非常有用的设计模式,尤其适合在需要从大量数据中进行条件筛选的场合。通过将过滤逻辑与业务逻辑分离,过滤器模式能够提供一种灵活的方式来组合和应用多个过滤条件。
在Python中,利用类和继承的特性,我们可以简单而高效地实现过滤器模式。希望本文能够帮助你更深入地理解过滤器模式,并能够灵活运用到实际开发中。
如果你有任何问题或建议,欢迎在评论区讨论!
相关文章:
【设计模式】深入理解Python中的过滤器模式(Filter Pattern)
深入理解Python中的过滤器模式(Filter Pattern) 在软件设计中,面对复杂的数据处理需求时,我们常常需要从一组数据中筛选出符合特定条件的子集。**过滤器模式(Filter Pattern)**是一种能够简化这种操作的设…...

vue的动态组件 keep-alive
1. 什么是动态组件 动态组件指的是 动态切换组件的显示与隐藏 2. 如何实现动态组件渲染 vue提供了一个内置的<component>组件,专门用来实现动态组件的渲染。 作用:组件的占位符is的值表示要渲染的组件 示例代码如下: Left.vue的代…...
现代框架开发官网
一、项目背景 维护过 灵犀官网、企业邮官网、免费邮官网 均使用 jquery webpack多页面打包的方式 开发起来较为繁琐 新的官网项目,想使用现代前端框架,但SPA应用不利于SEO 使用SSR方案又依赖运维,增加维护和沟通成本 二、SSG vs 预渲染 S…...

一篇文章快速认识YOLO11 | 关键改进点 | 安装使用 | 模型训练和推理
前言 本文分享YOLO11的关键改进点、性能对比、安装使用、模型训练和推理等内容。 YOLO11 是 Ultralytics 最新的实时目标检测器,凭借更高的精度、速度和效率重新定义了可能性。 除了传统的目标检测外,YOLO11 还支持目标跟踪、实例分割、姿态估计、OBB…...
AtCoder Beginner Contest 375(A,B,C,D,E,F)(大模拟,前缀和,dp,离线处理,Floyd)
比赛链接 AtCoder Beginner Contest 375 A题 代码 #pragma GCC optimize("O2") #pragma GCC optimize("O3") #include <bits/stdc.h> using namespace std; #define int long long const int N 2e5 5, M 1e6 5; const int inf 0x3f3f3f3f3f…...
认识maven
什么是 Maven? Maven 是一个开源的项目管理工具,主要用于 Java 项目的构建、依赖管理和项目生命周期管理。它提供了一种标准的项目结构和管理流程,使得开发人员能够更轻松地管理项目的构建过程,提高代码的可重用性和可维护性。 …...
OSINT技术情报精选·2024年10月第2周
OSINT技术情报精选2024年10月第2周 2024.10.16版权声明:本文为博主chszs的原创文章,未经博主允许不得转载。 1、亿欧智库:《2024中国高精定位服务产业白皮书》 报告的主要内容如下: 产业背景:在“北斗”发展态势的…...

中企通信赋能中信戴卡入选工信部颁发的2023年工业互联网试点示范名单
2024年10月17日,北京-随着工业互联网的迅猛发展,网络安全已成为国家关注的重点议题之一。日前,工业和信息化部(工信部)公布了2023年工业互联网试点示范名单,中企网络通信技术有限公司(简称“中企…...

【C语言】函数的声明与定义
函数的声明 用户自定义函数需要在main函数之前进行声明,用分号结尾。 函数的定义 用户自定义函数在main函数之后进行定义,需要写出具体形参的变量名。注意函数的返回值和返回值类型要一一对应。 函数的调用 调用时,直接使用函数名进行调用&am…...

游戏如何应对薅羊毛问题
在大众眼里,“薅羊毛”是指在电商领域,“羊毛党”利用平台、商家的促销规则,低价获取商品和服务的行为。如前不久“小天鹅被一夜薅走7000万”的案例震惊全网。 然而实际上,“薅羊毛”现象不仅存在于电商场景,在游戏中…...
Chromium html<script>对应c++接口定义
<script>:脚本元素 <script> 元素用于嵌入可执行代码或数据,这通常用作嵌入或者引用 JavaScript 代码。<script> 元素也能在其他语言中使用,比如 WebGL 的 GLSL 着色器语言和 JSON。 更多参考:<script>&…...

ollama + fastgpt+m3e本地部署
ollama fastgptm3e本地部署 开启WSL更新wsl安装ubuntu docker下载修改docker镜像源开启WSL integration 安装fastgpt先创建一个文件夹来放置一些配置文件用命令下载fastgpt配置文件用命令下载docker的部署文件 启动容器M3E下载ollama下载oneapi配置登录oneapi配置ollama渠道配…...

Linux执行source /etc/profile命令报错:权限不够问(已解决)
1.问题 明明以root账号登录Linux系统,在终端执行命令source /etc/profile时 显示权限不够 如下图: 2.问题原因 可能在编辑 /etc/profile 这个文件时不小心把开头的 井号 ‘#’ 给删除了 如图: 这里一定要有# 3.解决办法 进入/etc/pro…...
Windows 11开发全解析
Windows 11开发全解析 一、搭建开发环境 在开始Windows 11开发之前,搭建一个高效的开发环境是至关重要的。Windows 11提供了多种工具和框架,可以帮助开发者快速搭建起一个强大的开发环境。 1. Visual Studio 2024 Visual Studio 2024是微软为Windows…...
如何进行数学家式的学习思考?
如何进行数学家式的学习思考? 学生阶段的数学学习是非常重要的,对这一点很少有人质疑。一提起数学学习,一些学生、家长甚至一些教师认为,学生的数学学习往往侧重于掌握基本概念、公式和解题技巧,通过做题来巩固知识和提…...

自定义类型--结构体
目录 1. 结构体类型的声明 1.1结构的声明 1.2 结构体变量的创建和初始化 1.3不完全结构体 1.4结构的⾃引⽤ 2 结构体的内存对齐 2.1offsetof 2.2 对⻬规则 2.3 为什么存在内存对⻬? 2.4修改默认对⻬数 3. 结构体传参 4 结构体实现位段 4.1什么是位段 4.2 位段的内…...

笔试练习day7
目录 OR59 字符串中找出连续最长的数字串题目解析解法(双指针遍历)代码 NC109 岛屿数量题目解析解法代码(dfs)dfs的实现 拼三角题目解析解法(枚举)代码 感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接 🐒🐒🐒 个人主页 &…...

python 爬虫 入门 一、基础工具
目录 一,网页开发者工具的使用 二、通过python发送请求 (一)、get (二)、带参数的get (三)、post 后续:数据解析 一,网页开发者工具的使用 我们可以用 requests 库…...
金融衍生品中的风险对冲策略分析
金融衍生品是现代金融市场中不可或缺的一部分,它们通过标的资产的价格波动为投资者提供了多样的风险管理工具。随着市场的不确定性和复杂性增加,风险对冲成为企业和个人投资者的首要任务。本文将深入探讨金融衍生品中的常见风险对冲策略,分析…...

linux下建立软链接
深度学习训练中经常会遇到数据量庞大或者工程中模型报错太多导致磁盘空间不够,但是又不想修改原来在代码中写的路径,这个时候制作软连接很有作用,把占用量大的目录移到别的空闲磁盘,然后在原来的目录做一个软连接指向那个移到的空…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

3.3.1_1 检错编码(奇偶校验码)
从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...

HTML 列表、表格、表单
1 列表标签 作用:布局内容排列整齐的区域 列表分类:无序列表、有序列表、定义列表。 例如: 1.1 无序列表 标签:ul 嵌套 li,ul是无序列表,li是列表条目。 注意事项: ul 标签里面只能包裹 li…...

select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...

selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...
Linux C语言网络编程详细入门教程:如何一步步实现TCP服务端与客户端通信
文章目录 Linux C语言网络编程详细入门教程:如何一步步实现TCP服务端与客户端通信前言一、网络通信基础概念二、服务端与客户端的完整流程图解三、每一步的详细讲解和代码示例1. 创建Socket(服务端和客户端都要)2. 绑定本地地址和端口&#x…...

基于TurtleBot3在Gazebo地图实现机器人远程控制
1. TurtleBot3环境配置 # 下载TurtleBot3核心包 mkdir -p ~/catkin_ws/src cd ~/catkin_ws/src git clone -b noetic-devel https://github.com/ROBOTIS-GIT/turtlebot3.git git clone -b noetic https://github.com/ROBOTIS-GIT/turtlebot3_msgs.git git clone -b noetic-dev…...

VM虚拟机网络配置(ubuntu24桥接模式):配置静态IP
编辑-虚拟网络编辑器-更改设置 选择桥接模式,然后找到相应的网卡(可以查看自己本机的网络连接) windows连接的网络点击查看属性 编辑虚拟机设置更改网络配置,选择刚才配置的桥接模式 静态ip设置: 我用的ubuntu24桌…...