AntPathMatcher实现Ant风格具体实现

时间: 2024-03-16 12:44:35 浏览: 16
AntPathMatcher是Spring框架中的一个工具类,它提供了对Ant风格路径匹配的支持。它的具体实现如下: ```java public class AntPathMatcher { // Ant风格路径分隔符 private static final String PATH_SEPARATOR = "/"; // Ant风格路径分隔符 private static final char PATH_SEPARATOR_CHAR = '/'; // 单字符通配符 private static final char SINGLE_CHAR = '?'; // 多字符通配符 private static final char MULTI_CHAR = '*'; // 匹配路径时是否区分大小写 private final boolean caseSensitive; public AntPathMatcher() { this(true); } public AntPathMatcher(boolean caseSensitive) { this.caseSensitive = caseSensitive; } /** * 判断指定的路径是否匹配模式. */ public boolean match(String pattern, String path) { if (path.startsWith(PATH_SEPARATOR) != pattern.startsWith(PATH_SEPARATOR)) { return false; } String[] pattDirs = tokenizePattern(pattern); String[] pathDirs = tokenizePath(path); int pattIdxStart = 0; int pattIdxEnd = pattDirs.length - 1; int pathIdxStart = 0; int pathIdxEnd = pathDirs.length - 1; // 处理路径中前缀部分 while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) { String pattDir = pattDirs[pattIdxStart]; if (MULTI_CHAR == pattDir.charAt(0)) { break; } if (!matchStrings(pattDir, pathDirs[pathIdxStart])) { return false; } pattIdxStart++; pathIdxStart++; } if (pathIdxStart > pathIdxEnd) { // 如果路径已经比较完了 if (pattIdxStart > pattIdxEnd) { // 如果模式也已经比较完了 return (pattern.endsWith(PATH_SEPARATOR) == path.endsWith(PATH_SEPARATOR)); } if (pattIdxStart == pattIdxEnd && MULTI_CHAR == pattDirs[pattIdxStart].charAt(0) && path.endsWith(PATH_SEPARATOR)) { return true; } for (int i = pattIdxStart; i <= pattIdxEnd; i++) { if (MULTI_CHAR == pattDirs[i].charAt(0)) { return true; } } return false; } else if (pattIdxStart > pattIdxEnd) { return false; } // 处理路径中后缀部分 while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) { String pattDir = pattDirs[pattIdxEnd]; if (MULTI_CHAR == pattDir.charAt(0)) { break; } if (!matchStrings(pattDir, pathDirs[pathIdxEnd])) { return false; } pattIdxEnd--; pathIdxEnd--; } if (pathIdxStart > pathIdxEnd) { // 如果路径已经比较完了 while (pattIdxStart <= pattIdxEnd) { if (MULTI_CHAR == pattDirs[pattIdxEnd].charAt(0)) { pattIdxEnd--; continue; } return false; } return true; } while (pattIdxStart != pattIdxEnd && pathIdxStart <= pathIdxEnd) { int patIdxTmp = -1; for (int i = pattIdxStart + 1; i <= pattIdxEnd; i++) { if (MULTI_CHAR == pattDirs[i].charAt(0)) { patIdxTmp = i; break; } } if (patIdxTmp == pattIdxStart + 1) { pattIdxStart++; continue; } int patLength = (patIdxTmp - pattIdxStart - 1); int strLength = (pathIdxEnd - pathIdxStart + 1); int foundIdx = -1; strLoop: for (int i = 0; i <= strLength - patLength; i++) { for (int j = 0; j < patLength; j++) { String subPat = pattDirs[pattIdxStart + j + 1]; String subStr = pathDirs[pathIdxStart + i + j]; if (!matchStrings(subPat, subStr)) { continue strLoop; } } foundIdx = pathIdxStart + i; break; } if (foundIdx == -1) { return false; } pattIdxStart = patIdxTmp; pathIdxStart = foundIdx + patLength; } for (int i = pattIdxStart; i <= pattIdxEnd; i++) { if (MULTI_CHAR == pattDirs[i].charAt(0)) { continue; } if (!matchStrings(pattDirs[i], pathDirs[pathIdxStart])) { return false; } pathIdxStart++; } return true; } /** * 判断两个字符串是否匹配. */ private boolean matchStrings(String pattern, String str) { return getStringMatcher(pattern, str).matches(); } /** * 将模式字符串分隔成目录. */ private String[] tokenizePattern(String pattern) { if (pattern == null) { return new String[0]; } return tokenizeToStringArray(pattern, PATH_SEPARATOR, false, true); } /** * 将路径字符串分隔成目录. */ private String[] tokenizePath(String path) { if (path == null) { return new String[0]; } return tokenizeToStringArray(path, PATH_SEPARATOR, false, true); } /** * 创建字符串匹配器. */ private StringMatcher getStringMatcher(String pattern, String str) { if (caseSensitive) { return new AntPathStringMatcher(pattern, str); } else { return new AntPathStringMatcher(pattern.toLowerCase(), str.toLowerCase()); } } /** * 将字符串分隔成数组. */ private static String[] tokenizeToStringArray(String str, String delimiters, boolean trimTokens, boolean ignoreEmptyTokens) { StringTokenizer st = new StringTokenizer(str, delimiters); List<String> tokens = new ArrayList<String>(); while (st.hasMoreTokens()) { String token = st.nextToken(); if (trimTokens) { token = token.trim(); } if (!ignoreEmptyTokens || token.length() > 0) { tokens.add(token); } } return tokens.toArray(new String[tokens.size()]); } } ``` AntPathMatcher类的主要方法是`match`方法,它使用了一些辅助方法来实现Ant风格的路径匹配。 在`match`方法中,首先处理路径中的前缀,然后处理路径中的后缀,最后处理路径中剩余的部分。在处理前缀和后缀时,如果遇到`*`通配符,就跳过当前部分的匹配,继续处理下一部分。在处理剩余部分时,如果遇到`*`通配符,就在路径中查找匹配的部分。如果找到了,就将匹配的部分从路径中删除,继续匹配下一个部分;如果找不到,就返回false。 在实现Ant风格的匹配时,AntPathMatcher使用了一些特殊的字符,例如`?`表示匹配单个字符,`*`表示匹配任意个字符。它还支持路径分隔符的匹配,例如`/`可以匹配路径中的分隔符。 总的来说,AntPathMatcher实现了一种简单而灵活的路径匹配算法,可以满足大多数情况下的路径匹配需求。

