【过滤器设计模式详解】C/Java/JS/Go/Python/TS不同语言实现
简介
过滤器模式(Filter Pattern)或标准模式(Criteria Pattern),是一种结构型模式。这种模式允许使用不同的标准条件来过滤一组对象,并通过逻辑运算的方式把各条件连接起来,它结合多个标准来获得单一标准。
例子将创建一个 Person 对象、Criteria 接口和实现了该接口的实体类,来过滤 Person 对象的列表。Test 类使用 Criteria 对象,基于各种标准和它们的结合来过滤 Person 对象的列表。
作用
- 可以通过层层筛选,过滤出自己想要的结果。
- 面向接口编程,将对象过滤,接口不依赖对象;对象无入侵,每次只需要增加规则就行了,不用修改过滤对象类。
实现步骤
- 创建Criteria基础接口,定义过滤方法。
- 创建多个条件类,实现过滤逻辑。
- 客户端调用时,通条件类过滤数据。
UML

Java代码
标准接口类
// Criteria.java 定义抽象标准接口,聚合实体对象
public interface Criteria {
public List<Person> filter(List<Person> persons);
}
具体标准实现类
// AndCriteria.java 定义And过滤标准
public class AndCriteria implements Criteria { private Criteria criteria;
private Criteria otherCriteria; public AndCriteria(Criteria criteria, Criteria otherCriteria) {
this.criteria = criteria;
this.otherCriteria = otherCriteria;
} @Override
public List<Person> filter(List<Person> persons) {
List<Person> firstCriteriaPersons = criteria.filter(persons);
return otherCriteria.filter(firstCriteriaPersons);
}
} // OrCriteria.java 定义Or过滤标准
public class OrCriteria implements Criteria { private Criteria criteria;
private Criteria otherCriteria; public OrCriteria(Criteria criteria, Criteria otherCriteria) {
this.criteria = criteria;
this.otherCriteria = otherCriteria;
} @Override
public List<Person> filter(List<Person> persons) {
List<Person> firstCriteriaItems = criteria.filter(persons);
List<Person> otherCriteriaItems = otherCriteria.filter(persons); for (Person person : otherCriteriaItems) {
if (!firstCriteriaItems.contains(person)) {
firstCriteriaItems.add(person);
}
}
return firstCriteriaItems;
}
} // CriteriaFemale.java 根据标准接口实现的过滤
public class CriteriaFemale implements Criteria { @Override
public List<Person> filter(List<Person> persons) {
List<Person> femalePersons = new ArrayList<Person>();
for (Person person : persons) {
if (person.getGender().equalsIgnoreCase("FEMALE")) {
femalePersons.add(person);
}
}
return femalePersons;
}
} // CriteriaMale.java 根据标准接口实现的过滤
public class CriteriaMale implements Criteria { @Override
public List<Person> filter(List<Person> persons) {
List<Person> malePersons = new ArrayList<Person>();
for (Person person : persons) {
if (person.getGender().equalsIgnoreCase("MALE")) {
malePersons.add(person);
}
}
return malePersons;
}
} // CriteriaSingle.java 根据标准接口实现按属性的过滤
public class CriteriaSingle implements Criteria { @Override
public List<Person> filter(List<Person> persons) {
List<Person> singlePersons = new ArrayList<Person>();
for (Person person : persons) {
if (person.getStatus().equalsIgnoreCase("SINGLE")) {
singlePersons.add(person);
}
}
return singlePersons;
}
}
业务实体类
// Person.java 定义一个实体类,用来过滤的对象
public class Person { private String name;
private String gender;
private String status; public Person(String name, String gender, String status) {
this.name = name;
this.gender = gender;
this.status = status;
} public String getName() {
return name;
} public String getGender() {
return gender;
} public String getStatus() {
return status;
} public String toString() {
return "Person : [ Name : " + getName() + ", Gender : "
+ getGender() + ", Marital Status : " + getStatus() + " ]";
}
}
测试调用
/**
* 过滤器模式就是不断组合过滤条件,然后层层过滤的模式
* 这里是简单演示,用List筛选来模拟过滤,实际例子有各种数据结构
*/
List<Person> persons = new ArrayList<Person>(); persons.add(new Person("王男单", "Male", "Single"));
persons.add(new Person("李男婚", "Male", "Married"));
persons.add(new Person("张女婚", "Female", "Married"));
persons.add(new Person("赵女单", "Female", "Single"));
persons.add(new Person("刘男单", "Male", "Single"));
persons.add(new Person("杨男单", "Male", "Single")); Criteria male = new CriteriaMale();
Criteria female = new CriteriaFemale();
Criteria single = new CriteriaSingle();
Criteria singleMale = new AndCriteria(single, male);
Criteria singleOrFemale = new OrCriteria(single, female); // 查询男性
System.out.println("Males: ");
printPersons(male.filter(persons)); // 查询女性
System.out.println("\nFemales: ");
printPersons(female.filter(persons)); // 嵌套查询女性且单身
System.out.println("\nFemales and Single: ");
printPersons(single.filter(female.filter(persons))); // 查询男性男性单身
System.out.println("\nSingle Males: ");
printPersons(singleMale.filter(persons)); // 查询女性或单身
System.out.println("\nSingle Or Females: ");
printPersons(singleOrFemale.filter(persons));
C代码
head文件
// func.h文件
#include <stdio.h>
#include <ctype.h>
#include <stdlib.h>
#include <stdbool.h>
#include <string.h> char *str_toupper(char str[]);
char *str_tolower(char str[]); // 定义用于过滤的Person数组查询对象
// 因C语言无法定义动态数组,将数组长度存在此处
typedef struct FilterPersons
{
int length;
struct Person **persons;
} FilterPersons; // 定义一个实体类,用来过滤的对象
typedef struct Person
{
char name[50];
char gender[20];
char status[20];
char *(*get_name)(struct Person *);
char *(*to_string)(struct Person *);
bool (*is_contained)(struct Person *, struct FilterPersons *);
} Person;
Person *person_constructor(char *name, char *gender, char *status); // 定义抽象标准接口,聚合要过滤的对象集合
typedef struct Criteria
{
struct Criteria *first_criteria;
struct Criteria *other_criteria;
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} Criteria; // 根据标准接口实现的过滤
typedef struct CriteriaFemale
{
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} CriteriaFemale;
CriteriaFemale *criteria_female_constructor(); // 根据标准接口实现的过滤
typedef struct CriteriaMale
{
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} CriteriaMale;
CriteriaMale *criteria_male_constructor(); // 根据标准接口实现的过滤
typedef struct CriteriaSingle
{
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} CriteriaSingle;
CriteriaSingle *criteria_single_constructor(); // 定义And过滤标准
typedef struct AndCriteria
{
struct Criteria *first_criteria;
struct Criteria *other_criteria;
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} AndCriteria;
AndCriteria *and_criteria_constructor(Criteria *, Criteria *); // 定义Or过滤标准
typedef struct OrCriteria
{
struct Criteria *first_criteria;
struct Criteria *other_criteria;
FilterPersons *(*filter)(FilterPersons *, struct Criteria *);
} OrCriteria;
OrCriteria *or_criteria_constructor(Criteria *, Criteria *);
条件接口类
// criteria.c 定义抽象标准接口,聚合实体对象
#include "func.h"
// c语言没有抽象类或接口,Criterial作为基础struct定义在head, 这里放一些公共函数
char *str_toupper(char str[])
{
int size = strlen(str);
char *result = (char *)malloc(size * sizeof(char));
int i = 0;
while (str[i])
{
result[i] = toupper(str[i]);
i++;
} return result;
} char *str_tolower(char str[])
{
int size = strlen(str);
char *result = (char *)malloc(size * sizeof(char));
int i = 0;
while (str[i])
{
result[i] = tolower(str[i]);
i++;
} return result;
}
具体标准实现类
// and_criteria.c 定义And过滤标准
#include "func.h"
// 先过滤条件1,再把结果按照条件2进行过滤
FilterPersons *and_criteria_filter(FilterPersons *filter, Criteria *criteria)
{
FilterPersons *first_filter = criteria->first_criteria->filter(filter, criteria);
return criteria->other_criteria->filter(first_filter, criteria);
} AndCriteria *and_criteria_constructor(Criteria *first_criteria, Criteria *other_criteria)
{
Criteria *criteria = (Criteria *)malloc(sizeof(Criteria));
criteria->filter = &and_criteria_filter;
AndCriteria *and_criteria = (AndCriteria *)criteria;
and_criteria->first_criteria = first_criteria;
and_criteria->other_criteria = other_criteria;
and_criteria->filter = &and_criteria_filter;
return and_criteria;
} // or_criteria.c 定义Or过滤标准
#include "func.h"
// 先过滤条件1,再把结果按照条件2进行过滤
FilterPersons *or_criteria_filter(FilterPersons *filter, Criteria *criteria)
{
FilterPersons *first_filter = criteria->first_criteria->filter(filter, criteria);
FilterPersons *other_filter = criteria->other_criteria->filter(filter, criteria); int first_size = first_filter->length;
int other_size = other_filter->length; for (int i = 0; i < other_size; i++)
{
Person *person = other_filter->persons[i];
// 符合项如果不存在条件1里,则追加到条件1里去
if (!person->is_contained(person, first_filter))
{
// 这里数组长度直接增加
first_filter->persons[first_size++] = person;
}
} first_filter->length = first_size; return first_filter;
} OrCriteria *or_criteria_constructor(Criteria *first_criteria, Criteria *other_criteria)
{
Criteria *criteria = (Criteria *)malloc(sizeof(Criteria));
criteria->filter = &or_criteria_filter;
OrCriteria *or_criteria = (OrCriteria *)criteria;
or_criteria->first_criteria = first_criteria;
or_criteria->other_criteria = other_criteria;
or_criteria->filter = &or_criteria_filter;
return or_criteria;
} // ccriteria_female.c 根据标准接口实现的过滤
#include "func.h"
// 根据是否女性进行过滤
FilterPersons *criteria_female_filter(FilterPersons *filter, Criteria *criteria)
{
int person_size = filter->length;
int *female_index_list = (int *)malloc(person_size * sizeof(int));
int count = 0;
char *gender;
for (int i = 0; i < person_size; i++)
{
if (filter->persons[i] == NULL)
{
break;
}
gender = str_toupper(filter->persons[i]->gender);
if (strcmp(gender, "FEMALE") == 0)
{
// 记录下所有符合条件的person下标
female_index_list[count] = i;
count += 1;
}
}
free(gender);
free(female_index_list); // 将符合条件的person追加到新数组
Person **female_persons = (Person **)calloc(count, sizeof(Person));
for (int i = 0; i < count; i++)
{
female_persons[i] = filter->persons[female_index_list[i]];
}
FilterPersons *female_filter = (FilterPersons *)calloc(1, sizeof(FilterPersons));
female_filter->length = count;
female_filter->persons = female_persons;
return female_filter;
} CriteriaFemale *criteria_female_constructor()
{
Criteria *criteria = (Criteria *)malloc(sizeof(Criteria));
criteria->filter = &criteria_female_filter;
CriteriaFemale *criteria_female = (CriteriaFemale *)criteria;
criteria_female->filter = &criteria_female_filter;
return criteria_female;
} // criteria_male.c 根据标准接口实现的过滤
#include "func.h" // 根据是否男性进行过滤
FilterPersons *criteria_male_filter(FilterPersons *filter, Criteria *criteria)
{
int person_size = filter->length;
int *male_index_list = (int *)malloc(person_size * sizeof(int));
int count = 0;
char *gender;
for (int i = 0; i < person_size; i++)
{
if (filter->persons[i] == NULL)
{
break;
}
gender = str_tolower(filter->persons[i]->gender);
if (strcmp(gender, "male") == 0)
{
// 记录下所有符合条件的person下标
male_index_list[count] = i;
count += 1;
}
}
free(gender);
free(male_index_list); // 将符合条件的person追加到新数组
Person **male_persons = (Person **)calloc(count, sizeof(Person));
for (int i = 0; i < count; i++)
{
male_persons[i] = filter->persons[male_index_list[i]];
}
FilterPersons *male_filter = (FilterPersons *)calloc(1, sizeof(FilterPersons));
male_filter->length = count;
male_filter->persons = male_persons;
return male_filter;
} CriteriaMale *criteria_male_constructor()
{
Criteria *criteria = (Criteria *)malloc(sizeof(Criteria));
criteria->filter = &criteria_male_filter;
CriteriaMale *criteria_male = (CriteriaMale *)criteria;
criteria_male->filter = &criteria_male_filter;
return criteria_male;
} // criteria_single.c 根据标准接口实现按属性的过滤
#include "func.h" // 根据是否单身进行过滤
FilterPersons *criteria_single_filter(FilterPersons *filter, Criteria *criteria)
{
int person_size = filter->length;
int *single_index_list = (int *)malloc(person_size * sizeof(int));
int count = 0;
char *status;
for (int i = 0; i < person_size; i++)
{
if (filter->persons[i] == NULL)
{
break;
}
status = str_tolower(filter->persons[i]->status);
if (strcmp(status, "single") == 0)
{
// 记录下所有符合条件的person下标
single_index_list[count] = i;
count += 1;
}
}
free(status);
free(single_index_list); // 将符合条件的person追加到新数组
Person **single_persons = (Person **)calloc(count, sizeof(Person));
for (int i = 0; i < count; i++)
{
single_persons[i] = filter->persons[single_index_list[i]];
} FilterPersons *single_filter = (FilterPersons *)calloc(1, sizeof(FilterPersons));
single_filter->length = count;
single_filter->persons = single_persons;
return single_filter;
} CriteriaSingle *criteria_single_constructor()
{
Criteria *criteria = (Criteria *)malloc(sizeof(Criteria));
criteria->filter = &criteria_single_filter;
CriteriaSingle *criteria_single = (CriteriaSingle *)criteria;
criteria_single->filter = &criteria_single_filter;
return criteria_single;
}
业务实体类
// person.c 定义一个实体类,用来过滤的对象
#include "func.h"
// 是否被包含在对象数组中
bool person_is_contained(Person *person, FilterPersons *filter)
{
int persons_size = filter->length;
for (int i = 0; i < persons_size; i++)
{
if (filter->persons[i] == person)
{
return true;
}
}
return false;
} char *person_get_name(Person *person)
{
return person->name;
} // 返回字符串
char *person_to_string(Person *person)
{
char *result = (char *)malloc(500 * sizeof(char));
strcat(result, "Person :[ name :");
strcat(result, person->name);
strcat(result, ", gender : ");
strcat(result, person->gender);
strcat(result, ", status : ");
strcat(result, person->status);
strcat(result, "]");
return result;
} Person *person_constructor(char *name, char *gender, char *status)
{
Person *person = (Person *)malloc(sizeof(Person));
strncpy(person->name, name, 50);
strncpy(person->gender, gender, 20);
strncpy(person->status, status, 20);
person->get_name = &person_get_name;
person->to_string = &person_to_string;
person->is_contained = &person_is_contained;
return person;
}
测试调用
/**
* 过滤器模式就是不断组合过滤条件,然后层层过滤的模式
* 这里是简单演示,用List筛选来模拟过滤,实际例子有各种数据结构。
*/
int data_size = 6;
// 定义一些数据
char data[6][3][100] = {
{"王男单", "Male", "Single"},
{"李男婚", "Male", "Married"},
{"张女婚", "Female", "Married"},
{"赵女单", "Female", "Single"},
{"刘男单", "Male", "Single"},
{"杨男单", "Male", "Single"}};
// 定义persons数组
Person *persons[data_size];
for (int i = 0; i < data_size; i++)
{
char *name = data[i][0];
char *gender = data[i][1];
char *status = data[i][2];
Person *person = person_constructor(name, gender, status);
persons[i] = person;
}
// 构建查询对象
FilterPersons *filter_persons = (FilterPersons *)malloc(sizeof(FilterPersons *));
filter_persons->length = data_size;
filter_persons->persons = persons;
// 声明属性过滤条件,可用Criteria或具体条件声明
Criteria *criteria_male = (Criteria *)criteria_male_constructor();
CriteriaFemale *criteria_female = criteria_female_constructor();
Criteria *criteria_single = (Criteria *)criteria_single_constructor();
// 声明逻辑条件,传入属性过滤条件
Criteria *single_male = (Criteria *)and_criteria_constructor(criteria_single, criteria_male);
OrCriteria *single_or_female = or_criteria_constructor(criteria_single, (Criteria *)criteria_female);
// 查询男性
printf("\n Males: ");
print_persons(((CriteriaMale *)criteria_male)->filter(filter_persons, criteria_male));
// 查询女性
printf("\nFemales: ");
print_persons(criteria_female->filter(filter_persons, (Criteria *)criteria_female));
// 嵌套查询女性且单身
printf("\nFemales and Single: ");
FilterPersons *females = criteria_female->filter(filter_persons, (Criteria *)criteria_female);
print_persons(((CriteriaSingle *)criteria_single)->filter(females, criteria_single));
// 查询男性男性单身
printf("\nSingle Males: ");
// 逐个条件过滤,与下面AndCriteria效果相同
FilterPersons *single_males = ((CriteriaMale *)criteria_male)->filter(filter_persons, criteria_male);
print_persons(((CriteriaSingle *)criteria_single)->filter(single_males, criteria_single));
printf("\nSingle Males: ");
// 通过AndCriteria来过滤
FilterPersons *single_males2 = ((AndCriteria *)single_male)->filter(filter_persons, single_male);
print_persons(single_males2);
// 查询女性或单身
printf("\nSingle Or Females: ");
print_persons(single_or_female->filter(filter_persons, (Criteria *)single_or_female));
free(filter_persons);
free(criteria_male);
free(criteria_female);
free(criteria_single);
free(single_male);
free(single_or_female);
return 0;
更多语言版本
不同语言实现设计模式:https://github.com/microwind/design-pattern
【过滤器设计模式详解】C/Java/JS/Go/Python/TS不同语言实现的更多相关文章
- [ 转载 ] Java开发中的23种设计模式详解(转)
Java开发中的23种设计模式详解(转) 设计模式(Design Patterns) ——可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类 ...
- android java 设计模式详解 Demo
android java 设计模式详解 最近看了一篇设计模式的文章,深得体会,在此基础我将每种设计模式的案例都写成Demo的形式,方便读者研究学习, 首先先将文章分享给大家: 设计模式(Design ...
- Java温故而知新(5)设计模式详解(23种)
一.设计模式的理解 刚开始“不懂”为什么要把很简单的东西搞得那么复杂.后来随着软件开发经验的增加才开始明白我所看到的“复杂”恰恰就是设计模式的精髓所在,我所理解的“简单”就是一把钥匙开一把锁的模式,目 ...
- JAVA设计模式简介及六种常见设计模式详解
一.什么是设计模式 ...
- Javascript设计模式详解
Javascript常用的设计模式详解 阅读目录 一:理解工厂模式 二:理解单体模式 三:理解模块模式 四:理解代理模式 五:理解职责链模式 六:命令模式的理解: 七:模板方法模式 八:理解javas ...
- Javascript常用的设计模式详解
Javascript常用的设计模式详解 阅读目录 一:理解工厂模式 二:理解单体模式 三:理解模块模式 四:理解代理模式 五:理解职责链模式 六:命令模式的理解: 七:模板方法模式 八:理解javas ...
- javascript设计模式详解之策略模式
接上篇命令模式来继续看下js设计模式中另一种常用的模式,策略模式.策略模式也是js开发中常用的一种实例,不要被这么略显深邃的名字给迷惑了.接下来我们慢慢看一下. 一.基本概念与使用场景: 基本概念:定 ...
- iOS中MVC等设计模式详解
iOS中MVC等设计模式详解 在iOS编程,利用设计模式可以大大提高你的开发效率,虽然在编写代码之初你需要花费较大时间把各种业务逻辑封装起来.(事实证明这是值得的!) 模型-视图-控制器(MVC)设计 ...
- javascript设计模式详解之命令模式
每种设计模式的出现都是为了弥补语言在某方面的不足,解决特定环境下的问题.思想是相通的.只不过不同的设计语言有其特定的实现.对javascript这种动态语言来说,弱类型的特性,与生俱来的多态性,导致某 ...
- 16个PHP设计模式详解
说明:该教程全部截选自实验楼教程[16个PHP设计模式详解]:主要介绍16个常用的设计模式的基础概念和技术要点,通过UML类图帮助理解设计模式中各个类之间的关联关系,针对每种设计模式都使用PHP完成了 ...
随机推荐
- OpenStack 云主机ping通外网
- 尝试改善科研V2
参考链接: https://fulequn.github.io/2022/09/26/Article202209261/ https://www.xljsci.com/ https://apps.an ...
- 页面与java后台之 上传文件与服务器加载
添加相关依赖包commons-fileupload等: 页面(注:编码enctype="multipart/form-data" 文件按钮属性multiple) java ...
- 学生管理系统CLI版
学生管理系统CLI版 学生类 package com.itheima_03; public class Student { String sid; String name; String age; S ...
- 更改docker里mysql的字符编码
进入容器: docker exec -it 容器id/容器名称 bash cp时容器中的目录写法 容器名称/容器id:容器目录 退出容器使用exit 1 首先去mysql容器中寻找mysq ...
- 游戏内存优化之使用16位纹理/NPOT
转自:https://blog.csdn.net/oqqQuZi1234567/article/details/41749599 图片文件大小和纹理内存占用是两码事.假设他们是帐篷.图片文件就相当于帐 ...
- 设计模式 - 单例模式 Singleton Pattern - C#
单例模式 Singleton Pattern 1.单例模式设计模式属于创建型模式 2.是单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建.这个类提供了一种访问其唯一的对象的方式,可以直接访 ...
- jquery的ajax方法获取不到return返回值
/** 2 * 方式:(1)同步调用 (2)在ajax函数中return值 3 * 结果:返回 1.未成功获取返回值 4 * 失败原因:ajax内部是一个或多个定义的函数,ajax中return返回值 ...
- shell命令查找文件
1.find命令的参数下面是find命令一些常用参数的例子,有用到的时候查查就行了,像上面前几个贴子,都用到了其中的的一些参数,也可以用man或查看论坛里其它贴子有find命令手册使用name选项文件 ...
- win10edge浏览器个人账户退出登录后再次登录自动登录问题
edge浏览器退出登录后,再次点击登录以同步数据会自动登录,可查看书签等个人数据 解决方法: 先在浏览器里面退出账户. 1.设置--电子邮件和账户--管理 2.登录后--安全--安全仪表板--高级安全 ...