diff --git a/CHANGELOG.md b/CHANGELOG.md
index 2ed832a7b..d1d3f6df2 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -15,6 +15,7 @@
 * 【db     】     MetaUtil增加getTableMeta重载(issue#2157@Github)
 * 【http   】     增加HttpGlobalConfig.setDecodeUrl(issue#I4U8YQ@Gitee)
 * 【core   】     增加Base58(pr#2162@Github)
+* 【core   】     增加AntPathMatcher(issue#I4T7K5@Gitee)
 
 ### 🐞Bug修复
 * 【cache  】     修复ReentrantCache.toString方法线程不安全问题(issue#2140@Github)
diff --git a/hutool-core/src/main/java/cn/hutool/core/text/AntPathMatcher.java b/hutool-core/src/main/java/cn/hutool/core/text/AntPathMatcher.java
new file mode 100755
index 000000000..aabc3e97b
--- /dev/null
+++ b/hutool-core/src/main/java/cn/hutool/core/text/AntPathMatcher.java
@@ -0,0 +1,942 @@
+package cn.hutool.core.text;
+
+
+import cn.hutool.core.util.StrUtil;
+
+import java.util.ArrayList;
+import java.util.Comparator;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+/**
+ * Ant风格的路径匹配器。
+ * 来自Spring-core和Ant
+ *
+ * 
匹配URL的规则如下:
+ * 
+ * - {@code ?} 匹配单个字符+ *
- {@code *} 匹配0个或多个字符+ *
- {@code **} 0个或多个路径中的目录节点+ *
- {@code {hutool:[a-z]+}} 匹配以"hutool"命名的正则 {@code [a-z]+}+ *
+ *
+ *例子: 
+ * 
+ * - {@code com/t?st.jsp} — 匹配 {@code com/test.jsp} 或 {@code com/tast.jsp} 或 {@code com/txst.jsp}+ *
- {@code com/*.jsp} — 匹配{@code com}目录下全部 {@code .jsp}文件+ *
- {@code com/**/test.jsp} — 匹配{@code com}目录下全部 {@code test.jsp}文件+ *
- {@code cn/hutool/**/*.jsp} — 匹配{@code cn/hutool}路径下全部{@code .jsp} 文件+ *
- {@code org/**/servlet/bla.jsp} — 匹配{@code cn/hutool/servlet/bla.jsp} 或{@code cn/hutool/testing/servlet/bla.jsp} 或 {@code org/servlet/bla.jsp}+ *
- {@code com/{filename:\\w+}.jsp} 匹配 {@code com/test.jsp} 并将 {@code test} 关联到 {@code filename} 变量+ *
+ *
+ *注意: 表达式和路径必须都为绝对路径或都为相对路径。
+ *
+ * @author Alef Arendsen, Juergen Hoeller, Rob Harrop, Arjen Poutsma, Rossen Stoyanchev, Sam Brannen, Vladislav Kisel
+ * @since 5.7.22
+ */
+public class AntPathMatcher {
+
+	/**
+	 * Default path separator: "/".
+	 */
+	public static final String DEFAULT_PATH_SEPARATOR = StrUtil.SLASH;
+
+	private static final int CACHE_TURNOFF_THRESHOLD = 65536;
+
+	private static final Pattern VARIABLE_PATTERN = Pattern.compile("\\{[^/]+?}");
+
+	private static final char[] WILDCARD_CHARS = {'*', '?', '{'};
+
+	private String pathSeparator;
+
+	private PathSeparatorPatternCache pathSeparatorPatternCache;
+
+	private boolean caseSensitive = true;
+
+	private boolean trimTokens = false;
+
+	private volatile Boolean cachePatterns;
+
+	private final Map tokenizedPatternCache = new ConcurrentHashMap<>(256);
+
+	private final Map stringMatcherCache = new ConcurrentHashMap<>(256);
+
+
+	/**
+	 * 使用 {@link #DEFAULT_PATH_SEPARATOR} 作为分隔符构造
+	 */
+	public AntPathMatcher() {
+		this(DEFAULT_PATH_SEPARATOR);
+	}
+
+	/**
+	 * 使用自定义的分隔符构造
+	 *
+	 * @param pathSeparator the path separator to use, must not be {@code null}.
+	 * @since 4.1
+	 */
+	public AntPathMatcher(String pathSeparator) {
+		if (null == pathSeparator) {
+			pathSeparator = DEFAULT_PATH_SEPARATOR;
+		}
+		setPathSeparator(pathSeparator);
+	}
+
+
+	/**
+	 * 设置路径分隔符
+	 *
+	 * @param pathSeparator 分隔符,{@code null}表示使用默认分隔符{@link #DEFAULT_PATH_SEPARATOR}
+	 * @return this
+	 */
+	public AntPathMatcher setPathSeparator(String pathSeparator) {
+		if (null == pathSeparator) {
+			pathSeparator = DEFAULT_PATH_SEPARATOR;
+		}
+		this.pathSeparator = pathSeparator;
+		this.pathSeparatorPatternCache = new PathSeparatorPatternCache(this.pathSeparator);
+		return this;
+	}
+
+	/**
+	 * 设置是否大小写敏感,默认为{@code true}
+	 *
+	 * @param caseSensitive 是否大小写敏感
+	 * @return this
+	 */
+	public AntPathMatcher setCaseSensitive(boolean caseSensitive) {
+		this.caseSensitive = caseSensitive;
+		return this;
+	}
+
+	/**
+	 * 设置是否去除路径节点两边的空白符,默认为{@code false}
+	 *
+	 * @param trimTokens 是否去除路径节点两边的空白符
+	 * @return this
+	 */
+	public AntPathMatcher setTrimTokens(boolean trimTokens) {
+		this.trimTokens = trimTokens;
+		return this;
+	}
+
+	/**
+	 * Specify whether to cache parsed pattern metadata for patterns passed
+	 * into this matcher's {@link #match} method. A value of {@code true}
+	 * activates an unlimited pattern cache; a value of {@code false} turns
+	 * the pattern cache off completely.
+	 * Default is for the cache to be on, but with the variant to automatically
+	 * turn it off when encountering too many patterns to cache at runtime
+	 * (the threshold is 65536), assuming that arbitrary permutations of patterns
+	 * are coming in, with little chance for encountering a recurring pattern.
+	 *
+	 * @param cachePatterns 是否缓存表达式
+	 * @see #getStringMatcher(String)
+	 */
+	public AntPathMatcher setCachePatterns(boolean cachePatterns) {
+		this.cachePatterns = cachePatterns;
+		return this;
+	}
+
+	/**
+	 * 判断给定路径是否是表达式
+	 *
+	 * @param path 路径
+	 * @return 是否为表达式
+	 */
+	public boolean isPattern(String path) {
+		if (path == null) {
+			return false;
+		}
+		boolean uriVar = false;
+		final int length = path.length();
+		char c;
+		for (int i = 0; i < length; i++) {
+			c = path.charAt(i);
+			// 含有通配符
+			if (c == '*' || c == '?') {
+				return true;
+			}
+			if (c == CharPool.DELIM_START) {
+				uriVar = true;
+				continue;
+			}
+			if (c == CharPool.DELIM_END && uriVar) {
+				return true;
+			}
+		}
+		return false;
+	}
+
+	/**
+	 * 给定路径是否匹配表达式
+	 *
+	 * @param pattern 表达式
+	 * @param path 路径
+	 * @return 是否匹配
+	 */
+	public boolean match(String pattern, String path) {
+		return doMatch(pattern, path, true, null);
+	}
+
+	/**
+	 * 前置部分匹配
+	 *
+	 * @param pattern 表达式
+	 * @param path 路径
+	 * @return 是否匹配
+	 */
+	public boolean matchStart(String pattern, String path) {
+		return doMatch(pattern, path, false, null);
+	}
+
+	/**
+	 * 执行匹配,判断给定的{@code path}是否匹配{@code pattern}
+	 *
+	 * @param pattern              表达式
+	 * @param path                 路径
+	 * @param fullMatch            是否全匹配。{@code true} 表示全路径匹配,{@code false}表示只匹配开始
+	 * @param uriTemplateVariables 变量映射
+	 * @return {@code true} 表示提供的 {@code path} 匹配, {@code false} 表示不匹配
+	 */
+	protected boolean doMatch(String pattern, String path, boolean fullMatch, Map uriTemplateVariables) {
+		if (path == null || path.startsWith(this.pathSeparator) != pattern.startsWith(this.pathSeparator)) {
+			return false;
+		}
+
+		final String[] pattDirs = tokenizePattern(pattern);
+		if (fullMatch && this.caseSensitive && false == isPotentialMatch(path, pattDirs)) {
+			return false;
+		}
+
+		final String[] pathDirs = tokenizePath(path);
+		int pattIdxStart = 0;
+		int pattIdxEnd = pattDirs.length - 1;
+		int pathIdxStart = 0;
+		int pathIdxEnd = pathDirs.length - 1;
+
+		// Match all elements up to the first **
+		while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+			String pattDir = pattDirs[pattIdxStart];
+			if ("**".equals(pattDir)) {
+				break;
+			}
+			if (notMatchStrings(pattDir, pathDirs[pathIdxStart], uriTemplateVariables)) {
+				return false;
+			}
+			pattIdxStart++;
+			pathIdxStart++;
+		}
+
+		if (pathIdxStart > pathIdxEnd) {
+			// Path is exhausted, only match if rest of pattern is * or **'s
+			if (pattIdxStart > pattIdxEnd) {
+				return (pattern.endsWith(this.pathSeparator) == path.endsWith(this.pathSeparator));
+			}
+			if (false == fullMatch) {
+				return true;
+			}
+			if (pattIdxStart == pattIdxEnd && pattDirs[pattIdxStart].equals("*") && path.endsWith(this.pathSeparator)) {
+				return true;
+			}
+			for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+				if (false == pattDirs[i].equals("**")) {
+					return false;
+				}
+			}
+			return true;
+		} else if (pattIdxStart > pattIdxEnd) {
+			// String not exhausted, but pattern is. Failure.
+			return false;
+		} else if (false == fullMatch && "**".equals(pattDirs[pattIdxStart])) {
+			// Path start definitely matches due to "**" part in pattern.
+			return true;
+		}
+
+		// up to last '**'
+		while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+			String pattDir = pattDirs[pattIdxEnd];
+			if (pattDir.equals("**")) {
+				break;
+			}
+			if (notMatchStrings(pattDir, pathDirs[pathIdxEnd], uriTemplateVariables)) {
+				return false;
+			}
+			pattIdxEnd--;
+			pathIdxEnd--;
+		}
+		if (pathIdxStart > pathIdxEnd) {
+			// String is exhausted
+			for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+				if (false == pattDirs[i].equals("**")) {
+					return false;
+				}
+			}
+			return true;
+		}
+
+		while (pattIdxStart != pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+			int patIdxTmp = -1;
+			for (int i = pattIdxStart + 1; i <= pattIdxEnd; i++) {
+				if (pattDirs[i].equals("**")) {
+					patIdxTmp = i;
+					break;
+				}
+			}
+			if (patIdxTmp == pattIdxStart + 1) {
+				// '**/**' situation, so skip one
+				pattIdxStart++;
+				continue;
+			}
+			// Find the pattern between padIdxStart & padIdxTmp in str between
+			// strIdxStart & strIdxEnd
+			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 (notMatchStrings(subPat, subStr, uriTemplateVariables)) {
+						continue strLoop;
+					}
+				}
+				foundIdx = pathIdxStart + i;
+				break;
+			}
+
+			if (foundIdx == -1) {
+				return false;
+			}
+
+			pattIdxStart = patIdxTmp;
+			pathIdxStart = foundIdx + patLength;
+		}
+
+		for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+			if (false == pattDirs[i].equals("**")) {
+				return false;
+			}
+		}
+
+		return true;
+	}
+
+	private boolean isPotentialMatch(String path, String[] pattDirs) {
+		if (!this.trimTokens) {
+			int pos = 0;
+			for (String pattDir : pattDirs) {
+				int skipped = skipSeparator(path, pos, this.pathSeparator);
+				pos += skipped;
+				skipped = skipSegment(path, pos, pattDir);
+				if (skipped < pattDir.length()) {
+					return (skipped > 0 || (pattDir.length() > 0 && isWildcardChar(pattDir.charAt(0))));
+				}
+				pos += skipped;
+			}
+		}
+		return true;
+	}
+
+	private int skipSegment(String path, int pos, String prefix) {
+		int skipped = 0;
+		for (int i = 0; i < prefix.length(); i++) {
+			char c = prefix.charAt(i);
+			if (isWildcardChar(c)) {
+				return skipped;
+			}
+			int currPos = pos + skipped;
+			if (currPos >= path.length()) {
+				return 0;
+			}
+			if (c == path.charAt(currPos)) {
+				skipped++;
+			}
+		}
+		return skipped;
+	}
+
+	private int skipSeparator(String path, int pos, String separator) {
+		int skipped = 0;
+		while (path.startsWith(separator, pos + skipped)) {
+			skipped += separator.length();
+		}
+		return skipped;
+	}
+
+	private boolean isWildcardChar(char c) {
+		for (char candidate : WILDCARD_CHARS) {
+			if (c == candidate) {
+				return true;
+			}
+		}
+		return false;
+	}
+
+	/**
+	 * Tokenize the given path pattern into parts, based on this matcher's settings.
+	 * Performs caching based on {@link #setCachePatterns}, delegating to
+	 * {@link #tokenizePath(String)} for the actual tokenization algorithm.
+	 *
+	 * @param pattern the pattern to tokenize
+	 * @return the tokenized pattern parts
+	 */
+	protected String[] tokenizePattern(String pattern) {
+		String[] tokenized = null;
+		Boolean cachePatterns = this.cachePatterns;
+		if (cachePatterns == null || cachePatterns) {
+			tokenized = this.tokenizedPatternCache.get(pattern);
+		}
+		if (tokenized == null) {
+			tokenized = tokenizePath(pattern);
+			if (cachePatterns == null && this.tokenizedPatternCache.size() >= CACHE_TURNOFF_THRESHOLD) {
+				// Try to adapt to the runtime situation that we're encountering:
+				// There are obviously too many different patterns coming in here...
+				// So let's turn off the cache since the patterns are unlikely to be reoccurring.
+				deactivatePatternCache();
+				return tokenized;
+			}
+			if (cachePatterns == null || cachePatterns) {
+				this.tokenizedPatternCache.put(pattern, tokenized);
+			}
+		}
+		return tokenized;
+	}
+
+	private void deactivatePatternCache() {
+		this.cachePatterns = false;
+		this.tokenizedPatternCache.clear();
+		this.stringMatcherCache.clear();
+	}
+
+	/**
+	 * Tokenize the given path into parts, based on this matcher's settings.
+	 *
+	 * @param path the path to tokenize
+	 * @return the tokenized path parts
+	 */
+	protected String[] tokenizePath(String path) {
+		return StrSplitter.splitToArray(path, this.pathSeparator, 0, this.trimTokens, true);
+	}
+
+	/**
+	 * Test whether or not a string matches against a pattern.
+	 *
+	 * @param pattern the pattern to match against (never {@code null})
+	 * @param str     the String which must be matched against the pattern (never {@code null})
+	 * @return {@code true} if the string matches against the pattern, or {@code false} otherwise
+	 */
+	private boolean notMatchStrings(String pattern, String str, Map uriTemplateVariables) {
+		return false == getStringMatcher(pattern).matchStrings(str, uriTemplateVariables);
+	}
+
+	/**
+	 * Build or retrieve an {@link AntPathStringMatcher} for the given pattern.
+	 * The default implementation checks this AntPathMatcher's internal cache
+	 * (see {@link #setCachePatterns}), creating a new AntPathStringMatcher instance
+	 * if no cached copy is found.
+	 * 
When encountering too many patterns to cache at runtime (the threshold is 65536),
+	 * it turns the default cache off, assuming that arbitrary permutations of patterns
+	 * are coming in, with little chance for encountering a recurring pattern.
+	 * 
This method may be overridden to implement a custom cache strategy.
+	 *
+	 * @param pattern the pattern to match against (never {@code null})
+	 * @return a corresponding AntPathStringMatcher (never {@code null})
+	 * @see #setCachePatterns
+	 */
+	protected AntPathStringMatcher getStringMatcher(String pattern) {
+		AntPathStringMatcher matcher = null;
+		Boolean cachePatterns = this.cachePatterns;
+		if (cachePatterns == null || cachePatterns) {
+			matcher = this.stringMatcherCache.get(pattern);
+		}
+		if (matcher == null) {
+			matcher = new AntPathStringMatcher(pattern, this.caseSensitive);
+			if (cachePatterns == null && this.stringMatcherCache.size() >= CACHE_TURNOFF_THRESHOLD) {
+				// Try to adapt to the runtime situation that we're encountering:
+				// There are obviously too many different patterns coming in here...
+				// So let's turn off the cache since the patterns are unlikely to be reoccurring.
+				deactivatePatternCache();
+				return matcher;
+			}
+			if (cachePatterns == null || cachePatterns) {
+				this.stringMatcherCache.put(pattern, matcher);
+			}
+		}
+		return matcher;
+	}
+
+	/**
+	 * Given a pattern and a full path, determine the pattern-mapped part. 
For example: 
+	 * - '{@code /docs/cvs/commit.html}' and '{@code /docs/cvs/commit.html} → ''+	 *
- '{@code /docs/*}' and '{@code /docs/cvs/commit} → '{@code cvs/commit}'+	 *
- '{@code /docs/cvs/*.html}' and '{@code /docs/cvs/commit.html} → '{@code commit.html}'+	 *
- '{@code /docs/**}' and '{@code /docs/cvs/commit} → '{@code cvs/commit}'+	 *
- '{@code /docs/**\/*.html}' and '{@code /docs/cvs/commit.html} → '{@code cvs/commit.html}'+	 *
- '{@code /*.html}' and '{@code /docs/cvs/commit.html} → '{@code docs/cvs/commit.html}'+	 *
- '{@code *.html}' and '{@code /docs/cvs/commit.html} → '{@code /docs/cvs/commit.html}'+	 *
- '{@code *}' and '{@code /docs/cvs/commit.html} → '{@code /docs/cvs/commit.html}'
+	 *Assumes that {@link #match} returns {@code true} for '{@code pattern}' and '{@code path}', but
+	 * does not enforce this.
+	 *
+	 * @param pattern 表达式
+	 * @param path    路径
+	 * @return 表达式匹配到的部分
+	 */
+	public String extractPathWithinPattern(String pattern, String path) {
+		String[] patternParts = tokenizePath(pattern);
+		String[] pathParts = tokenizePath(path);
+		StringBuilder builder = new StringBuilder();
+		boolean pathStarted = false;
+
+		for (int segment = 0; segment < patternParts.length; segment++) {
+			String patternPart = patternParts[segment];
+			if (patternPart.indexOf('*') > -1 || patternPart.indexOf('?') > -1) {
+				for (; segment < pathParts.length; segment++) {
+					if (pathStarted || (segment == 0 && !pattern.startsWith(this.pathSeparator))) {
+						builder.append(this.pathSeparator);
+					}
+					builder.append(pathParts[segment]);
+					pathStarted = true;
+				}
+			}
+		}
+
+		return builder.toString();
+	}
+
+	public Map extractUriTemplateVariables(String pattern, String path) {
+		Map variables = new LinkedHashMap<>();
+		boolean result = doMatch(pattern, path, true, variables);
+		if (!result) {
+			throw new IllegalStateException("Pattern \"" + pattern + "\" is not a match for \"" + path + "\"");
+		}
+		return variables;
+	}
+
+	/**
+	 * Combine two patterns into a new pattern.
+	 * This implementation simply concatenates the two patterns, unless
+	 * the first pattern contains a file extension match (e.g., {@code *.html}).
+	 * In that case, the second pattern will be merged into the first. Otherwise,
+	 * an {@code IllegalArgumentException} will be thrown.
+	 * 
Examples
+	 * 
+	 * | Pattern 1 | Pattern 2 | Result | 
|---|
+	 * | {@code null} | {@code null} |  | 
+	 * | /hotels | {@code null} | /hotels | 
+	 * | {@code null} | /hotels | /hotels | 
+	 * | /hotels | /bookings | /hotels/bookings | 
+	 * | /hotels | bookings | /hotels/bookings | 
+	 * | /hotels/* | /bookings | /hotels/bookings | 
+	 * | /hotels/** | /bookings | /hotels/**/bookings | 
+	 * | /hotels | {hotel} | /hotels/{hotel} | 
+	 * | /hotels/* | {hotel} | /hotels/{hotel} | 
+	 * | /hotels/** | {hotel} | /hotels/**/{hotel} | 
+	 * | /*.html | /hotels.html | /hotels.html | 
+	 * | /*.html | /hotels | /hotels.html | 
+	 * | /*.html | /*.txt | {@code IllegalArgumentException} | 
+	 * 
+	 *
+	 * @param pattern1 the first pattern
+	 * @param pattern2 the second pattern
+	 * @return the combination of the two patterns
+	 * @throws IllegalArgumentException if the two patterns cannot be combined
+	 */
+	public String combine(String pattern1, String pattern2) {
+		if (StrUtil.isEmpty(pattern1) && StrUtil.isEmpty(pattern2)) {
+			return StrUtil.EMPTY;
+		}
+		if (StrUtil.isEmpty(pattern1)) {
+			return pattern2;
+		}
+		if (StrUtil.isEmpty(pattern2)) {
+			return pattern1;
+		}
+
+		boolean pattern1ContainsUriVar = (pattern1.indexOf('{') != -1);
+		if (!pattern1.equals(pattern2) && !pattern1ContainsUriVar && match(pattern1, pattern2)) {
+			// /* + /hotel -> /hotel ; "/*.*" + "/*.html" -> /*.html
+			// However /user + /user -> /usr/user ; /{foo} + /bar -> /{foo}/bar
+			return pattern2;
+		}
+
+		// /hotels/* + /booking -> /hotels/booking
+		// /hotels/* + booking -> /hotels/booking
+		if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnWildCard())) {
+			return concat(pattern1.substring(0, pattern1.length() - 2), pattern2);
+		}
+
+		// /hotels/** + /booking -> /hotels/**/booking
+		// /hotels/** + booking -> /hotels/**/booking
+		if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnDoubleWildCard())) {
+			return concat(pattern1, pattern2);
+		}
+
+		int starDotPos1 = pattern1.indexOf("*.");
+		if (pattern1ContainsUriVar || starDotPos1 == -1 || this.pathSeparator.equals(".")) {
+			// simply concatenate the two patterns
+			return concat(pattern1, pattern2);
+		}
+
+		String ext1 = pattern1.substring(starDotPos1 + 1);
+		int dotPos2 = pattern2.indexOf('.');
+		String file2 = (dotPos2 == -1 ? pattern2 : pattern2.substring(0, dotPos2));
+		String ext2 = (dotPos2 == -1 ? "" : pattern2.substring(dotPos2));
+		boolean ext1All = (ext1.equals(".*") || ext1.isEmpty());
+		boolean ext2All = (ext2.equals(".*") || ext2.isEmpty());
+		if (!ext1All && !ext2All) {
+			throw new IllegalArgumentException("Cannot combine patterns: " + pattern1 + " vs " + pattern2);
+		}
+		String ext = (ext1All ? ext2 : ext1);
+		return file2 + ext;
+	}
+
+	private String concat(String path1, String path2) {
+		boolean path1EndsWithSeparator = path1.endsWith(this.pathSeparator);
+		boolean path2StartsWithSeparator = path2.startsWith(this.pathSeparator);
+
+		if (path1EndsWithSeparator && path2StartsWithSeparator) {
+			return path1 + path2.substring(1);
+		} else if (path1EndsWithSeparator || path2StartsWithSeparator) {
+			return path1 + path2;
+		} else {
+			return path1 + this.pathSeparator + path2;
+		}
+	}
+
+	/**
+	 * Given a full path, returns a {@link Comparator} suitable for sorting patterns in order of
+	 * explicitness.
+	 * This {@code Comparator} will {@linkplain List#sort(Comparator) sort}
+	 * a list so that more specific patterns (without URI templates or wild cards) come before
+	 * generic patterns. So given a list with the following patterns, the returned comparator
+	 * will sort this list so that the order will be as indicated.
+	 * 
+	 * - {@code /hotels/new}+	 *
- {@code /hotels/{hotel}}+	 *
- {@code /hotels/*}+	 *
+	 *The full path given as parameter is used to test for exact matches. So when the given path
+	 * is {@code /hotels/2}, the pattern {@code /hotels/2} will be sorted before {@code /hotels/1}.
+	 *
+	 * @param path the full path to use for comparison
+	 * @return a comparator capable of sorting patterns in order of explicitness
+	 */
+	public Comparator getPatternComparator(String path) {
+		return new AntPatternComparator(path);
+	}
+
+
+	/**
+	 * Tests whether or not a string matches against a pattern via a {@link Pattern}.
+	 * The pattern may contain special characters: '*' means zero or more characters; '?' means one and
+	 * only one character; '{' and '}' indicate a URI template pattern. For example /users/{user}.
+	 */
+	protected static class AntPathStringMatcher {
+
+		private static final Pattern GLOB_PATTERN = Pattern.compile("\\?|\\*|\\{((?:\\{[^/]+?}|[^/{}]|\\\\[{}])+?)}");
+
+		private static final String DEFAULT_VARIABLE_PATTERN = "((?s).*)";
+
+		private final String rawPattern;
+
+		private final boolean caseSensitive;
+
+		private final boolean exactMatch;
+
+		private final Pattern pattern;
+
+		private final List variableNames = new ArrayList<>();
+
+		public AntPathStringMatcher(String pattern, boolean caseSensitive) {
+			this.rawPattern = pattern;
+			this.caseSensitive = caseSensitive;
+			StringBuilder patternBuilder = new StringBuilder();
+			Matcher matcher = GLOB_PATTERN.matcher(pattern);
+			int end = 0;
+			while (matcher.find()) {
+				patternBuilder.append(quote(pattern, end, matcher.start()));
+				String match = matcher.group();
+				if ("?".equals(match)) {
+					patternBuilder.append('.');
+				} else if ("*".equals(match)) {
+					patternBuilder.append(".*");
+				} else if (match.startsWith("{") && match.endsWith("}")) {
+					int colonIdx = match.indexOf(':');
+					if (colonIdx == -1) {
+						patternBuilder.append(DEFAULT_VARIABLE_PATTERN);
+						this.variableNames.add(matcher.group(1));
+					} else {
+						String variablePattern = match.substring(colonIdx + 1, match.length() - 1);
+						patternBuilder.append('(');
+						patternBuilder.append(variablePattern);
+						patternBuilder.append(')');
+						String variableName = match.substring(1, colonIdx);
+						this.variableNames.add(variableName);
+					}
+				}
+				end = matcher.end();
+			}
+			// No glob pattern was found, this is an exact String match
+			if (end == 0) {
+				this.exactMatch = true;
+				this.pattern = null;
+			} else {
+				this.exactMatch = false;
+				patternBuilder.append(quote(pattern, end, pattern.length()));
+				this.pattern = (this.caseSensitive ? Pattern.compile(patternBuilder.toString()) :
+						Pattern.compile(patternBuilder.toString(), Pattern.CASE_INSENSITIVE));
+			}
+		}
+
+		private String quote(String s, int start, int end) {
+			if (start == end) {
+				return "";
+			}
+			return Pattern.quote(s.substring(start, end));
+		}
+
+		/**
+		 * Main entry point.
+		 *
+		 * @return {@code true} if the string matches against the pattern, or {@code false} otherwise.
+		 */
+		public boolean matchStrings(String str, Map uriTemplateVariables) {
+			if (this.exactMatch) {
+				return this.caseSensitive ? this.rawPattern.equals(str) : this.rawPattern.equalsIgnoreCase(str);
+			} else if (this.pattern != null) {
+				Matcher matcher = this.pattern.matcher(str);
+				if (matcher.matches()) {
+					if (uriTemplateVariables != null) {
+						if (this.variableNames.size() != matcher.groupCount()) {
+							throw new IllegalArgumentException("The number of capturing groups in the pattern segment " +
+									this.pattern + " does not match the number of URI template variables it defines, " +
+									"which can occur if capturing groups are used in a URI template regex. " +
+									"Use non-capturing groups instead.");
+						}
+						for (int i = 1; i <= matcher.groupCount(); i++) {
+							String name = this.variableNames.get(i - 1);
+							if (name.startsWith("*")) {
+								throw new IllegalArgumentException("Capturing patterns (" + name + ") are not " +
+										"supported by the AntPathMatcher. Use the PathPatternParser instead.");
+							}
+							String value = matcher.group(i);
+							uriTemplateVariables.put(name, value);
+						}
+					}
+					return true;
+				}
+			}
+			return false;
+		}
+
+	}
+
+
+	/**
+	 * The default {@link Comparator} implementation returned by
+	 * {@link #getPatternComparator(String)}.
+	 * In order, the most "generic" pattern is determined by the following:
+	 * 
+	 * - if it's null or a capture all pattern (i.e. it is equal to "/**")+	 *
- if the other pattern is an actual match+	 *
- if it's a catch-all pattern (i.e. it ends with "**"+	 *
- if it's got more "*" than the other pattern+	 *
- if it's got more "{foo}" than the other pattern+	 *
- if it's shorter than the other pattern+	 *
+	 */
+	protected static class AntPatternComparator implements Comparator {
+
+		private final String path;
+
+		public AntPatternComparator(String path) {
+			this.path = path;
+		}
+
+		/**
+		 * Compare two patterns to determine which should match first, i.e. which
+		 * is the most specific regarding the current path.
+		 *
+		 * @param pattern1 表达式1
+		 * @param pattern2 表达式2
+		 * @return a negative integer, zero, or a positive integer as pattern1 is
+		 * more specific, equally specific, or less specific than pattern2.
+		 */
+		@Override
+		public int compare(String pattern1, String pattern2) {
+			PatternInfo info1 = new PatternInfo(pattern1);
+			PatternInfo info2 = new PatternInfo(pattern2);
+
+			if (info1.isLeastSpecific() && info2.isLeastSpecific()) {
+				return 0;
+			} else if (info1.isLeastSpecific()) {
+				return 1;
+			} else if (info2.isLeastSpecific()) {
+				return -1;
+			}
+
+			boolean pattern1EqualsPath = pattern1.equals(this.path);
+			boolean pattern2EqualsPath = pattern2.equals(this.path);
+			if (pattern1EqualsPath && pattern2EqualsPath) {
+				return 0;
+			} else if (pattern1EqualsPath) {
+				return -1;
+			} else if (pattern2EqualsPath) {
+				return 1;
+			}
+
+			if (info1.isPrefixPattern() && info2.isPrefixPattern()) {
+				return info2.getLength() - info1.getLength();
+			} else if (info1.isPrefixPattern() && info2.getDoubleWildcards() == 0) {
+				return 1;
+			} else if (info2.isPrefixPattern() && info1.getDoubleWildcards() == 0) {
+				return -1;
+			}
+
+			if (info1.getTotalCount() != info2.getTotalCount()) {
+				return info1.getTotalCount() - info2.getTotalCount();
+			}
+
+			if (info1.getLength() != info2.getLength()) {
+				return info2.getLength() - info1.getLength();
+			}
+
+			if (info1.getSingleWildcards() < info2.getSingleWildcards()) {
+				return -1;
+			} else if (info2.getSingleWildcards() < info1.getSingleWildcards()) {
+				return 1;
+			}
+
+			if (info1.getUriVars() < info2.getUriVars()) {
+				return -1;
+			} else if (info2.getUriVars() < info1.getUriVars()) {
+				return 1;
+			}
+
+			return 0;
+		}
+
+
+		/**
+		 * Value class that holds information about the pattern, e.g. number of
+		 * occurrences of "*", "**", and "{" pattern elements.
+		 */
+		private static class PatternInfo {
+
+			private final String pattern;
+			private int uriVars;
+			private int singleWildcards;
+			private int doubleWildcards;
+			private boolean catchAllPattern;
+			private boolean prefixPattern;
+			private Integer length;
+
+			public PatternInfo(String pattern) {
+				this.pattern = pattern;
+				if (this.pattern != null) {
+					initCounters();
+					this.catchAllPattern = this.pattern.equals("/**");
+					this.prefixPattern = !this.catchAllPattern && this.pattern.endsWith("/**");
+				}
+				if (this.uriVars == 0) {
+					this.length = (this.pattern != null ? this.pattern.length() : 0);
+				}
+			}
+
+			protected void initCounters() {
+				int pos = 0;
+				if (this.pattern != null) {
+					while (pos < this.pattern.length()) {
+						if (this.pattern.charAt(pos) == '{') {
+							this.uriVars++;
+							pos++;
+						} else if (this.pattern.charAt(pos) == '*') {
+							if (pos + 1 < this.pattern.length() && this.pattern.charAt(pos + 1) == '*') {
+								this.doubleWildcards++;
+								pos += 2;
+							} else if (pos > 0 && !this.pattern.substring(pos - 1).equals(".*")) {
+								this.singleWildcards++;
+								pos++;
+							} else {
+								pos++;
+							}
+						} else {
+							pos++;
+						}
+					}
+				}
+			}
+
+			public int getUriVars() {
+				return this.uriVars;
+			}
+
+			public int getSingleWildcards() {
+				return this.singleWildcards;
+			}
+
+			public int getDoubleWildcards() {
+				return this.doubleWildcards;
+			}
+
+			public boolean isLeastSpecific() {
+				return (this.pattern == null || this.catchAllPattern);
+			}
+
+			public boolean isPrefixPattern() {
+				return this.prefixPattern;
+			}
+
+			public int getTotalCount() {
+				return this.uriVars + this.singleWildcards + (2 * this.doubleWildcards);
+			}
+
+			/**
+			 * Returns the length of the given pattern, where template variables are considered to be 1 long.
+			 *
+			 * @return 长度
+			 */
+			public int getLength() {
+				if (this.length == null) {
+					this.length = (this.pattern != null ?
+							VARIABLE_PATTERN.matcher(this.pattern).replaceAll("#").length() : 0);
+				}
+				return this.length;
+			}
+		}
+	}
+
+
+	/**
+	 * A simple cache for patterns that depend on the configured path separator.
+	 */
+	private static class PathSeparatorPatternCache {
+
+		private final String endsOnWildCard;
+
+		private final String endsOnDoubleWildCard;
+
+		public PathSeparatorPatternCache(String pathSeparator) {
+			this.endsOnWildCard = pathSeparator + "*";
+			this.endsOnDoubleWildCard = pathSeparator + "**";
+		}
+
+		public String getEndsOnWildCard() {
+			return this.endsOnWildCard;
+		}
+
+		public String getEndsOnDoubleWildCard() {
+			return this.endsOnDoubleWildCard;
+		}
+	}
+
+}
diff --git a/hutool-core/src/test/java/cn/hutool/core/text/AntPathMatcherTest.java b/hutool-core/src/test/java/cn/hutool/core/text/AntPathMatcherTest.java
new file mode 100755
index 000000000..3b1db3b67
--- /dev/null
+++ b/hutool-core/src/test/java/cn/hutool/core/text/AntPathMatcherTest.java
@@ -0,0 +1,115 @@
+package cn.hutool.core.text;
+
+import org.junit.Assert;
+import org.junit.Test;
+
+import java.util.HashMap;
+
+public class AntPathMatcherTest {
+
+	@Test
+	public void matchesTest() {
+		AntPathMatcher antPathMatcher = new AntPathMatcher();
+		boolean matched = antPathMatcher.match("/api/org/organization/{orgId}", "/api/org/organization/999");
+		Assert.assertTrue(matched);
+	}
+
+	@Test
+	public void matchesTest2() {
+		AntPathMatcher antPathMatcher = new AntPathMatcher();
+
+		String pattern = "/**/*.xml*";
+		String path = "/WEB-INF/web.xml";
+		boolean isMatched = antPathMatcher.match(pattern, path);
+		Assert.assertTrue(isMatched);
+
+		pattern = "org/codelabor/*/**/*Service";
+		path = "org/codelabor/example/HelloWorldService";
+		isMatched = antPathMatcher.match(pattern, path);
+		Assert.assertTrue(isMatched);
+
+		pattern = "org/codelabor/*/**/*Service?";
+		path = "org/codelabor/example/HelloWorldServices";
+		isMatched = antPathMatcher.match(pattern, path);
+		Assert.assertTrue(isMatched);
+	}
+
+	@Test
+	public void matchesTest3(){
+		AntPathMatcher pathMatcher = new AntPathMatcher();
+		pathMatcher.setCachePatterns(true);
+		pathMatcher.setCaseSensitive(true);
+		pathMatcher.setPathSeparator("/");
+		pathMatcher.setTrimTokens(true);
+
+		Assert.assertTrue(pathMatcher.match("a", "a"));
+		Assert.assertTrue(pathMatcher.match("a*", "ab"));
+		Assert.assertTrue(pathMatcher.match("a*/**/a", "ab/asdsa/a"));
+		Assert.assertTrue(pathMatcher.match("a*/**/a", "ab/asdsa/asdasd/a"));
+
+		Assert.assertTrue(pathMatcher.match("*", "a"));
+		Assert.assertTrue(pathMatcher.match("*/*", "a/a"));
+	}
+
+	/**
+	 * AntPathMatcher默认路径分隔符为“/”,而在匹配文件路径时,需要注意Windows下路径分隔符为“\”,Linux下为“/”。靠谱写法如下两种方式:
+	 * AntPathMatcher matcher = new AntPathMatcher(File.separator);
+	 * AntPathMatcher matcher = new AntPathMatcher(System.getProperty("file.separator"));
+	 */
+	@Test
+	public void matchesTest4() {
+		AntPathMatcher pathMatcher = new AntPathMatcher();
+
+		// 精确匹配
+		Assert.assertTrue(pathMatcher.match("/test", "/test"));
+		Assert.assertFalse(pathMatcher.match("test", "/test"));
+
+		//测试通配符?
+		Assert.assertTrue(pathMatcher.match("t?st", "test"));
+		Assert.assertTrue(pathMatcher.match("te??", "test"));
+		Assert.assertFalse(pathMatcher.match("tes?", "tes"));
+		Assert.assertFalse(pathMatcher.match("tes?", "testt"));
+
+		//测试通配符*
+		Assert.assertTrue(pathMatcher.match("*", "test"));
+		Assert.assertTrue(pathMatcher.match("test*", "test"));
+		Assert.assertTrue(pathMatcher.match("test/*", "test/Test"));
+		Assert.assertTrue(pathMatcher.match("*.*", "test."));
+		Assert.assertTrue(pathMatcher.match("*.*", "test.test.test"));
+		Assert.assertFalse(pathMatcher.match("test*", "test/")); //注意这里是false 因为路径不能用*匹配
+		Assert.assertFalse(pathMatcher.match("test*", "test/t")); //这同理
+		Assert.assertFalse(pathMatcher.match("test*aaa", "testblaaab")); //这个是false 因为最后一个b无法匹配了 前面都是能匹配成功的
+
+		//测试通配符** 匹配多级URL
+		Assert.assertTrue(pathMatcher.match("/*/**", "/testing/testing"));
+		Assert.assertTrue(pathMatcher.match("/**/*", "/testing/testing"));
+		Assert.assertTrue(pathMatcher.match("/bla/**/bla", "/bla/testing/testing/bla/bla")); //这里也是true哦
+		Assert.assertFalse(pathMatcher.match("/bla*bla/test", "/blaXXXbl/test"));
+
+		Assert.assertFalse(pathMatcher.match("/????", "/bala/bla"));
+		Assert.assertFalse(pathMatcher.match("/**/*bla", "/bla/bla/bla/bbb"));
+
+		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing/"));
+		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/*", "/XXXblaXXXX/testing/testing/bla/testing"));
+		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing"));
+		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing.jpg"));
+		Assert.assertTrue(pathMatcher.match("/foo/bar/**", "/foo/bar"));
+
+		//这个需要特别注意:{}里面的相当于Spring MVC里接受一个参数一样,所以任何东西都会匹配的
+		Assert.assertTrue(pathMatcher.match("/{bla}.*", "/testing.html"));
+		Assert.assertFalse(pathMatcher.match("/{bla}.htm", "/testing.html")); //这样就是false了
+	}
+
+	/**
+	 * 测试 URI 模板变量提取
+	 */
+	@Test
+	public void testExtractUriTemplateVariables() {
+		AntPathMatcher antPathMatcher = new AntPathMatcher();
+		HashMap map = (HashMap) antPathMatcher.extractUriTemplateVariables("/api/org/organization/{orgId}",
+				"/api/org" +
+						"/organization" +
+						"/999");
+		Assert.assertEquals(1, map.size());
+	}
+}
diff --git a/hutool-core/src/test/java/cn/hutool/core/util/ReflectUtilTest.java b/hutool-core/src/test/java/cn/hutool/core/util/ReflectUtilTest.java
index 70f6dd6e2..9a90c8fe3 100644
--- a/hutool-core/src/test/java/cn/hutool/core/util/ReflectUtilTest.java
+++ b/hutool-core/src/test/java/cn/hutool/core/util/ReflectUtilTest.java
@@ -1,8 +1,13 @@
 package cn.hutool.core.util;
 
+import cn.hutool.core.date.DateUtil;
+import cn.hutool.core.date.TimeInterval;
+import cn.hutool.core.lang.Console;
 import cn.hutool.core.lang.test.bean.ExamInfoDict;
 import cn.hutool.core.util.ClassUtilTest.TestSubClass;
+import lombok.Data;
 import org.junit.Assert;
+import org.junit.Ignore;
 import org.junit.Test;
 
 import java.lang.reflect.Field;
@@ -91,33 +96,77 @@ public class ReflectUtilTest {
 
 	@Test
 	public void noneStaticInnerClassTest() {
-		final TestAClass testAClass = ReflectUtil.newInstanceIfPossible(TestAClass.class);
+		final NoneStaticClass testAClass = ReflectUtil.newInstanceIfPossible(NoneStaticClass.class);
 		Assert.assertNotNull(testAClass);
 		Assert.assertEquals(2, testAClass.getA());
 	}
 
+	@Data
 	static class TestClass {
 		private int a;
-
-		public int getA() {
-			return a;
-		}
-
-		public void setA(int a) {
-			this.a = a;
-		}
 	}
 
+	@Data
 	@SuppressWarnings("InnerClassMayBeStatic")
-	class TestAClass {
+	class NoneStaticClass {
 		private int a = 2;
+	}
 
-		public int getA() {
-			return a;
+	@Test
+	@Ignore
+	public void getMethodBenchTest(){
+		// 预热
+		getMethod(TestBenchClass.class, false, "getH");
+
+		final TimeInterval timer = DateUtil.timer();
+		timer.start();
+		for (int i = 0; i < 100000000; i++) {
+			ReflectUtil.getMethod(TestBenchClass.class, false, "getH");
+		}
+		Console.log(timer.interval());
+
+		timer.restart();
+		for (int i = 0; i < 100000000; i++) {
+			getMethod(TestBenchClass.class, false, "getH");
+		}
+		Console.log(timer.interval());
+	}
+
+	@Data
+	static class TestBenchClass {
+		private int a;
+		private String b;
+		private String c;
+		private String d;
+		private String e;
+		private String f;
+		private String g;
+		private String h;
+		private String i;
+		private String j;
+		private String k;
+		private String l;
+		private String m;
+		private String n;
+	}
+
+	public static Method getMethod(Class> clazz, boolean ignoreCase, String methodName, Class>... paramTypes) throws SecurityException {
+		if (null == clazz || StrUtil.isBlank(methodName)) {
+			return null;
 		}
 
-		public void setA(int a) {
-			this.a = a;
+		Method res = null;
+		final Method[] methods = ReflectUtil.getMethods(clazz);
+		if (ArrayUtil.isNotEmpty(methods)) {
+			for (Method method : methods) {
+				if (StrUtil.equals(methodName, method.getName(), ignoreCase)
+						&& ClassUtil.isAllAssignableFrom(method.getParameterTypes(), paramTypes)
+						&& (res == null
+						|| res.getReturnType().isAssignableFrom(method.getReturnType()))) {
+					res = method;
+				}
+			}
 		}
+		return res;
 	}
 }
diff --git a/hutool-db/pom.xml b/hutool-db/pom.xml
index c7347f002..fdffcbe3a 100644
--- a/hutool-db/pom.xml
+++ b/hutool-db/pom.xml
@@ -103,6 +103,14 @@
 			${mongo.version}
 			true
 		
+		
 		
 		
 			redis.clients
@@ -155,13 +163,13 @@
 		
 			com.microsoft.sqlserver
 			mssql-jdbc
-			9.4.1.jre8
+			10.2.0.jre8
 			test
 		
 		
 			org.slf4j
 			slf4j-simple
-			1.7.32
+			1.7.36
 			test