相关推荐

最新推荐

recommend-type

基于Vue实现tab栏切换内容不断实时刷新数据功能

在项目开发中遇到这样需求,就是有几个tab栏,每个tab栏对应的ajax请求不一样,内容区域一样,内容为实时刷新数据,实现方法其实很简单的,下面小编给大家带来了基于Vue实现tab栏切换内容不断实时刷新数据功能,需要...
recommend-type

react+ant design实现Table的增、删、改的示例代码

主要介绍了react+ant design实现Table的增、删、改的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
recommend-type

antd组件Upload实现自己上传的实现示例

主要介绍了antd组件Upload实现自己上传的实现示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
recommend-type

使用Vue+Spring Boot实现Excel上传功能

主要介绍了使用Vue+Spring Boot实现Excel上传,需要的朋友可以参考下
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

实现实时数据湖架构:Kafka与Hive集成

![实现实时数据湖架构:Kafka与Hive集成](https://img-blog.csdnimg.cn/img_convert/10eb2e6972b3b6086286fc64c0b3ee41.jpeg) # 1. 实时数据湖架构概述** 实时数据湖是一种现代数据管理架构,它允许企业以低延迟的方式收集、存储和处理大量数据。与传统数据仓库不同,实时数据湖不依赖于预先定义的模式,而是采用灵活的架构,可以处理各种数据类型和格式。这种架构为企业提供了以下优势: - **实时洞察:**实时数据湖允许企业访问最新的数据,从而做出更明智的决策。 - **数据民主化:**实时数据湖使各种利益相关者都可
recommend-type

list根据id查询pid 然后依次获取到所有的子节点数据

可以使用递归的方式来实现根据id查询pid并获取所有子节点数据。具体实现可以参考以下代码: ``` def get_children_nodes(nodes, parent_id): children = [] for node in nodes: if node['pid'] == parent_id: node['children'] = get_children_nodes(nodes, node['id']) children.append(node) return children # 测试数
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。
recommend-type

"互动学习:行动中的多样性与论文攻读经历"

多样性她- 事实上SCI NCES你的时间表ECOLEDO C Tora SC和NCESPOUR l’Ingén学习互动,互动学习以行动为中心的强化学习学会互动,互动学习,以行动为中心的强化学习计算机科学博士论文于2021年9月28日在Villeneuve d'Asq公开支持马修·瑟林评审团主席法布里斯·勒菲弗尔阿维尼翁大学教授论文指导奥利维尔·皮耶昆谷歌研究教授:智囊团论文联合主任菲利普·普雷教授,大学。里尔/CRISTAL/因里亚报告员奥利维耶·西格德索邦大学报告员卢多维奇·德诺耶教授,Facebook /索邦大学审查员越南圣迈IMT Atlantic高级讲师邀请弗洛里安·斯特鲁布博士,Deepmind对于那些及时看到自己错误的人...3谢谢你首先,我要感谢我的两位博士生导师Olivier和Philippe。奥利维尔,"站在巨人的肩膀上"这句话对你来说完全有意义了。从科学上讲,你知道在这篇论文的(许多)错误中,你是我可以依