generated from mingliqiye/lib-tem
	Compare commits
	
		
			3 Commits
		
	
	
		
			d991b4077b
			...
			01bfb052d0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 01bfb052d0 | |||
| 1fab0b02be | |||
| fb4e103da8 | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -44,4 +44,4 @@ log | ||||
| .idea | ||||
| node_modules | ||||
| *lock* | ||||
| 
 | ||||
| .kotlin | ||||
|  | ||||
| @ -16,7 +16,7 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils | ||||
|  * CurrentFile build.gradle.kts | ||||
|  * LastUpdate 2025-09-12 14:06:21 | ||||
|  * LastUpdate 2025-09-13 10:11:22 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| @ -59,10 +59,6 @@ sourceSets { | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| configurations { | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| java { | ||||
|     withSourcesJar() | ||||
|     toolchain.languageVersion.set(JavaLanguageVersion.of(8)) | ||||
| @ -81,8 +77,11 @@ dependencies { | ||||
|     implementation("com.github.f4b6a3:uuid-creator:6.1.0") | ||||
|     implementation("org.mindrot:jbcrypt:0.4") | ||||
|     implementation("org.jetbrains:annotations:24.0.0") | ||||
|     implementation("net.java.dev.jna:jna:5.17.0") | ||||
|     compileOnly("net.java.dev.jna:jna:5.17.0") | ||||
|     implementation("jakarta.annotation:jakarta.annotation-api:2.1.1") | ||||
|     implementation("org.slf4j:slf4j-api:2.0.17") | ||||
|     implementation("com.mingliqiye.utils.jna:WinKernel32Api:1.0.1") | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| @ -90,26 +89,10 @@ tasks.withType<JavaCompile> { | ||||
|     options.encoding = "UTF-8" | ||||
| } | ||||
| 
 | ||||
| tasks.register<Jar>("javaDocJar") { | ||||
|     group = "build" | ||||
|     archiveClassifier.set("javadoc") | ||||
|     dependsOn(tasks.dokkaGfm, tasks.dokkaHtml, tasks.dokkaJavadoc, tasks.dokkaJekyll) | ||||
|     from(buildDir.resolve("dokka/javadoc")) | ||||
| } | ||||
| tasks.register<Jar>("kotlinDocJar") { | ||||
|     group = "build" | ||||
|     archiveClassifier.set("kotlindoc") | ||||
|     dependsOn(tasks.dokkaHtml) | ||||
|     from(buildDir.resolve("dokka/html")) | ||||
| } | ||||
| tasks.build { | ||||
|     dependsOn("javaDocJar", "kotlinDocJar") | ||||
| } | ||||
| 
 | ||||
| tasks.withType<JavaExec>().configureEach { | ||||
|     jvmArgs = listOf( | ||||
|         "-Dfile.encoding=UTF-8", | ||||
|         "-Dsun.stdout.encoding=UTF-8", | ||||
|         "-Dsun.stderr.encoding=UTF-8" | ||||
|         "-Dfile.encoding=UTF-8", "-Dsun.stdout.encoding=UTF-8", "-Dsun.stderr.encoding=UTF-8" | ||||
|     ) | ||||
| } | ||||
| 
 | ||||
| @ -142,6 +125,7 @@ tasks.withType<org.gradle.jvm.tasks.Jar> { | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| val isJdk8Build = project.findProperty("buildForJdk8") == "true" | ||||
| 
 | ||||
| repositories { | ||||
|     maven { | ||||
| @ -150,6 +134,18 @@ repositories { | ||||
|     mavenCentral() | ||||
| } | ||||
| 
 | ||||
| tasks.register<Jar>("javaDocJar") { | ||||
|     group = "build" | ||||
|     archiveClassifier.set("javadoc") | ||||
|     dependsOn(tasks.dokkaJavadoc) | ||||
|     from(buildDir.resolve("dokka/javadoc")) | ||||
| } | ||||
| tasks.register<Jar>("kotlinDocJar") { | ||||
|     group = "build" | ||||
|     archiveClassifier.set("kotlindoc") | ||||
|     dependsOn(tasks.dokkaHtml) | ||||
|     from(buildDir.resolve("dokka/html")) | ||||
| } | ||||
| publishing { | ||||
|     repositories { | ||||
|         maven { | ||||
| @ -160,39 +156,34 @@ publishing { | ||||
|     publications { | ||||
|         create<MavenPublication>("mavenJava") { | ||||
|             from(components["java"]) | ||||
|             groupId = GROUPSID | ||||
|             artifactId = ARTIFACTID | ||||
|             version = VERSIONS | ||||
|             artifact(tasks.named("javaDocJar")) | ||||
|             artifact(tasks.named("kotlinDocJar")) | ||||
|             pom { | ||||
|                 name.set(project.name) | ||||
|                 url.set("https://github.com/minglipro/mingli-utils") | ||||
| 
 | ||||
|                 licenses { | ||||
|                     license { | ||||
|                         name.set("Apache-2.0") | ||||
|                         url.set("https://opensource.org/licenses/Apache-2.0") | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             artifactId = ARTIFACTID | ||||
|             java.toolchain.languageVersion.set(JavaLanguageVersion.of(8)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| tasks.build { | ||||
|     dependsOn("javaDocJar", "kotlinDocJar") | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| tasks.processResources { | ||||
|     duplicatesStrategy = DuplicatesStrategy.EXCLUDE | ||||
|     outputs.upToDateWhen { false } | ||||
|     filesMatching("META-INF/meta-data") { | ||||
|         expand( | ||||
|             mapOf( | ||||
|                 "buildTime" to LocalDateTime.now() | ||||
|                     .format( | ||||
|                         DateTimeFormatter.ofPattern( | ||||
|                             "yyyy-MM-dd HH:mm:ss.SSSSSSS" | ||||
|                         ) | ||||
|             project.properties + mapOf( | ||||
|                 "buildTime" to LocalDateTime.now().format( | ||||
|                     DateTimeFormatter.ofPattern( | ||||
|                         "yyyy-MM-dd HH:mm:ss.SSSSSSS" | ||||
|                     ) | ||||
|             ) + project.properties | ||||
| 
 | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -16,10 +16,10 @@ | ||||
| # ProjectName mingli-utils | ||||
| # ModuleName mingli-utils | ||||
| # CurrentFile gradle.properties | ||||
| # LastUpdate 2025-09-12 14:10:24 | ||||
| # LastUpdate 2025-09-13 10:14:51 | ||||
| # UpdateUser MingLiPro | ||||
| # | ||||
| JDKVERSIONS=1.8 | ||||
| GROUPSID=com.mingliqiye.utils | ||||
| ARTIFACTID=mingli-utils | ||||
| VERSIONS=3.2.7 | ||||
| VERSIONS=3.3.1 | ||||
|  | ||||
							
								
								
									
										57
									
								
								jdk8/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								jdk8/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.jdk8 | ||||
|  * CurrentFile build.gradle.kts | ||||
|  * LastUpdate 2025-09-14 18:19:04 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| plugins { | ||||
|     id("java-library") | ||||
|     id("maven-publish") | ||||
| } | ||||
| val GROUPSID = project.properties["GROUPSID"] as String | ||||
| val VERSIONS = project.properties["VERSIONS"] as String | ||||
| val ARTIFACTID = project.properties["ARTIFACTID"] as String | ||||
| version = VERSIONS | ||||
| group = GROUPSID | ||||
| 
 | ||||
| base.archivesName.set(ARTIFACTID) | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| publishing { | ||||
|     repositories { | ||||
|         maven { | ||||
|             name = "MavenRepositoryRaw" | ||||
|             url = uri("C:/data/git/maven-repository-raw") | ||||
|         } | ||||
|     } | ||||
|     publications { | ||||
|         create<MavenPublication>("mavenJava") { | ||||
|             from(components["java"]) | ||||
|             artifactId = "$ARTIFACTID-win-jdk8" | ||||
|             groupId = GROUPSID | ||||
|             version = VERSIONS | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| java.toolchain.languageVersion.set(JavaLanguageVersion.of(8)) | ||||
| dependencies { | ||||
|     api(rootProject) | ||||
|     implementation("com.mingliqiye.utils.jna:WinKernel32Platform:1.0.1") | ||||
| } | ||||
| @ -16,9 +16,10 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils | ||||
|  * CurrentFile settings.gradle.kts | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * LastUpdate 2025-09-13 02:37:04 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| include("jdk8") | ||||
| val ARTIFACTID: String by settings.extra | ||||
| rootProject.name = ARTIFACTID | ||||
|  | ||||
| @ -1,133 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile AesUtils.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.aes; | ||||
| 
 | ||||
| import com.mingliqiye.utils.base64.Base64Utils; | ||||
| import java.nio.charset.StandardCharsets; | ||||
| import java.security.GeneralSecurityException; | ||||
| import java.security.MessageDigest; | ||||
| import java.security.SecureRandom; | ||||
| import javax.crypto.Cipher; | ||||
| import javax.crypto.spec.GCMParameterSpec; | ||||
| import javax.crypto.spec.SecretKeySpec; | ||||
| 
 | ||||
| public class AesUtils { | ||||
| 
 | ||||
| 	private static final String ALGORITHM = "AES"; | ||||
| 	private static final String TRANSFORMATION = "AES/GCM/NoPadding"; | ||||
| 	private static final int GCM_IV_LENGTH = 12; | ||||
| 	private static final int GCM_TAG_LENGTH = 16; | ||||
| 	private static final SecureRandom SECURE_RANDOM = new SecureRandom(); | ||||
| 
 | ||||
| 	/** | ||||
| 	 * AES加密方法(使用GCM模式) | ||||
| 	 * @param sSrc 待加密的字符串 | ||||
| 	 * @param sKey 加密密钥 | ||||
| 	 * @return 加密后的字符串,格式为 IV:EncryptedData+Tag(均为Base64编码) | ||||
| 	 * @throws GeneralSecurityException 加密错误 | ||||
| 	 */ | ||||
| 	public static String encrypt(String sSrc, String sKey) | ||||
| 		throws GeneralSecurityException { | ||||
| 		if (sKey == null) { | ||||
| 			return null; | ||||
| 		} | ||||
| 
 | ||||
| 		// 生成密钥 | ||||
| 		SecretKeySpec secretKeySpec = createSecretKey(sKey); | ||||
| 
 | ||||
| 		// 生成安全随机IV | ||||
| 		byte[] iv = new byte[GCM_IV_LENGTH]; | ||||
| 		SECURE_RANDOM.nextBytes(iv); | ||||
| 
 | ||||
| 		// 初始化加密器 | ||||
| 		Cipher cipher = Cipher.getInstance(TRANSFORMATION); | ||||
| 		GCMParameterSpec gcmParameterSpec = new GCMParameterSpec( | ||||
| 			GCM_TAG_LENGTH * 8, | ||||
| 			iv | ||||
| 		); | ||||
| 		cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, gcmParameterSpec); | ||||
| 
 | ||||
| 		// 执行加密 | ||||
| 		byte[] encrypted = cipher.doFinal( | ||||
| 			sSrc.getBytes(StandardCharsets.UTF_8) | ||||
| 		); | ||||
| 
 | ||||
| 		// 将IV和加密数据(包含认证标签)组合返回 | ||||
| 		return Base64Utils.encode( | ||||
| 			(Base64Utils.encode(iv) + | ||||
| 				":" + | ||||
| 				Base64Utils.encode(encrypted)).getBytes() | ||||
| 		); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * AES解密方法(使用GCM模式) | ||||
| 	 * @param sSrc 待解密的字符串,格式为 IV:EncryptedData+Tag(均为Base64编码) | ||||
| 	 * @param sKey 解密密钥 | ||||
| 	 * @return 解密后的原始字符串 | ||||
| 	 */ | ||||
| 	public static String decrypt(String sSrc, String sKey) { | ||||
| 		if (sKey == null) { | ||||
| 			return null; | ||||
| 		} | ||||
| 
 | ||||
| 		try { | ||||
| 			// 分割IV和加密数据 | ||||
| 			String sSrcs = new String(Base64Utils.decode(sSrc)); | ||||
| 			String[] parts = sSrcs.split(":", 2); | ||||
| 			if (parts.length != 2) { | ||||
| 				return null; | ||||
| 			} | ||||
| 			byte[] iv = Base64Utils.decode(parts[0]); | ||||
| 			byte[] encryptedData = Base64Utils.decode(parts[1]); | ||||
| 			if (iv.length != GCM_IV_LENGTH) { | ||||
| 				return null; | ||||
| 			} | ||||
| 			SecretKeySpec secretKeySpec = createSecretKey(sKey); | ||||
| 			Cipher cipher = Cipher.getInstance(TRANSFORMATION); | ||||
| 			GCMParameterSpec gcmParameterSpec = new GCMParameterSpec( | ||||
| 				GCM_TAG_LENGTH * 8, | ||||
| 				iv | ||||
| 			); | ||||
| 			cipher.init(Cipher.DECRYPT_MODE, secretKeySpec, gcmParameterSpec); | ||||
| 			byte[] original = cipher.doFinal(encryptedData); | ||||
| 			return new String(original, StandardCharsets.UTF_8); | ||||
| 		} catch (Exception e) { | ||||
| 			return null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 创建AES密钥,支持任意长度的密钥 | ||||
| 	 * @param sKey 字符串密钥 | ||||
| 	 * @return SecretKeySpec对象 | ||||
| 	 * @throws Exception 可能抛出的异常 | ||||
| 	 */ | ||||
| 	private static SecretKeySpec createSecretKey(String sKey) | ||||
| 		throws GeneralSecurityException { | ||||
| 		byte[] key = sKey.getBytes(StandardCharsets.UTF_8); | ||||
| 		MessageDigest md = MessageDigest.getInstance("MD5"); | ||||
| 		byte[] digest = md.digest(key); | ||||
| 		return new SecretKeySpec(digest, ALGORITHM); | ||||
| 	} | ||||
| } | ||||
| @ -1,198 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Base64Utils.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.base64; | ||||
| 
 | ||||
| import java.io.File; | ||||
| import java.io.FileOutputStream; | ||||
| import java.io.IOException; | ||||
| import java.util.Base64; | ||||
| 
 | ||||
| /** | ||||
|  * Base64工具类,提供对字节数组、文件和字符串的Base64编码与解码功能。 | ||||
|  */ | ||||
| public class Base64Utils { | ||||
| 
 | ||||
| 	// Base64编码器实例 | ||||
| 	private static final Base64.Encoder BASE_64_ENCODER = Base64.getEncoder(); | ||||
| 	// Base64解码器实例 | ||||
| 	private static final Base64.Decoder BASE_64_DECODER = Base64.getDecoder(); | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对字节数组进行Base64编码。 | ||||
| 	 * | ||||
| 	 * @param bytes 待编码的字节数组 | ||||
| 	 * @return 编码后的Base64字符串 | ||||
| 	 */ | ||||
| 	public static String encode(byte[] bytes) { | ||||
| 		return BASE_64_ENCODER.encodeToString(bytes); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对文件内容进行Base64编码。 | ||||
| 	 * | ||||
| 	 * @param file 待编码的文件对象 | ||||
| 	 * @return 编码后的Base64字符串 | ||||
| 	 * @throws RuntimeException 如果读取文件时发生IO异常 | ||||
| 	 */ | ||||
| 	public static String encode(File file) { | ||||
| 		try { | ||||
| 			byte[] bytes = java.nio.file.Files.readAllBytes(file.toPath()); | ||||
| 			return encode(bytes); | ||||
| 		} catch (IOException e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 根据文件路径对文件内容进行Base64编码。 | ||||
| 	 * | ||||
| 	 * @param filePath 文件路径 | ||||
| 	 * @return 编码后的Base64字符串 | ||||
| 	 */ | ||||
| 	public static String encode(String filePath) { | ||||
| 		return encode(new File(filePath)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 安全地对文件内容进行Base64编码,出错时返回null。 | ||||
| 	 * | ||||
| 	 * @param file 待编码的文件对象 | ||||
| 	 * @return 编码后的Base64字符串,出错时返回null | ||||
| 	 */ | ||||
| 	public static String encodeSafe(File file) { | ||||
| 		try { | ||||
| 			return encode(file); | ||||
| 		} catch (Exception e) { | ||||
| 			return null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 安全地根据文件路径对文件内容进行Base64编码,出错时返回null。 | ||||
| 	 * | ||||
| 	 * @param filePath 文件路径 | ||||
| 	 * @return 编码后的Base64字符串,出错时返回null | ||||
| 	 */ | ||||
| 	public static String encodeSafe(String filePath) { | ||||
| 		try { | ||||
| 			return encode(filePath); | ||||
| 		} catch (Exception e) { | ||||
| 			return null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对Base64字符串进行解码。 | ||||
| 	 * | ||||
| 	 * @param base64 待解码的Base64字符串 | ||||
| 	 * @return 解码后的字节数组 | ||||
| 	 */ | ||||
| 	public static byte[] decode(String base64) { | ||||
| 		return BASE_64_DECODER.decode(base64); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 安全地对Base64字符串进行解码,出错时返回null。 | ||||
| 	 * | ||||
| 	 * @param base64 待解码的Base64字符串 | ||||
| 	 * @return 解码后的字节数组,出错时返回null | ||||
| 	 */ | ||||
| 	public static byte[] decodeSafe(String base64) { | ||||
| 		try { | ||||
| 			return decode(base64); | ||||
| 		} catch (Exception e) { | ||||
| 			return null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将Base64字符串解码并写入指定文件。 | ||||
| 	 * | ||||
| 	 * @param base64 待解码的Base64字符串 | ||||
| 	 * @param file   目标文件对象 | ||||
| 	 * @throws RuntimeException 如果写入文件时发生IO异常 | ||||
| 	 */ | ||||
| 	public static void decodeToFile(String base64, File file) { | ||||
| 		try (FileOutputStream fos = new FileOutputStream(file)) { | ||||
| 			byte[] bytes = decode(base64); | ||||
| 			fos.write(bytes); | ||||
| 		} catch (IOException e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将Base64字符串解码并写入指定路径的文件。 | ||||
| 	 * | ||||
| 	 * @param base64   待解码的Base64字符串 | ||||
| 	 * @param filePath 目标文件路径 | ||||
| 	 */ | ||||
| 	public static void decodeToFile(String base64, String filePath) { | ||||
| 		decodeToFile(base64, new File(filePath)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 安全地将Base64字符串解码并写入指定文件,出错时返回false。 | ||||
| 	 * | ||||
| 	 * @param base64 待解码的Base64字符串 | ||||
| 	 * @param file   目标文件对象 | ||||
| 	 * @return 成功写入返回true,否则返回false | ||||
| 	 */ | ||||
| 	public static boolean decodeToFileSafe(String base64, File file) { | ||||
| 		try { | ||||
| 			decodeToFile(base64, file); | ||||
| 			return true; | ||||
| 		} catch (Exception e) { | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 安全地将Base64字符串解码并写入指定路径的文件,出错时返回false。 | ||||
| 	 * | ||||
| 	 * @param base64   待解码的Base64字符串 | ||||
| 	 * @param filePath 目标文件路径 | ||||
| 	 * @return 成功写入返回true,否则返回false | ||||
| 	 */ | ||||
| 	public static boolean decodeToFileSafe(String base64, String filePath) { | ||||
| 		return decodeToFileSafe(base64, new File(filePath)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对字节数组中指定范围的数据进行Base64编码。 | ||||
| 	 * | ||||
| 	 * @param bytes  源字节数组 | ||||
| 	 * @param offset 起始偏移量 | ||||
| 	 * @param length 要编码的数据长度 | ||||
| 	 * @return 编码后的Base64字符串 | ||||
| 	 */ | ||||
| 	public static String encodeBytes(byte[] bytes, int offset, int length) { | ||||
| 		byte[] data = new byte[length]; | ||||
| 		System.arraycopy(bytes, offset, data, 0, length); | ||||
| 		return encode(data); | ||||
| 	} | ||||
| 
 | ||||
| 	public static String encodeBytes(byte[] bytes) { | ||||
| 		return encodeBytes(bytes, 0, bytes.length); | ||||
| 	} | ||||
| } | ||||
| @ -1,270 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Factory.java | ||||
|  * LastUpdate 2025-09-09 08:39:07 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean; | ||||
| 
 | ||||
| import com.mingliqiye.utils.bean.annotation.ComponentBean; | ||||
| import com.mingliqiye.utils.bean.annotation.InjectBean; | ||||
| import java.io.File; | ||||
| import java.lang.reflect.Field; | ||||
| import java.net.URL; | ||||
| import java.util.Enumeration; | ||||
| import java.util.concurrent.ConcurrentHashMap; | ||||
| import java.util.concurrent.ConcurrentMap; | ||||
| 
 | ||||
| /** | ||||
|  * 类似于SpringBoot的Bean管理器 | ||||
|  * | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| public class Factory { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 存储所有已注册的Bean实例,键为Bean名称,值为Bean实例 | ||||
| 	 */ | ||||
| 	public static final ConcurrentMap<String, Object> BEANS = | ||||
| 		new ConcurrentHashMap<>(); | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 存储按类型查找的Bean实例,键为Bean的Class对象,值为Bean实例 | ||||
| 	 */ | ||||
| 	private static final ConcurrentMap<Class<?>, Object> TYPE_BEANS = | ||||
| 		new ConcurrentHashMap<>(); | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 私有构造函数,防止外部实例化该类 | ||||
| 	 */ | ||||
| 	private Factory() {} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 自动扫描指定类所在包下的所有类并注册为Bean | ||||
| 	 * | ||||
| 	 * @param c 指定的类,用于获取其所在的包 | ||||
| 	 * @throws IllegalArgumentException 如果传入的类为null或位于默认包中 | ||||
| 	 */ | ||||
| 	public static void autoScan(Class<?> c) { | ||||
| 		if (c == null) { | ||||
| 			throw new IllegalArgumentException("Class cannot be null"); | ||||
| 		} | ||||
| 		Package pkg = c.getPackage(); | ||||
| 		if (pkg == null) { | ||||
| 			throw new IllegalArgumentException( | ||||
| 				"Class is in the default package" | ||||
| 			); | ||||
| 		} | ||||
| 		scan(pkg.getName()); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 扫描指定包路径下的所有类文件,并注册其中带有@ComponentBean注解的类为Bean | ||||
| 	 * | ||||
| 	 * @param basePackage 要扫描的基础包名 | ||||
| 	 * @throws RuntimeException 如果在扫描过程中发生异常 | ||||
| 	 */ | ||||
| 	public static void scan(String basePackage) { | ||||
| 		try { | ||||
| 			String path = basePackage.replace('.', '/'); | ||||
| 			ClassLoader classLoader = | ||||
| 				Thread.currentThread().getContextClassLoader(); | ||||
| 			Enumeration<URL> resources = null; | ||||
| 			resources = classLoader.getResources(path); | ||||
| 			while (resources.hasMoreElements()) { | ||||
| 				URL resource = resources.nextElement(); | ||||
| 				File file = new File(resource.toURI()); | ||||
| 				scanDirectory(file, basePackage); | ||||
| 			} | ||||
| 			injectDependencies(); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 递归扫描目录中的所有类文件,并注册符合条件的类为Bean | ||||
| 	 * | ||||
| 	 * @param directory   当前要扫描的目录 | ||||
| 	 * @param packageName 当前目录对应的包名 | ||||
| 	 * @throws Exception 如果在扫描或类加载过程中发生异常 | ||||
| 	 */ | ||||
| 	private static void scanDirectory(File directory, String packageName) | ||||
| 		throws Exception { | ||||
| 		File[] files = directory.listFiles(); | ||||
| 		if (files == null) { | ||||
| 			return; | ||||
| 		} | ||||
| 
 | ||||
| 		for (File file : files) { | ||||
| 			if (file.isDirectory()) { | ||||
| 				scanDirectory(file, packageName + "." + file.getName()); | ||||
| 			} else if (file.getName().endsWith(".class")) { | ||||
| 				String className = | ||||
| 					packageName + '.' + file.getName().replace(".class", ""); | ||||
| 				registerComponent(Class.forName(className)); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 注册一个带有@ComponentBean注解的类为Bean实例 | ||||
| 	 * | ||||
| 	 * @param clazz 要注册的类 | ||||
| 	 * @throws Exception 如果在实例化类或处理注解时发生异常 | ||||
| 	 */ | ||||
| 	private static void registerComponent(Class<?> clazz) throws Exception { | ||||
| 		if (clazz.isAnnotationPresent(ComponentBean.class)) { | ||||
| 			ComponentBean component = clazz.getAnnotation(ComponentBean.class); | ||||
| 			String name = component.value().isEmpty() | ||||
| 				? clazz.getName() | ||||
| 				: component.value(); | ||||
| 			Object instance = clazz.getDeclaredConstructor().newInstance(); | ||||
| 			BEANS.put(name, instance); | ||||
| 			TYPE_BEANS.put(clazz, instance); | ||||
| 
 | ||||
| 			for (Class<?> interfaceClass : clazz.getInterfaces()) { | ||||
| 				TYPE_BEANS.putIfAbsent(interfaceClass, instance); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对所有已注册的Bean进行依赖注入处理 | ||||
| 	 * | ||||
| 	 * @throws Exception 如果在注入过程中发生异常 | ||||
| 	 */ | ||||
| 	private static void injectDependencies() throws Exception { | ||||
| 		for (Object bean : BEANS.values()) { | ||||
| 			for (Field field : bean.getClass().getDeclaredFields()) { | ||||
| 				if (field.isAnnotationPresent(InjectBean.class)) { | ||||
| 					InjectBean inject = field.getAnnotation(InjectBean.class); | ||||
| 					Object dependency = findDependency( | ||||
| 						field.getType(), | ||||
| 						inject.value() | ||||
| 					); | ||||
| 					if (dependency == null) { | ||||
| 						throw new IllegalStateException( | ||||
| 							"No suitable dependency found for field " + | ||||
| 							field.getName() + | ||||
| 							" in class " + | ||||
| 							bean.getClass().getName() | ||||
| 						); | ||||
| 					} | ||||
| 					field.setAccessible(true); | ||||
| 					field.set(bean, dependency); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 根据类型和名称查找对应的依赖实例 | ||||
| 	 * | ||||
| 	 * @param type 依赖的类型 | ||||
| 	 * @param name 依赖的名称(可为空) | ||||
| 	 * @return 找到的依赖实例,未找到则返回null | ||||
| 	 */ | ||||
| 	private static Object findDependency(Class<?> type, String name) { | ||||
| 		if (!name.isEmpty()) { | ||||
| 			return BEANS.get(name); | ||||
| 		} | ||||
| 
 | ||||
| 		Object dependency = TYPE_BEANS.get(type); | ||||
| 		if (dependency != null) { | ||||
| 			return dependency; | ||||
| 		} | ||||
| 
 | ||||
| 		for (Class<?> interfaceType : TYPE_BEANS.keySet()) { | ||||
| 			if (type.isAssignableFrom(interfaceType)) { | ||||
| 				return TYPE_BEANS.get(interfaceType); | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		return null; | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将一个对象添加到Bean容器中,使用其类名作为键 | ||||
| 	 * | ||||
| 	 * @param object 要添加的对象 | ||||
| 	 * @throws RuntimeException 如果在注入依赖时发生异常 | ||||
| 	 */ | ||||
| 	public static void add(Object object) { | ||||
| 		Class<?> clazz = object.getClass(); | ||||
| 		String name = clazz.getName(); | ||||
| 		BEANS.put(name, object); | ||||
| 		TYPE_BEANS.put(clazz, object); | ||||
| 		try { | ||||
| 			injectDependencies(); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将一个对象以指定名称添加到Bean容器中 | ||||
| 	 * | ||||
| 	 * @param name   Bean的名称 | ||||
| 	 * @param object 要添加的对象 | ||||
| 	 * @throws RuntimeException 如果在注入依赖时发生异常 | ||||
| 	 */ | ||||
| 	public static void add(String name, Object object) { | ||||
| 		BEANS.put(name, object); | ||||
| 		TYPE_BEANS.put(object.getClass(), object); | ||||
| 		try { | ||||
| 			injectDependencies(); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 根据类型获取对应的Bean实例 | ||||
| 	 * | ||||
| 	 * @param objclass Bean的类型 | ||||
| 	 * @param <T>      Bean的泛型类型 | ||||
| 	 * @return 对应类型的Bean实例,未找到则返回null | ||||
| 	 */ | ||||
| 	public static <T> T get(Class<T> objclass) { | ||||
| 		return objclass.cast(TYPE_BEANS.get(objclass)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 根据名称和类型获取对应的Bean实例 | ||||
| 	 * | ||||
| 	 * @param name     Bean的名称 | ||||
| 	 * @param objclass Bean的类型 | ||||
| 	 * @param <T>      Bean的泛型类型 | ||||
| 	 * @return 对应名称和类型的Bean实例,未找到则返回null | ||||
| 	 */ | ||||
| 	public static <T> T get(String name, Class<T> objclass) { | ||||
| 		return objclass.cast(BEANS.get(name)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 根据名称获取对应的Bean实例 | ||||
| 	 * | ||||
| 	 * @param name Bean的名称 | ||||
| 	 * @return 对应名称的Bean实例,未找到则返回null | ||||
| 	 */ | ||||
| 	public static Object get(String name) { | ||||
| 		return BEANS.get(name); | ||||
| 	} | ||||
| } | ||||
| @ -1,37 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile ComponentBean.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean.annotation; | ||||
| 
 | ||||
| import java.lang.annotation.ElementType; | ||||
| import java.lang.annotation.Retention; | ||||
| import java.lang.annotation.RetentionPolicy; | ||||
| import java.lang.annotation.Target; | ||||
| 
 | ||||
| /** | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @Retention(RetentionPolicy.RUNTIME) | ||||
| @Target(ElementType.TYPE) | ||||
| public @interface ComponentBean { | ||||
| 	String value() default ""; | ||||
| } | ||||
| @ -1,97 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile SpringBeanUtil.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean.springboot; | ||||
| 
 | ||||
| import lombok.Getter; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.springframework.beans.BeansException; | ||||
| import org.springframework.context.ApplicationContext; | ||||
| import org.springframework.context.ApplicationContextAware; | ||||
| import org.springframework.stereotype.Component; | ||||
| 
 | ||||
| /** | ||||
|  * Spring Bean工具类 | ||||
|  * 实现ApplicationContextAware接口,并加入Component注解,让spring扫描到该bean | ||||
|  * 该类用于在普通Java类中注入bean,普通Java类中用@Autowired是无法注入bean的 | ||||
|  * <p> | ||||
|  * 需要放入扫描类中 | ||||
|  * | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @Component | ||||
| public class SpringBeanUtil implements ApplicationContextAware { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 获取applicationContext | ||||
| 	 */ | ||||
| 	@Getter | ||||
| 	private static ApplicationContext applicationContext; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 通过bean名称获取Bean实例 | ||||
| 	 * | ||||
| 	 * @param name bean名称 | ||||
| 	 * @return bean实例对象 | ||||
| 	 */ | ||||
| 	public static Object getBean(String name) { | ||||
| 		return getApplicationContext().getBean(name); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 通过bean类型获取Bean实例 | ||||
| 	 * | ||||
| 	 * @param clazz bean的Class类型 | ||||
| 	 * @param <T>   泛型类型 | ||||
| 	 * @return 指定类型的bean实例 | ||||
| 	 */ | ||||
| 	public static <T> T getBean(Class<T> clazz) { | ||||
| 		return getApplicationContext().getBean(clazz); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 通过bean名称和类型获取指定的Bean实例 | ||||
| 	 * | ||||
| 	 * @param name  bean名称 | ||||
| 	 * @param clazz bean的Class类型 | ||||
| 	 * @param <T>   泛型类型 | ||||
| 	 * @return 指定名称和类型的bean实例 | ||||
| 	 */ | ||||
| 	public static <T> T getBean(String name, Class<T> clazz) { | ||||
| 		return getApplicationContext().getBean(name, clazz); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 设置ApplicationContext上下文对象 | ||||
| 	 * 当Spring容器初始化时会自动调用此方法,将ApplicationContext注入到本工具类中 | ||||
| 	 * 通过判断避免重复赋值,确保只设置一次ApplicationContext | ||||
| 	 * | ||||
| 	 * @param applicationContext Spring应用上下文对象 | ||||
| 	 * @throws BeansException bean异常 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public void setApplicationContext( | ||||
| 		@NotNull ApplicationContext applicationContext | ||||
| 	) throws BeansException { | ||||
| 		SpringBeanUtil.applicationContext = applicationContext; | ||||
| 	} | ||||
| } | ||||
| @ -1,65 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile ByteUtil.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bytes; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
| 
 | ||||
| /** | ||||
|  * @author MingLiPro | ||||
|  * | ||||
|  * 字节数组处理工具类 | ||||
|  */ | ||||
| public class ByteUtil { | ||||
| 
 | ||||
| 	public static final byte ESC_ASC = 0x1A; | ||||
| 	public static final byte ESC_DESC = 0x1B; | ||||
| 	public static final byte ESC_NONE = 0x00; | ||||
| 	public static final byte ESC_START = 0x01; | ||||
| 	public static final byte ESC_END = 0x02; | ||||
| 	public static final byte ESC_ESC = 0x03; | ||||
| 	public static final byte ESC_CONTROL = 0x04; | ||||
| 	public static final byte ESC_DATA = 0x05; | ||||
| 	public static final byte ESC_RESERVED = 0x06; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将字节数组转换为十六进制字符串列表 | ||||
| 	 * <p> | ||||
| 	 * 每个字节都会被转换为两位的十六进制字符串表示形式 | ||||
| 	 * 例如: 字节值为10的字节会被转换为"0a",值为255的字节会被转换为"ff" | ||||
| 	 * | ||||
| 	 * @param bytes 输入的字节数组 | ||||
| 	 * @return 包含每个字节对应十六进制字符串的列表 | ||||
| 	 */ | ||||
| 	public static List<String> getByteArrayString(byte[] bytes) { | ||||
| 		List<Byte> byteList = new ArrayList<>(bytes.length); | ||||
| 		for (byte aByte : bytes) { | ||||
| 			byteList.add(aByte); | ||||
| 		} | ||||
| 		return byteList | ||||
| 			.stream() | ||||
| 			.map(a -> String.format("%02x", a & 0xFF)) | ||||
| 			.collect(Collectors.toList()); | ||||
| 	} | ||||
| } | ||||
| @ -1,80 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile CloneUtil.java | ||||
|  * LastUpdate 2025-09-09 09:32:17 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.clone; | ||||
| 
 | ||||
| import com.mingliqiye.utils.json.JsonApi; | ||||
| import com.mingliqiye.utils.json.JsonException; | ||||
| import java.io.*; | ||||
| 
 | ||||
| public class CloneUtil { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 对指定的可序列化对象进行深拷贝 | ||||
| 	 * | ||||
| 	 * @param object 需要进行深拷贝的对象,必须实现Serializable接口 | ||||
| 	 * @param <T> 对象的类型,必须实现Serializable接口 | ||||
| 	 * @return 深拷贝后的新对象,与原对象内容相同但内存地址不同 | ||||
| 	 * @throws RuntimeException 当序列化或反序列化过程中发生IO异常或类未找到异常时抛出 | ||||
| 	 */ | ||||
| 	public static <T extends Serializable> T deepClone(T object) { | ||||
| 		try { | ||||
| 			ByteArrayOutputStream bao = new ByteArrayOutputStream(); | ||||
| 			ObjectOutputStream oos = new ObjectOutputStream(bao); | ||||
| 			oos.writeObject(object); | ||||
| 			ByteArrayInputStream bis = new ByteArrayInputStream( | ||||
| 				bao.toByteArray() | ||||
| 			); | ||||
| 			ObjectInputStream ois = new ObjectInputStream(bis); | ||||
| 			return (T) ois.readObject(); | ||||
| 		} catch (IOException | ClassNotFoundException e) { | ||||
| 			throw new RuntimeException(e); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 深度克隆对象,使用JSON序列化和反序列化实现 | ||||
| 	 * | ||||
| 	 * @param <T> 对象类型参数 | ||||
| 	 * @param object 需要克隆的对象实例 | ||||
| 	 * @param jsonApi JSON序列化接口实现 | ||||
| 	 * @return 克隆后的对象实例 | ||||
| 	 */ | ||||
| 	public static <T> T deepJsonClone(T object, JsonApi jsonApi) { | ||||
| 		if (object == null) { | ||||
| 			return null; | ||||
| 		} | ||||
| 
 | ||||
| 		if (jsonApi == null) { | ||||
| 			throw new IllegalArgumentException("jsonApi cannot be null"); | ||||
| 		} | ||||
| 
 | ||||
| 		try { | ||||
| 			return (T) jsonApi.convert(object, object.getClass()); | ||||
| 		} catch (Exception e) { | ||||
| 			throw new JsonException( | ||||
| 				"Failed to deep clone object using JSON", | ||||
| 				e | ||||
| 			); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @ -16,22 +16,20 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Lists.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * LastUpdate 2025-09-13 00:26:11 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.collection; | ||||
| 
 | ||||
| import static com.mingliqiye.utils.collection.Collection.findFirst; | ||||
| 
 | ||||
| import com.github.f4b6a3.uuid.util.internal.RandomUtil; | ||||
| import com.mingliqiye.utils.random.RandomInt; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| 
 | ||||
| import java.util.*; | ||||
| import java.util.Collection; | ||||
| import java.util.stream.Collectors; | ||||
| import java.util.stream.Stream; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| 
 | ||||
| /** | ||||
|  * Lists工具类提供了一系列创建List实现的便捷方法。 | ||||
| @ -457,6 +455,12 @@ public class Lists { | ||||
| 		return list; | ||||
| 	} | ||||
| 
 | ||||
| 	public static <T> List<T> toList(Iterator<T> iterator) { | ||||
| 		List<T> list = newArrayList(10); | ||||
| 		ForEach.forEach(iterator, item -> list.add(item)); | ||||
| 		return list; | ||||
| 	} | ||||
| 
 | ||||
| 	public <T> T getFirst(Collection<T> collectors) { | ||||
| 		return com.mingliqiye.utils.collection.Collection.getFirst(collectors); | ||||
| 	} | ||||
|  | ||||
| @ -16,13 +16,16 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Range.java | ||||
|  * LastUpdate 2025-09-12 17:12:29 | ||||
|  * LastUpdate 2025-09-14 20:23:26 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.iterator; | ||||
| 
 | ||||
| import kotlin.ranges.ClosedRange; | ||||
| import kotlin.ranges.OpenEndRange; | ||||
| import lombok.Getter; | ||||
| import lombok.val; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| import java.util.Iterator; | ||||
| @ -35,7 +38,8 @@ import java.util.Iterator; | ||||
|  * @since 3.2.6 | ||||
|  */ | ||||
| @Getter | ||||
| public class Range implements Iterable<Integer> { | ||||
| public class Range | ||||
| 	implements Iterable<Integer>, ClosedRange<Integer>, OpenEndRange<Integer> { | ||||
| 
 | ||||
| 	private final int start; | ||||
| 	private final int end; | ||||
| @ -183,4 +187,35 @@ public class Range implements Iterable<Integer> { | ||||
| 			} | ||||
| 		}; | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public boolean isEmpty() { | ||||
| 		return current < end; | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public @NotNull Integer getEndInclusive() { | ||||
| 		val va = end - step; | ||||
| 		return Math.max(va, 0); | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public boolean contains(@NotNull Integer integer) { | ||||
| 		if (step == 0) return false; | ||||
| 		if (step > 0) { | ||||
| 			if (integer < start || integer > end) return false; | ||||
| 		} else { | ||||
| 			if (integer > start || integer < end) return false; | ||||
| 		} | ||||
| 		return (integer - start) % step == 0; | ||||
| 	} | ||||
| 
 | ||||
| 	@Override | ||||
| 	public @NotNull Integer getEndExclusive() { | ||||
| 		return end; | ||||
| 	} | ||||
| 	@Override | ||||
| 	public @NotNull Integer getStart() { | ||||
| 		return start; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -1,59 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile FieldStructure.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.jna; | ||||
| 
 | ||||
| import com.sun.jna.Structure; | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.Modifier; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| /** | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| public class FieldStructure extends Structure { | ||||
| 
 | ||||
| 	@Override | ||||
| 	protected List<String> getFieldOrder() { | ||||
| 		List<String> fieldOrderList = new ArrayList<>(); | ||||
| 		for ( | ||||
| 			Class<?> cls = getClass(); | ||||
| 			!cls.equals(FieldStructure.class); | ||||
| 			cls = cls.getSuperclass() | ||||
| 		) { | ||||
| 			Field[] fields = cls.getDeclaredFields(); | ||||
| 			int modifiers; | ||||
| 			for (Field field : fields) { | ||||
| 				modifiers = field.getModifiers(); | ||||
| 				if ( | ||||
| 					Modifier.isStatic(modifiers) || | ||||
| 					!Modifier.isPublic(modifiers) | ||||
| 				) { | ||||
| 					continue; | ||||
| 				} | ||||
| 				fieldOrderList.add(field.getName()); | ||||
| 			} | ||||
| 		} | ||||
| 		return fieldOrderList; | ||||
| 	} | ||||
| } | ||||
| @ -1,40 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile WinKernel32.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.jna.time; | ||||
| 
 | ||||
| import com.sun.jna.Library; | ||||
| import com.sun.jna.Native; | ||||
| import com.sun.jna.ptr.LongByReference; | ||||
| 
 | ||||
| /** | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| public interface WinKernel32 extends Library { | ||||
| 	static WinKernel32 load() { | ||||
| 		return Native.load("kernel32", WinKernel32.class); | ||||
| 	} | ||||
| 
 | ||||
| 	boolean QueryPerformanceCounter(LongByReference lpPerformanceCount); | ||||
| 	boolean QueryPerformanceFrequency(LongByReference lpFrequency); | ||||
| 	void GetSystemTimePreciseAsFileTime(byte[] lpSystemTimeAsFileTime); | ||||
| } | ||||
| @ -16,22 +16,26 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile AutoConfiguration.java | ||||
|  * LastUpdate 2025-09-09 08:37:34 | ||||
|  * LastUpdate 2025-09-13 01:23:09 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.springboot.autoconfigure; | ||||
| 
 | ||||
| import com.mingliqiye.utils.collection.ForEach; | ||||
| import com.mingliqiye.utils.collection.Lists; | ||||
| import com.mingliqiye.utils.system.SystemUtil; | ||||
| import com.mingliqiye.utils.time.DateTime; | ||||
| import com.mingliqiye.utils.time.Formatter; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import lombok.val; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.boot.context.properties.EnableConfigurationProperties; | ||||
| import org.springframework.context.annotation.ComponentScan; | ||||
| 
 | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| 
 | ||||
| @org.springframework.boot.autoconfigure.AutoConfiguration | ||||
| @EnableConfigurationProperties(AutoConfiguration.class) | ||||
| @ComponentScan( | ||||
| @ -80,7 +84,10 @@ public class AutoConfiguration { | ||||
| 			while ((readlen = inputStream.read(buffer)) != -1) { | ||||
| 				metaData.append(new String(buffer, 0, readlen)); | ||||
| 			} | ||||
| 			ForEach.forEach(metaData.toString().split("\n"), (s, i) -> { | ||||
| 			val da = Lists.toList(metaData.toString().split("\n")); | ||||
| 			da.add("time=" + DateTime.now().format(Formatter.STANDARD_DATETIME_MILLISECOUND7)); | ||||
| 			da.add("jdkRuntime=" + SystemUtil.getJdkVersion()); | ||||
| 			ForEach.forEach(da, (s, i) -> { | ||||
| 				String[] d = s.trim().split("=", 2); | ||||
| 				if (d.length >= 2) { | ||||
| 					String content = "|  " + d[0] + ": " + d[1]; | ||||
| @ -100,9 +107,8 @@ public class AutoConfiguration { | ||||
| 			e.printStackTrace(); | ||||
| 		} | ||||
| 		banner2 = bannerBuilder.toString(); | ||||
| 		System.out.printf( | ||||
| 			banner2, | ||||
| 			DateTime.now().format(Formatter.STANDARD_DATETIME_MILLISECOUND7) | ||||
| 		System.out.print( | ||||
| 			banner2 | ||||
| 		); | ||||
| 		System.out.println( | ||||
| 			"---------------------------------------------------------" | ||||
|  | ||||
| @ -16,7 +16,7 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile SuperStream.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * LastUpdate 2025-09-14 20:16:59 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| @ -25,12 +25,13 @@ package com.mingliqiye.utils.stream; | ||||
| import com.mingliqiye.utils.collection.Lists; | ||||
| import com.mingliqiye.utils.collection.Maps; | ||||
| import com.mingliqiye.utils.stream.interfaces.Getable; | ||||
| import lombok.val; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| import java.util.*; | ||||
| import java.util.concurrent.ConcurrentMap; | ||||
| import java.util.function.*; | ||||
| import java.util.stream.*; | ||||
| import lombok.val; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| /** | ||||
|  * 自定义的 SuperStream 实现类,用于对集合进行流式操作。 | ||||
|  | ||||
| @ -1,271 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile StringUtil.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.string; | ||||
| 
 | ||||
| import com.mingliqiye.utils.collection.Lists; | ||||
| import com.mingliqiye.utils.functions.P1RFunction; | ||||
| import java.text.MessageFormat; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| 
 | ||||
| /** | ||||
|  * 字符串工具类,提供常用的字符串处理方法 | ||||
|  */ | ||||
| public class StringUtil { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将对象转换为字符串表示形式 | ||||
| 	 * | ||||
| 	 * @param obj 需要转换的对象,可以为null | ||||
| 	 * @return 如果对象为null则返回空字符串,否则返回对象的字符串表示 | ||||
| 	 */ | ||||
| 	public static String toString(Object obj) { | ||||
| 		// 如果对象为null,返回空字符串;否则调用对象的toString方法 | ||||
| 		return obj == null ? "" : obj.toString(); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 格式化字符串,将格式字符串中的占位符{}替换为对应的参数值<br> | ||||
| 	 * 示例:输出 {} StringUtil.format("{},{},{}", "666", "{}", "777") - "666,{},777"<br> | ||||
| 	 * 示例 StringUtil.format("{},{},{},{}", "666", "{}", "777") - "666,{},777,"<br> | ||||
| 	 * 没有实际{} 会替换为 "" 空字符串 | ||||
| 	 * | ||||
| 	 * @param format 格式字符串,使用{}作为占位符,如果为null则返回null | ||||
| 	 * @param args   用于替换占位符的参数数组 | ||||
| 	 * @return 格式化后的字符串 | ||||
| 	 */ | ||||
| 	public static String format(String format, Object... args) { | ||||
| 		if (format == null) { | ||||
| 			return null; | ||||
| 		} | ||||
| 
 | ||||
| 		StringBuilder sb = new StringBuilder(); | ||||
| 		int placeholderCount = 0; | ||||
| 		int lastIndex = 0; | ||||
| 		int len = format.length(); | ||||
| 
 | ||||
| 		for (int i = 0; i < len - 1; i++) { | ||||
| 			if (format.charAt(i) == '{' && format.charAt(i + 1) == '}') { | ||||
| 				// 添加前面的部分 | ||||
| 				sb.append(format, lastIndex, i); | ||||
| 				// 替换为 MessageFormat 占位符 {index} | ||||
| 				sb.append('{').append(placeholderCount).append('}'); | ||||
| 				placeholderCount++; | ||||
| 				i++; // 跳过 '}' | ||||
| 				lastIndex = i + 1; | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		// 添加剩余部分 | ||||
| 		sb.append(format.substring(lastIndex)); | ||||
| 
 | ||||
| 		// 构造实际参数数组 | ||||
| 		Object[] actualArgs; | ||||
| 		if (args.length < placeholderCount) { | ||||
| 			actualArgs = new String[placeholderCount]; | ||||
| 			System.arraycopy(args, 0, actualArgs, 0, args.length); | ||||
| 			for (int i = args.length; i < placeholderCount; i++) { | ||||
| 				actualArgs[i] = ""; | ||||
| 			} | ||||
| 		} else { | ||||
| 			actualArgs = args; | ||||
| 		} | ||||
| 
 | ||||
| 		// 如果没有占位符,直接返回格式化后的字符串 | ||||
| 		if (placeholderCount == 0) { | ||||
| 			return sb.toString(); | ||||
| 		} | ||||
| 
 | ||||
| 		try { | ||||
| 			return MessageFormat.format( | ||||
| 				sb.toString(), | ||||
| 				(Object[]) Lists.toStringList(actualArgs) | ||||
| 			); | ||||
| 		} catch (IllegalArgumentException e) { | ||||
| 			// 返回原始格式化字符串或抛出自定义异常,视业务需求而定 | ||||
| 			return sb.toString(); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 判断字符串是否为空 | ||||
| 	 * | ||||
| 	 * @param str 待检查的字符串 | ||||
| 	 * @return 如果字符串为null或空字符串则返回true,否则返回false | ||||
| 	 */ | ||||
| 	public static boolean isEmpty(String str) { | ||||
| 		return str == null || str.isEmpty(); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 使用指定的分隔符将多个对象连接成一个字符串 | ||||
| 	 * | ||||
| 	 * @param spec    用作分隔符的字符串 | ||||
| 	 * @param objects 要连接的对象数组 | ||||
| 	 * @return 使用指定分隔符连接后的字符串 | ||||
| 	 */ | ||||
| 	public static String joinOf(String spec, String... objects) { | ||||
| 		return join(spec, Arrays.asList(objects)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将字符串按照指定分隔符分割成字符串列表,并移除列表开头的空字符串元素 | ||||
| 	 * | ||||
| 	 * @param str       待分割的字符串 | ||||
| 	 * @param separator 用作分割符的字符串 | ||||
| 	 * @return 分割后的字符串列表,不包含开头的空字符串元素 | ||||
| 	 */ | ||||
| 	public static List<String> split(String str, String separator) { | ||||
| 		List<String> data = new ArrayList<>( | ||||
| 			Arrays.asList(str.split(separator)) | ||||
| 		); | ||||
| 		// 移除列表开头的所有空字符串元素 | ||||
| 		while (!data.isEmpty() && data.get(0).isEmpty()) { | ||||
| 			data.remove(0); | ||||
| 		} | ||||
| 		return data; | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将列表中的元素按照指定分隔符连接成字符串 | ||||
| 	 * | ||||
| 	 * @param <P>       列表元素的类型 | ||||
| 	 * @param separator 分隔符,用于连接各个元素 | ||||
| 	 * @param list      待连接的元素列表 | ||||
| 	 * @param fun       转换函数,用于将列表元素转换为字符串,如果为null则使用toString()方法 | ||||
| 	 * @return 连接后的字符串,如果列表为空或null则返回空字符串 | ||||
| 	 */ | ||||
| 	public static <P> String join( | ||||
| 		String separator, | ||||
| 		List<P> list, | ||||
| 		P1RFunction<P, String> fun | ||||
| 	) { | ||||
| 		// 处理空列表情况 | ||||
| 		if (list == null || list.isEmpty()) { | ||||
| 			return ""; | ||||
| 		} | ||||
| 
 | ||||
| 		// 构建结果字符串 | ||||
| 		StringBuilder sb = StringUtil.stringBuilder(list.size() * 16); | ||||
| 		for (int i = 0; i < list.size(); i++) { | ||||
| 			P item = list.get(i); | ||||
| 			// 将元素转换为字符串 | ||||
| 			String itemStr = fun == null | ||||
| 				? (item == null ? "null" : item.toString()) | ||||
| 				: fun.call(item); | ||||
| 
 | ||||
| 			// 第一个元素直接添加,其他元素先添加分隔符再添加元素 | ||||
| 			if (i == 0) { | ||||
| 				sb.append(itemStr); | ||||
| 			} else { | ||||
| 				sb.append(separator).append(itemStr); | ||||
| 			} | ||||
| 		} | ||||
| 		return sb.toString(); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 使用指定分隔符连接字符串列表 | ||||
| 	 * | ||||
| 	 * @param separator 分隔符,不能为null | ||||
| 	 * @param list      字符串列表,不能为null | ||||
| 	 * @return 连接后的字符串 | ||||
| 	 * @throws IllegalArgumentException 当separator或list为null时抛出 | ||||
| 	 */ | ||||
| 	public static String join(String separator, List<String> list) { | ||||
| 		if (separator == null) { | ||||
| 			throw new IllegalArgumentException("Separator cannot be null"); | ||||
| 		} | ||||
| 		if (list == null) { | ||||
| 			throw new IllegalArgumentException("List cannot be null"); | ||||
| 		} | ||||
| 		return join(separator, list, null); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 创建一个新的StringBuilder实例 | ||||
| 	 * | ||||
| 	 * @param i 指定StringBuilder的初始容量 | ||||
| 	 * @return 返回一个新的StringBuilder对象,其初始容量为指定的大小 | ||||
| 	 */ | ||||
| 	public static StringBuilder stringBuilder(int i) { | ||||
| 		return new StringBuilder(i); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将字符串转换为Unicode编码格式 | ||||
| 	 * | ||||
| 	 * @param str 待转换的字符串 | ||||
| 	 * @return 转换后的Unicode编码字符串,每个字符都以\\u开头的十六进制形式表示 | ||||
| 	 */ | ||||
| 	public static String stringToUnicode(String str) { | ||||
| 		StringBuilder sb = new StringBuilder(); | ||||
| 		char[] c = str.toCharArray(); | ||||
| 		for (char value : c) { | ||||
| 			sb.append(stringToUnicode(value)); | ||||
| 		} | ||||
| 		return sb.toString(); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将字符转换为Unicode转义字符串 | ||||
| 	 * | ||||
| 	 * @param c 需要转换的字符 | ||||
| 	 * @return 返回格式为"\\uXXXX"的Unicode转义字符串,其中XXXX为字符的十六进制Unicode码点 | ||||
| 	 */ | ||||
| 	public static String stringToUnicode(char c) { | ||||
| 		return "\\u" + String.format("%04x", (int) c); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将整数转换为Unicode字符串表示形式 | ||||
| 	 * | ||||
| 	 * @param c 要转换的整数,表示Unicode码点 | ||||
| 	 * @return 返回格式为"\\uXXXX"的Unicode字符串,其中XXXX是参数c的十六进制表示 | ||||
| 	 */ | ||||
| 	public static String stringToUnicode(Integer c) { | ||||
| 		return "\\u" + Integer.toHexString(c); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将Unicode编码字符串转换为普通字符串 | ||||
| 	 * 该函数接收一个包含Unicode转义序列的字符串,将其解析并转换为对应的字符序列 | ||||
| 	 * | ||||
| 	 * @param unicode 包含Unicode转义序列的字符串,格式如"\\uXXXX",其中XXXX为十六进制数 | ||||
| 	 * @return 转换后的普通字符串,包含对应的Unicode字符 | ||||
| 	 */ | ||||
| 	public static String unicodeToString(String unicode) { | ||||
| 		StringBuilder sb = new StringBuilder(); | ||||
| 		// 按照Unicode转义符分割字符串,得到包含十六进制编码的数组 | ||||
| 		String[] hex = unicode.split("\\\\u"); | ||||
| 		// 从索引1开始遍历,因为分割后的第一个元素是转义符前面的内容(可能为空) | ||||
| 		for (int i = 1; i < hex.length; i++) { | ||||
| 			// 将十六进制字符串转换为整数,作为字符的Unicode码点 | ||||
| 			int index = Integer.parseInt(hex[i], 16); | ||||
| 			// 将Unicode码点转换为对应字符并添加到结果中 | ||||
| 			sb.append((char) index); | ||||
| 		} | ||||
| 		return sb.toString(); | ||||
| 	} | ||||
| } | ||||
| @ -1,32 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Main.java | ||||
|  * LastUpdate 2025-09-12 12:19:48 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.test; | ||||
| 
 | ||||
| import com.mingliqiye.utils.version.VersionUtils; | ||||
| 
 | ||||
| public class Main { | ||||
| 
 | ||||
| 	public static void main(String[] args) { | ||||
| 		System.out.println(VersionUtils.is("1.3", "1.2.3")); | ||||
| 	} | ||||
| } | ||||
| @ -16,14 +16,22 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile DateTime.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * LastUpdate 2025-09-13 10:14:09 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.time; | ||||
| 
 | ||||
| import com.mingliqiye.utils.jna.time.WinKernel32; | ||||
| import com.mingliqiye.utils.jna.WinKernel32Api; | ||||
| import com.mingliqiye.utils.jna.WinKernel32ApiFactory; | ||||
| import com.mingliqiye.utils.system.SystemUtil; | ||||
| import lombok.Getter; | ||||
| import lombok.Setter; | ||||
| import lombok.val; | ||||
| import lombok.var; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.slf4j.Logger; | ||||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.time.Instant; | ||||
| import java.time.LocalDateTime; | ||||
| @ -31,10 +39,10 @@ import java.time.ZoneId; | ||||
| import java.time.format.DateTimeFormatter; | ||||
| import java.time.temporal.ChronoUnit; | ||||
| import java.util.Date; | ||||
| import lombok.Getter; | ||||
| import lombok.Setter; | ||||
| import lombok.var; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| import static com.mingliqiye.utils.jna.WinKernel32ApiFactory.FILETIME_EPOCH_OFFSET; | ||||
| import static com.mingliqiye.utils.jna.WinKernel32ApiFactory.NANOS_PER_100NS; | ||||
| import static com.mingliqiye.utils.logger.Loggers.getMingLiLoggerFactory; | ||||
| 
 | ||||
| /** | ||||
|  * 时间类,用于处理日期时间的转换、格式化等操作。 | ||||
| @ -54,15 +62,38 @@ import org.jetbrains.annotations.NotNull; | ||||
| @Setter | ||||
| public final class DateTime implements Serializable { | ||||
| 
 | ||||
| 	private static final WinKernel32 WIN_KERNEL_32; | ||||
| 	private static final long FILETIME_EPOCH_OFFSET = -116444736000000000L; | ||||
| 	private static final long NANOS_PER_100NS = 100; | ||||
| 	private static final WinKernel32Api WIN_KERNEL_32_API; | ||||
| 
 | ||||
| 	static { | ||||
| 		if (!SystemUtil.isJdk8Plus() && SystemUtil.isWindows()) { | ||||
| 			WIN_KERNEL_32 = WinKernel32.load(); | ||||
| 		if (SystemUtil.getJavaVersionAsInteger() == 8 && SystemUtil.isWindows()) { | ||||
| 
 | ||||
| 			final Logger log = getMingLiLoggerFactory().getLogger( | ||||
| 				"mingli-utils DateTime" | ||||
| 			); | ||||
| 			val a = WinKernel32ApiFactory.getWinKernel32Apis(); | ||||
| 
 | ||||
| 			if (a.size() > 1) { | ||||
| 				log.warn( | ||||
| 					"Multiple Size:{} WinKernel32Api implementations found.", | ||||
| 					a.size() | ||||
| 				); | ||||
| 				a.forEach(api -> | ||||
| 					log.warn( | ||||
| 						"Found WinKernel32Api: {}", | ||||
| 						api.getClass().getName() | ||||
| 					) | ||||
| 				); | ||||
| 			} | ||||
| 
 | ||||
| 			if (a.isEmpty()) { | ||||
| 				WIN_KERNEL_32_API = null; | ||||
| 				log.warn("No WinKernel32Api implementation found. Use Jdk1.8 LocalDateTime"); | ||||
| 			} else { | ||||
| 				WIN_KERNEL_32_API = a.get(a.size() - 1); | ||||
| 				log.info("Found and Use WinKernel32Api: {}", WIN_KERNEL_32_API.getClass().getName()); | ||||
| 			} | ||||
| 		} else { | ||||
| 			WIN_KERNEL_32 = null; | ||||
| 			WIN_KERNEL_32_API = null; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| @ -95,26 +126,9 @@ public final class DateTime implements Serializable { | ||||
| 	 * @return 返回当前时间的 DateTime 实例 | ||||
| 	 */ | ||||
| 	public static DateTime now() { | ||||
| 		if (WIN_KERNEL_32 != null) { | ||||
| 			byte[] fileTimeBuffer = new byte[8]; | ||||
| 			WIN_KERNEL_32.GetSystemTimePreciseAsFileTime(fileTimeBuffer); | ||||
| 			long fileTime = | ||||
| 				(long) (fileTimeBuffer[0] & 0xFF) | | ||||
| 				((long) (fileTimeBuffer[1] & 0xFF) << 8) | | ||||
| 				((long) (fileTimeBuffer[2] & 0xFF) << 16) | | ||||
| 				((long) (fileTimeBuffer[3] & 0xFF) << 24) | | ||||
| 				((long) (fileTimeBuffer[4] & 0xFF) << 32) | | ||||
| 				((long) (fileTimeBuffer[5] & 0xFF) << 40) | | ||||
| 				((long) (fileTimeBuffer[6] & 0xFF) << 48) | | ||||
| 				((long) (fileTimeBuffer[7] & 0xFF) << 56); | ||||
| 			long unixNanos = | ||||
| 				(fileTime + FILETIME_EPOCH_OFFSET) * NANOS_PER_100NS; | ||||
| 			Instant instant = Instant.ofEpochSecond( | ||||
| 				unixNanos / 1_000_000_000L, | ||||
| 				unixNanos % 1_000_000_000L | ||||
| 			); | ||||
| 		if (WIN_KERNEL_32_API != null) { | ||||
| 			return DateTime.of( | ||||
| 				instant.atZone(ZoneId.systemDefault()).toLocalDateTime() | ||||
| 				WIN_KERNEL_32_API.getTime().atZone(ZoneId.systemDefault()).toLocalDateTime() | ||||
| 			); | ||||
| 		} | ||||
| 		return new DateTime(); | ||||
|  | ||||
| @ -1,88 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile MysqlUUIDv1.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid; | ||||
| 
 | ||||
| import lombok.var; | ||||
| 
 | ||||
| /** | ||||
|  * MySQL UUID格式与标准UUID格式相互转换工具类 | ||||
|  * <p> | ||||
|  * MySQL使用不同的字节顺序存储UUID,此类提供了在MySQL格式和标准UUID格式之间转换的方法 | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| public class MysqlUUIDv1 { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将标准UUID格式转换为MySQL格式的UUID | ||||
| 	 * | ||||
| 	 * @param uuid 标准UUID格式的字节数组,长度必须为16字节 | ||||
| 	 * @return MySQL格式的UUID字节数组,长度为16字节 | ||||
| 	 */ | ||||
| 	public static byte[] uuidToMysql(byte[] uuid) { | ||||
| 		var reuuid = new byte[16]; | ||||
| 		// 转换时间戳低位部分 | ||||
| 		reuuid[4] = uuid[0]; | ||||
| 		reuuid[5] = uuid[1]; | ||||
| 		reuuid[6] = uuid[2]; | ||||
| 		reuuid[7] = uuid[3]; | ||||
| 
 | ||||
| 		// 转换时间戳中位部分 | ||||
| 		reuuid[2] = uuid[4]; | ||||
| 		reuuid[3] = uuid[5]; | ||||
| 
 | ||||
| 		// 转换时间戳高位部分 | ||||
| 		reuuid[0] = uuid[6]; | ||||
| 		reuuid[1] = uuid[7]; | ||||
| 
 | ||||
| 		// 复制时钟序列和节点标识部分 | ||||
| 		System.arraycopy(uuid, 8, reuuid, 8, 8); | ||||
| 		return reuuid; | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 将MySQL格式的UUID转换为标准UUID格式 | ||||
| 	 * | ||||
| 	 * @param uuid MySQL格式的UUID字节数组,长度必须为16字节 | ||||
| 	 * @return 标准UUID格式的字节数组,长度为16字节 | ||||
| 	 */ | ||||
| 	public static byte[] mysqlToUuid(byte[] uuid) { | ||||
| 		var reuuid = new byte[16]; | ||||
| 		// 转换时间戳高位部分 | ||||
| 		reuuid[6] = uuid[0]; | ||||
| 		reuuid[7] = uuid[1]; | ||||
| 
 | ||||
| 		// 转换时间戳中位部分 | ||||
| 		reuuid[4] = uuid[2]; | ||||
| 		reuuid[5] = uuid[3]; | ||||
| 
 | ||||
| 		// 转换时间戳低位部分 | ||||
| 		reuuid[0] = uuid[4]; | ||||
| 		reuuid[1] = uuid[5]; | ||||
| 		reuuid[2] = uuid[6]; | ||||
| 		reuuid[3] = uuid[7]; | ||||
| 
 | ||||
| 		// 复制时钟序列和节点标识部分 | ||||
| 		System.arraycopy(uuid, 8, reuuid, 8, 8); | ||||
| 		return reuuid; | ||||
| 	} | ||||
| } | ||||
| @ -1,34 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUIDException.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid; | ||||
| 
 | ||||
| public class UUIDException extends RuntimeException { | ||||
| 
 | ||||
| 	public UUIDException(String message) { | ||||
| 		super(message); | ||||
| 	} | ||||
| 
 | ||||
| 	public UUIDException(String message, Throwable cause) { | ||||
| 		super(message, cause); | ||||
| 	} | ||||
| } | ||||
| @ -1,53 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUIDConverter.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid.typehandlers; | ||||
| 
 | ||||
| import com.mingliqiye.utils.uuid.MysqlUUIDv1; | ||||
| import com.mingliqiye.utils.uuid.UUID; | ||||
| 
 | ||||
| public class UUIDConverter { | ||||
| 
 | ||||
| 	public static String UUID_TO_STR(UUID uuid) { | ||||
| 		return uuid.toUUIDString(); | ||||
| 	} | ||||
| 
 | ||||
| 	public static UUID STR_TO_UUID(String string) { | ||||
| 		return UUID.of(string); | ||||
| 	} | ||||
| 
 | ||||
| 	public static byte[] UUID_TO_BIN(UUID uuid) { | ||||
| 		return uuid.toBytes(); | ||||
| 	} | ||||
| 
 | ||||
| 	public static UUID BIN_TO_UUID(byte[] bytes) { | ||||
| 		return UUID.of(bytes); | ||||
| 	} | ||||
| 
 | ||||
| 	public static byte[] MYSQL_UUID_TO_BIN(UUID uuid) { | ||||
| 		return MysqlUUIDv1.uuidToMysql(uuid.toBytes()); | ||||
| 	} | ||||
| 
 | ||||
| 	public static UUID BIN_TO_MYSQL_UUID(byte[] bytes) { | ||||
| 		return UUID.of(MysqlUUIDv1.mysqlToUuid(bytes)); | ||||
| 	} | ||||
| } | ||||
| @ -1,106 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUIDBinaryTypeHandler.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid.typehandlers.mysql; | ||||
| 
 | ||||
| import com.mingliqiye.utils.uuid.UUID; | ||||
| import com.mingliqiye.utils.uuid.typehandlers.UUIDConverter; | ||||
| import java.sql.CallableStatement; | ||||
| import java.sql.PreparedStatement; | ||||
| import java.sql.ResultSet; | ||||
| import java.sql.SQLException; | ||||
| import org.apache.ibatis.type.BaseTypeHandler; | ||||
| import org.apache.ibatis.type.JdbcType; | ||||
| import org.apache.ibatis.type.MappedJdbcTypes; | ||||
| import org.apache.ibatis.type.MappedTypes; | ||||
| 
 | ||||
| /** | ||||
|  * UUIDBinaryTypeHandler 类用于处理 UUID 类型与数据库 BINARY 类型之间的转换 | ||||
|  * 该类继承自 BaseTypeHandler,专门处理 UUID 对象的序列化和反序列化 | ||||
|  * | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @MappedTypes({ UUID.class }) | ||||
| @MappedJdbcTypes(JdbcType.BINARY) | ||||
| public class UUIDBinaryTypeHandler extends BaseTypeHandler<UUID> { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 设置非空参数到 PreparedStatement 中 | ||||
| 	 * | ||||
| 	 * @param ps        PreparedStatement 对象 | ||||
| 	 * @param i         参数在 SQL 语句中的位置索引 | ||||
| 	 * @param parameter 要设置的 UUID 参数值 | ||||
| 	 * @param jdbcType  JDBC 类型信息 | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public void setNonNullParameter( | ||||
| 		PreparedStatement ps, | ||||
| 		int i, | ||||
| 		UUID parameter, | ||||
| 		JdbcType jdbcType | ||||
| 	) throws SQLException { | ||||
| 		ps.setBytes(i, UUIDConverter.UUID_TO_BIN(parameter)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 ResultSet 中根据列名获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param rs         ResultSet 对象 | ||||
| 	 * @param columnName 数据库列名 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(ResultSet rs, String columnName) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.BIN_TO_UUID(rs.getBytes(columnName)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 ResultSet 中根据列索引获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param rs          ResultSet 对象 | ||||
| 	 * @param columnIndex 数据库列索引 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(ResultSet rs, int columnIndex) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.BIN_TO_UUID(rs.getBytes(columnIndex)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 CallableStatement 中根据参数索引获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param cs          CallableStatement 对象 | ||||
| 	 * @param columnIndex 参数索引 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(CallableStatement cs, int columnIndex) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.BIN_TO_UUID(cs.getBytes(columnIndex)); | ||||
| 	} | ||||
| } | ||||
| @ -1,106 +0,0 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUIDStringTypeHandler.java | ||||
|  * LastUpdate 2025-09-09 08:37:33 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid.typehandlers.pgsql; | ||||
| 
 | ||||
| import com.mingliqiye.utils.uuid.UUID; | ||||
| import com.mingliqiye.utils.uuid.typehandlers.UUIDConverter; | ||||
| import java.sql.CallableStatement; | ||||
| import java.sql.PreparedStatement; | ||||
| import java.sql.ResultSet; | ||||
| import java.sql.SQLException; | ||||
| import org.apache.ibatis.type.BaseTypeHandler; | ||||
| import org.apache.ibatis.type.JdbcType; | ||||
| import org.apache.ibatis.type.MappedJdbcTypes; | ||||
| import org.apache.ibatis.type.MappedTypes; | ||||
| 
 | ||||
| /** | ||||
|  * UUIDBinaryTypeHandler 类用于处理 UUID 类型与数据库 BINARY 类型之间的转换 | ||||
|  * 该类继承自 BaseTypeHandler,专门处理 UUID 对象的序列化和反序列化 | ||||
|  * | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @MappedTypes({ UUID.class }) | ||||
| @MappedJdbcTypes(JdbcType.VARCHAR) | ||||
| public class UUIDStringTypeHandler extends BaseTypeHandler<UUID> { | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 设置非空参数到 PreparedStatement 中 | ||||
| 	 * | ||||
| 	 * @param ps        PreparedStatement 对象 | ||||
| 	 * @param i         参数在 SQL 语句中的位置索引 | ||||
| 	 * @param parameter 要设置的 UUID 参数值 | ||||
| 	 * @param jdbcType  JDBC 类型信息 | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public void setNonNullParameter( | ||||
| 		PreparedStatement ps, | ||||
| 		int i, | ||||
| 		UUID parameter, | ||||
| 		JdbcType jdbcType | ||||
| 	) throws SQLException { | ||||
| 		ps.setString(i, UUIDConverter.UUID_TO_STR(parameter)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 ResultSet 中根据列名获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param rs         ResultSet 对象 | ||||
| 	 * @param columnName 数据库列名 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(ResultSet rs, String columnName) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.STR_TO_UUID(rs.getString(columnName)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 ResultSet 中根据列索引获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param rs          ResultSet 对象 | ||||
| 	 * @param columnIndex 数据库列索引 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(ResultSet rs, int columnIndex) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.STR_TO_UUID(rs.getString(columnIndex)); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * 从 CallableStatement 中根据参数索引获取可为空的 UUID 结果 | ||||
| 	 * | ||||
| 	 * @param cs          CallableStatement 对象 | ||||
| 	 * @param columnIndex 参数索引 | ||||
| 	 * @return 转换后的 UUID 对象,可能为 null | ||||
| 	 * @throws SQLException 当数据库操作发生错误时抛出 | ||||
| 	 */ | ||||
| 	@Override | ||||
| 	public UUID getNullableResult(CallableStatement cs, int columnIndex) | ||||
| 		throws SQLException { | ||||
| 		return UUIDConverter.STR_TO_UUID(cs.getString(columnIndex)); | ||||
| 	} | ||||
| } | ||||
| @ -16,7 +16,7 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Main.kt | ||||
|  * LastUpdate 2025-09-12 17:11:48 | ||||
|  * LastUpdate 2025-09-14 20:08:44 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| @ -24,12 +24,11 @@ | ||||
| 
 | ||||
| package com.mingliqiye.utils | ||||
| 
 | ||||
| import com.mingliqiye.utils.path.OsPath | ||||
| import com.mingliqiye.utils.springboot.autoconfigure.AutoConfiguration | ||||
| 
 | ||||
| 
 | ||||
| fun main() { | ||||
|     AutoConfiguration.printBanner() | ||||
|     val path = OsPath.getCwd() | ||||
|     println(path.toAbsolutePath()) | ||||
| 
 | ||||
|     0..10 | ||||
| } | ||||
|  | ||||
							
								
								
									
										124
									
								
								src/main/kotlin/com/mingliqiye/utils/aes/AesUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								src/main/kotlin/com/mingliqiye/utils/aes/AesUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,124 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile AesUtils.kt | ||||
|  * LastUpdate 2025-09-14 18:43:04 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| 
 | ||||
| @file:JvmName("AesUtils") | ||||
| 
 | ||||
| package com.mingliqiye.utils.aes | ||||
| 
 | ||||
| import com.mingliqiye.utils.base64.decode | ||||
| import com.mingliqiye.utils.base64.encode | ||||
| import java.nio.charset.StandardCharsets | ||||
| import java.security.GeneralSecurityException | ||||
| import java.security.MessageDigest | ||||
| import java.security.SecureRandom | ||||
| import javax.crypto.Cipher | ||||
| import javax.crypto.spec.GCMParameterSpec | ||||
| import javax.crypto.spec.SecretKeySpec | ||||
| 
 | ||||
| const val ALGORITHM = "AES" | ||||
| const val TRANSFORMATION = "AES/GCM/NoPadding" | ||||
| const val GCM_IV_LENGTH = 12 | ||||
| const val GCM_TAG_LENGTH = 16 | ||||
| val SECURE_RANDOM = SecureRandom() | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * AES加密方法(使用GCM模式) | ||||
|  * @param sSrc 待加密的字符串 | ||||
|  * @param sKey 加密密钥 | ||||
|  * @return 加密后的字符串,格式为 IV:EncryptedData+Tag(均为Base64编码) | ||||
|  * @throws GeneralSecurityException 加密错误 | ||||
|  */ | ||||
| @Throws(GeneralSecurityException::class) | ||||
| fun encrypt(sSrc: String, sKey: String?): String? { | ||||
|     if (sKey == null) { | ||||
|         return null | ||||
|     } | ||||
| 
 | ||||
|     // 生成密钥 | ||||
|     val secretKeySpec = createSecretKey(sKey) | ||||
| 
 | ||||
|     // 生成安全随机IV | ||||
|     val iv = ByteArray(GCM_IV_LENGTH) | ||||
|     SECURE_RANDOM.nextBytes(iv) | ||||
| 
 | ||||
|     // 初始化加密器 | ||||
|     val cipher = Cipher.getInstance(TRANSFORMATION) | ||||
|     val gcmParameterSpec = GCMParameterSpec( | ||||
|         GCM_TAG_LENGTH * 8, iv | ||||
|     ) | ||||
|     cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, gcmParameterSpec) | ||||
| 
 | ||||
|     val encrypted = cipher.doFinal( | ||||
|         sSrc.toByteArray(StandardCharsets.UTF_8) | ||||
|     ) | ||||
|     return encode( | ||||
|         "${encode(iv)}:${encode(encrypted)}".toByteArray() | ||||
|     ) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * AES解密方法(使用GCM模式) | ||||
|  * @param sSrc 待解密的字符串,格式为 IV:EncryptedData+Tag(均为Base64编码) | ||||
|  * @param sKey 解密密钥 | ||||
|  * @return 解密后的原始字符串 | ||||
|  */ | ||||
| fun decrypt(sSrc: String, sKey: String): String? { | ||||
|     try { | ||||
|         // 分割IV和加密数据 | ||||
|         val sSrcs = String(decode(sSrc)) | ||||
|         val parts: Array<String?> = sSrcs.split(":".toRegex(), limit = 2).toTypedArray() | ||||
|         if (parts.size != 2) { | ||||
|             return null | ||||
|         } | ||||
|         val iv = decode(parts[0]!!) | ||||
|         val encryptedData = decode(parts[1]!!) | ||||
|         if (iv.size != GCM_IV_LENGTH) { | ||||
|             return null | ||||
|         } | ||||
|         val secretKeySpec = createSecretKey(sKey) | ||||
|         val cipher = Cipher.getInstance(TRANSFORMATION) | ||||
|         val gcmParameterSpec = GCMParameterSpec( | ||||
|             GCM_TAG_LENGTH * 8, iv | ||||
|         ) | ||||
|         cipher.init(Cipher.DECRYPT_MODE, secretKeySpec, gcmParameterSpec) | ||||
|         val original = cipher.doFinal(encryptedData) | ||||
|         return String(original, StandardCharsets.UTF_8) | ||||
|     } catch (e: Exception) { | ||||
|         return null | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 创建AES密钥,支持任意长度的密钥 | ||||
|  * @param sKey 字符串密钥 | ||||
|  * @return SecretKeySpec对象 | ||||
|  * @throws Exception 可能抛出的异常 | ||||
|  */ | ||||
| @Throws(Exception::class) | ||||
| private fun createSecretKey(sKey: String): SecretKeySpec { | ||||
|     val key = sKey.toByteArray(StandardCharsets.UTF_8) | ||||
|     val md = MessageDigest.getInstance("MD5") | ||||
|     val digest = md.digest(key) | ||||
|     return SecretKeySpec(digest, ALGORITHM) | ||||
| } | ||||
							
								
								
									
										160
									
								
								src/main/kotlin/com/mingliqiye/utils/base64/Base64Utils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								src/main/kotlin/com/mingliqiye/utils/base64/Base64Utils.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,160 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Base64Utils.kt | ||||
|  * LastUpdate 2025-09-14 18:44:22 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("Base64Utils") | ||||
| 
 | ||||
| package com.mingliqiye.utils.base64 | ||||
| 
 | ||||
| import java.io.File | ||||
| import java.io.IOException | ||||
| import java.nio.file.Path | ||||
| import java.util.* | ||||
| 
 | ||||
| val BASE_64_ENCODER: Base64.Encoder = Base64.getEncoder() | ||||
| val BASE_64_DECODER: Base64.Decoder = Base64.getDecoder() | ||||
| 
 | ||||
| /** | ||||
|  * 将字节数组编码为Base64字符串 | ||||
|  * | ||||
|  * @param bytes 需要编码的字节数组 | ||||
|  * @return 编码后的Base64字符串 | ||||
|  */ | ||||
| fun encode(bytes: ByteArray): String { | ||||
|     return BASE_64_ENCODER.encodeToString(bytes) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将Base64字符串解码为字节数组 | ||||
|  * | ||||
|  * @param string 需要解码的Base64字符串 | ||||
|  * @return 解码后的字节数组 | ||||
|  */ | ||||
| fun decode(string: String): ByteArray { | ||||
|     return BASE_64_DECODER.decode(string) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将文件内容编码为Base64字符串 | ||||
|  * | ||||
|  * @param file 需要编码的文件 | ||||
|  * @return 文件内容的Base64编码字符串 | ||||
|  * @throws IOException 当文件读取失败时抛出 | ||||
|  */ | ||||
| @Throws(IOException::class) | ||||
| fun encode(file: File): String { | ||||
|     return encode(file.readBytes()) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将Base64字符串解码并写入文件 | ||||
|  * | ||||
|  * @param file 目标文件 | ||||
|  * @param string 需要解码的Base64字符串 | ||||
|  * @throws IOException 当文件写入失败时抛出 | ||||
|  */ | ||||
| @Throws(IOException::class) | ||||
| fun decode(file: File, string: String) { | ||||
|     file.writeBytes(decode(string)) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 安全地将文件内容编码为Base64字符串,出现异常时返回null | ||||
|  * | ||||
|  * @param file 需要编码的文件 | ||||
|  * @return 文件内容的Base64编码字符串,失败时返回null | ||||
|  */ | ||||
| fun encodeSafe(file: File): String? { | ||||
|     return try { | ||||
|         encode(file) | ||||
|     } catch (_: Exception) { | ||||
|         null | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 安全地将Base64字符串解码并写入文件,返回操作是否成功 | ||||
|  * | ||||
|  * @param file 目标文件 | ||||
|  * @param string 需要解码的Base64字符串 | ||||
|  * @return 操作成功返回true,失败返回false | ||||
|  */ | ||||
| fun decodeSafe(file: File, string: String): Boolean { | ||||
|     return try { | ||||
|         decode(file, string) | ||||
|         true | ||||
|     } catch (_: Exception) { | ||||
|         false | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将路径对应的文件内容编码为Base64字符串 | ||||
|  * | ||||
|  * @param path 需要编码的文件路径 | ||||
|  * @return 文件内容的Base64编码字符串 | ||||
|  * @throws IOException 当文件读取失败时抛出 | ||||
|  */ | ||||
| @Throws(IOException::class) | ||||
| fun encode(path: Path): String { | ||||
|     return encode(path.toFile().readBytes()) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将Base64字符串解码并写入路径指定的文件 | ||||
|  * | ||||
|  * @param path 目标文件路径 | ||||
|  * @param string 需要解码的Base64字符串 | ||||
|  * @throws IOException 当文件写入失败时抛出 | ||||
|  */ | ||||
| @Throws(IOException::class) | ||||
| fun decode(path: Path, string: String) { | ||||
|     path.toFile().writeBytes(decode(string)) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 安全地将路径对应的文件内容编码为Base64字符串,出现异常时返回null | ||||
|  * | ||||
|  * @param path 需要编码的文件路径 | ||||
|  * @return 文件内容的Base64编码字符串,失败时返回null | ||||
|  */ | ||||
| fun encodeSafe(path: Path): String? { | ||||
|     return try { | ||||
|         encode(path) | ||||
|     } catch (_: Exception) { | ||||
|         null | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 安全地将Base64字符串解码并写入路径指定的文件,返回操作是否成功 | ||||
|  * | ||||
|  * @param path 目标文件路径 | ||||
|  * @param string 需要解码的Base64字符串 | ||||
|  * @return 操作成功返回true,失败返回false | ||||
|  */ | ||||
| fun decodeSafe(path: Path, string: String): Boolean { | ||||
|     return try { | ||||
|         decode(path, string) | ||||
|         true | ||||
|     } catch (_: Exception) { | ||||
|         false | ||||
|     } | ||||
| } | ||||
							
								
								
									
										241
									
								
								src/main/kotlin/com/mingliqiye/utils/bean/Factory.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										241
									
								
								src/main/kotlin/com/mingliqiye/utils/bean/Factory.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,241 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Factory.kt | ||||
|  * LastUpdate 2025-09-14 19:09:28 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("Factory") | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean | ||||
| 
 | ||||
| import com.mingliqiye.utils.bean.annotation.ComponentBean | ||||
| import com.mingliqiye.utils.bean.annotation.InjectBean | ||||
| import java.io.File | ||||
| import java.net.URL | ||||
| import java.util.* | ||||
| import java.util.concurrent.ConcurrentHashMap | ||||
| import kotlin.reflect.KClass | ||||
| 
 | ||||
| /** | ||||
|  * 存储所有已注册的Bean实例,键为Bean名称,值为Bean实例 | ||||
|  */ | ||||
| private val BEANS: ConcurrentHashMap<String, Any> = ConcurrentHashMap() | ||||
| 
 | ||||
| /** | ||||
|  * 存储按类型查找的Bean实例,键为Bean的Class对象,值为Bean实例 | ||||
|  */ | ||||
| private val TYPE_BEANS: ConcurrentHashMap<KClass<*>, Any> = ConcurrentHashMap() | ||||
| 
 | ||||
| /** | ||||
|  * 自动扫描指定类所在包下的所有类并注册为Bean | ||||
|  * | ||||
|  * @param c 指定的类,用于获取其所在的包 | ||||
|  * @throws IllegalArgumentException 如果传入的类为null或位于默认包中 | ||||
|  */ | ||||
| fun autoScan(c: Class<*>?) { | ||||
|     if (c == null) { | ||||
|         throw IllegalArgumentException("Class cannot be null") | ||||
|     } | ||||
|     val pkg = c.`package` | ||||
|     if (pkg == null) { | ||||
|         throw IllegalArgumentException("Class is in the default package") | ||||
|     } | ||||
|     scan(pkg.name) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 扫描指定包路径下的所有类文件,并注册其中带有@ComponentBean注解的类为Bean | ||||
|  * | ||||
|  * @param basePackage 要扫描的基础包名 | ||||
|  * @throws RuntimeException 如果在扫描过程中发生异常 | ||||
|  */ | ||||
| fun scan(basePackage: String) { | ||||
|     try { | ||||
|         val path = basePackage.replace('.', '/') | ||||
|         val classLoader = Thread.currentThread().contextClassLoader | ||||
|         val resources: Enumeration<URL> = classLoader.getResources(path) | ||||
|         while (resources.hasMoreElements()) { | ||||
|             val resource = resources.nextElement() | ||||
|             val file = File(resource.toURI()) | ||||
|             scanDirectory(file, basePackage) | ||||
|         } | ||||
|         injectDependencies() | ||||
|     } catch (e: Exception) { | ||||
|         throw RuntimeException(e) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 递归扫描目录中的所有类文件,并注册符合条件的类为Bean | ||||
|  * | ||||
|  * @param directory   当前要扫描的目录 | ||||
|  * @param packageName 当前目录对应的包名 | ||||
|  * @throws Exception 如果在扫描或类加载过程中发生异常 | ||||
|  */ | ||||
| private fun scanDirectory(directory: File, packageName: String) { | ||||
|     val files = directory.listFiles() ?: return | ||||
| 
 | ||||
|     for (file in files) { | ||||
|         if (file.isDirectory) { | ||||
|             scanDirectory(file, "$packageName.${file.name}") | ||||
|         } else if (file.name.endsWith(".class")) { | ||||
|             val className = packageName + '.' + file.name.replace(".class", "") | ||||
|             registerComponent(Class.forName(className)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 注册一个带有@ComponentBean注解的类为Bean实例 | ||||
|  * | ||||
|  * @param clazz 要注册的类 | ||||
|  * @throws Exception 如果在实例化类或处理注解时发生异常 | ||||
|  */ | ||||
| private fun registerComponent(clazz: Class<*>) { | ||||
|     if (clazz.isAnnotationPresent(ComponentBean::class.java)) { | ||||
|         val component = clazz.getAnnotation(ComponentBean::class.java) | ||||
|         val name = component.value.ifEmpty { clazz.name } | ||||
|         val instance = clazz.getDeclaredConstructor().newInstance() | ||||
|         BEANS[name] = instance | ||||
|         val kClass = clazz.kotlin | ||||
|         TYPE_BEANS[kClass] = instance | ||||
| 
 | ||||
|         for (interfaceClass in clazz.interfaces) { | ||||
|             TYPE_BEANS.putIfAbsent(interfaceClass.kotlin, instance) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 对所有已注册的Bean进行依赖注入处理 | ||||
|  * | ||||
|  * @throws Exception 如果在注入过程中发生异常 | ||||
|  */ | ||||
| private fun injectDependencies() { | ||||
|     for (bean in BEANS.values) { | ||||
|         for (field in bean.javaClass.declaredFields) { | ||||
|             if (field.isAnnotationPresent(InjectBean::class.java)) { | ||||
|                 val inject = field.getAnnotation(InjectBean::class.java) | ||||
|                 val dependency = findDependency(field.type, inject.value) | ||||
|                 if (dependency == null) { | ||||
|                     throw IllegalStateException( | ||||
|                         "No suitable dependency found for field " + field.name + " in class " + bean.javaClass.name | ||||
|                     ) | ||||
|                 } | ||||
|                 field.isAccessible = true | ||||
|                 field.set(bean, dependency) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 根据类型和名称查找对应的依赖实例 | ||||
|  * | ||||
|  * @param type 依赖的类型 | ||||
|  * @param name 依赖的名称(可为空) | ||||
|  * @return 找到的依赖实例,未找到则返回null | ||||
|  */ | ||||
| private fun findDependency(type: Class<*>, name: String): Any? { | ||||
|     if (name.isNotEmpty()) { | ||||
|         return BEANS[name] | ||||
|     } | ||||
| 
 | ||||
|     val dependency = TYPE_BEANS[type.kotlin] | ||||
|     if (dependency != null) { | ||||
|         return dependency | ||||
|     } | ||||
| 
 | ||||
|     for (interfaceType in TYPE_BEANS.keys) { | ||||
|         if (type.isAssignableFrom(interfaceType.java)) { | ||||
|             return TYPE_BEANS[interfaceType] | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return null | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将一个对象添加到Bean容器中,使用其类名作为键 | ||||
|  * | ||||
|  * @param obj 要添加的对象 | ||||
|  * @throws RuntimeException 如果在注入依赖时发生异常 | ||||
|  */ | ||||
| fun add(obj: Any) { | ||||
|     val clazz = obj.javaClass | ||||
|     val name = clazz.name | ||||
|     BEANS[name] = obj | ||||
|     TYPE_BEANS[clazz.kotlin] = obj | ||||
|     try { | ||||
|         injectDependencies() | ||||
|     } catch (e: Exception) { | ||||
|         throw RuntimeException(e) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将一个对象以指定名称添加到Bean容器中 | ||||
|  * | ||||
|  * @param name   Bean的名称 | ||||
|  * @param obj 要添加的对象 | ||||
|  * @throws RuntimeException 如果在注入依赖时发生异常 | ||||
|  */ | ||||
| fun add(name: String, obj: Any) { | ||||
|     BEANS[name] = obj | ||||
|     TYPE_BEANS[obj.javaClass.kotlin] = obj | ||||
|     try { | ||||
|         injectDependencies() | ||||
|     } catch (e: Exception) { | ||||
|         throw RuntimeException(e) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 根据类型获取对应的Bean实例 | ||||
|  * | ||||
|  * @param objclass Bean的类型 | ||||
|  * @param T      Bean的泛型类型 | ||||
|  * @return 对应类型的Bean实例,未找到则返回null | ||||
|  */ | ||||
| @Suppress("UNCHECKED_CAST") | ||||
| fun <T : Any> get(objclass: KClass<T>): T? { | ||||
|     return TYPE_BEANS[objclass] as? T | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 根据名称和类型获取对应的Bean实例 | ||||
|  * | ||||
|  * @param name     Bean的名称 | ||||
|  * @param objclass Bean的类型 | ||||
|  * @param T      Bean的泛型类型 | ||||
|  * @return 对应名称和类型的Bean实例,未找到则返回null | ||||
|  */ | ||||
| @Suppress("UNCHECKED_CAST") | ||||
| fun <T : Any> get(name: String, objclass: KClass<T>): T? { | ||||
|     return BEANS[name] as? T | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 根据名称获取对应的Bean实例 | ||||
|  * | ||||
|  * @param name Bean的名称 | ||||
|  * @return 对应名称的Bean实例,未找到则返回null | ||||
|  */ | ||||
| fun get(name: String): Any? { | ||||
|     return BEANS[name] | ||||
| } | ||||
| @ -15,23 +15,29 @@ | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile InjectBean.java | ||||
|  * LastUpdate 2025-09-09 08:37:34 | ||||
|  * CurrentFile ComponentBean.kt | ||||
|  * LastUpdate 2025-09-14 18:48:59 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean.annotation; | ||||
| package com.mingliqiye.utils.bean.annotation | ||||
| 
 | ||||
| import java.lang.annotation.ElementType; | ||||
| import java.lang.annotation.Retention; | ||||
| import java.lang.annotation.RetentionPolicy; | ||||
| import java.lang.annotation.Target; | ||||
| import kotlin.annotation.AnnotationRetention.RUNTIME | ||||
| import kotlin.annotation.AnnotationTarget.CLASS | ||||
| import kotlin.annotation.AnnotationTarget.FIELD | ||||
| 
 | ||||
| /** | ||||
|  * 组件bean注解 | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @Retention(RetentionPolicy.RUNTIME) | ||||
| @Target(ElementType.FIELD) | ||||
| public @interface InjectBean { | ||||
| 	String value() default ""; | ||||
| } | ||||
| @Retention(RUNTIME) | ||||
| @Target(CLASS) | ||||
| annotation class ComponentBean(val value: String = "") | ||||
| 
 | ||||
| /** | ||||
|  * 注入bean注解 | ||||
|  * @author MingLiPro | ||||
|  */ | ||||
| @Retention(RUNTIME) | ||||
| @Target(FIELD) | ||||
| annotation class InjectBean(val value: String = "") | ||||
| @ -0,0 +1,96 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile SpringBeanUtils.kt | ||||
|  * LastUpdate 2025-09-14 20:01:26 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.bean.annotation.springboot | ||||
| 
 | ||||
| import org.springframework.beans.BeansException | ||||
| import org.springframework.context.ApplicationContext | ||||
| import org.springframework.context.ApplicationContextAware | ||||
| import org.springframework.stereotype.Component | ||||
| 
 | ||||
| /** | ||||
|  * Spring Bean工具类,用于获取Spring容器中的Bean实例 | ||||
|  * 实现ApplicationContextAware接口以获取Spring的应用上下文 | ||||
|  */ | ||||
| @Component | ||||
| class SpringBeanUtils : ApplicationContextAware { | ||||
| 
 | ||||
|     companion object { | ||||
| 
 | ||||
|         @JvmStatic | ||||
|         private var applicationContext: ApplicationContext? = null | ||||
| 
 | ||||
|         /** | ||||
|          * 根据Bean名称获取Bean实例 | ||||
|          * | ||||
|          * @param name Bean的名称 | ||||
|          * @return 指定名称的Bean实例 | ||||
|          * @throws BeansException 当获取Bean失败时抛出 | ||||
|          * @throws ClassCastException 当类型转换失败时抛出 | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         @Throws(BeansException::class, ClassCastException::class) | ||||
|         @Suppress("UNCHECKED_CAST") | ||||
|         fun <T> getBean(name: String): T { | ||||
|             return applicationContext!!.getBean(name) as T | ||||
|         } | ||||
| 
 | ||||
|         /** | ||||
|          * 根据Bean类型获取Bean实例 | ||||
|          * | ||||
|          * @param clazz Bean的类型 | ||||
|          * @return 指定类型的Bean实例 | ||||
|          * @throws BeansException 当获取Bean失败时抛出 | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         @Throws(BeansException::class) | ||||
|         fun <T> getBean(clazz: Class<T>): T { | ||||
|             return applicationContext!!.getBean(clazz) | ||||
|         } | ||||
| 
 | ||||
|         /** | ||||
|          * 根据Bean名称和类型获取Bean实例 | ||||
|          * | ||||
|          * @param name Bean的名称 | ||||
|          * @param clazz Bean的类型 | ||||
|          * @return 指定名称和类型的Bean实例 | ||||
|          * @throws BeansException 当获取Bean失败时抛出 | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         @Throws(BeansException::class) | ||||
|         fun <T> getBean(name: String, clazz: Class<T>): T { | ||||
|             return applicationContext!!.getBean<T>(name, clazz) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     /** | ||||
|      * 设置Spring的应用上下文 | ||||
|      * | ||||
|      * @param applicationContext Spring的应用上下文 | ||||
|      * @throws BeansException 当设置应用上下文失败时抛出 | ||||
|      */ | ||||
|     @Throws(BeansException::class) | ||||
|     override fun setApplicationContext(applicationContext: ApplicationContext) { | ||||
|         SpringBeanUtils.applicationContext = applicationContext | ||||
|     } | ||||
| } | ||||
							
								
								
									
										49
									
								
								src/main/kotlin/com/mingliqiye/utils/bytes/ByteUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								src/main/kotlin/com/mingliqiye/utils/bytes/ByteUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,49 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile ByteUtils.kt | ||||
|  * LastUpdate 2025-09-14 19:28:38 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("ByteUtils") | ||||
| 
 | ||||
| package com.mingliqiye.utils.bytes | ||||
| 
 | ||||
| import com.mingliqiye.utils.collection.Lists | ||||
| import com.mingliqiye.utils.stream.SuperStream | ||||
| 
 | ||||
| const val ESC_ASC: Byte = 0x10 | ||||
| const val ESC_DESC: Byte = 0x1B | ||||
| const val ESC_NONE: Byte = 0x00 | ||||
| const val ESC_START: Byte = 0x01 | ||||
| const val ESC_END: Byte = 0x02 | ||||
| const val ESC_ESC: Byte = 0x03 | ||||
| const val ESC_CONTROL: Byte = 0x04 | ||||
| const val ESC_DATA: Byte = 0x05 | ||||
| const val ESC_RESERVED: Byte = 0x06 | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * 将字节数组转换为十六进制字符串列表 | ||||
|  * @return 包含每个字节对应十六进制字符串的列表 | ||||
|  */ | ||||
| fun ByteArray.getByteArrayString(): MutableList<String> { | ||||
|     return Lists.toList(this)!!.stream() | ||||
|         .map { a -> String.format("0X%02X", a!!.toInt() and 0xFF) } | ||||
|         .collect(SuperStream.Collectors.toList()) | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										63
									
								
								src/main/kotlin/com/mingliqiye/utils/clone/CloneUtil.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								src/main/kotlin/com/mingliqiye/utils/clone/CloneUtil.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile CloneUtil.kt | ||||
|  * LastUpdate 2025-09-14 19:53:41 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("CloneUtils") | ||||
| 
 | ||||
| package com.mingliqiye.utils.clone | ||||
| 
 | ||||
| import com.mingliqiye.utils.json.JsonApi | ||||
| import com.mingliqiye.utils.json.JsonException | ||||
| import java.io.* | ||||
| 
 | ||||
| 
 | ||||
| inline fun <reified T> Serializable.deepClone(): T { | ||||
|     return deepClone(this) as T | ||||
| } | ||||
| 
 | ||||
| inline fun <reified T> T.deepJsonClone(jsonApi: JsonApi): T { | ||||
|     try { | ||||
|         return jsonApi.convert(this, this!!.javaClass) as T | ||||
| 
 | ||||
|     } catch (e: Exception) { | ||||
|         throw JsonException( | ||||
|             "Failed to deep clone object using JSON", | ||||
|             e | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @Suppress("UNCHECKED_CAST") | ||||
| fun <T : Serializable?> deepClone(obj: T): T { | ||||
|     try { | ||||
|         val bao = ByteArrayOutputStream() | ||||
|         val oos = ObjectOutputStream(bao) | ||||
|         oos.writeObject(obj) | ||||
|         val bis = ByteArrayInputStream( | ||||
|             bao.toByteArray() | ||||
|         ) | ||||
|         val ois = ObjectInputStream(bis) | ||||
|         return ois.readObject() as T | ||||
|     } catch (e: IOException) { | ||||
|         throw RuntimeException(e) | ||||
|     } catch (e: ClassNotFoundException) { | ||||
|         throw RuntimeException(e) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										60
									
								
								src/main/kotlin/com/mingliqiye/utils/jna/FieldStructure.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								src/main/kotlin/com/mingliqiye/utils/jna/FieldStructure.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,60 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile FieldStructure.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.jna | ||||
| 
 | ||||
| import com.sun.jna.Structure | ||||
| import java.lang.reflect.Modifier | ||||
| 
 | ||||
| /** | ||||
|  * JNA结构体基类,自动处理字段顺序 | ||||
|  * 该类继承自JNA的Structure类,通过反射自动获取子类的公共非静态字段, | ||||
|  * 并按声明顺序返回字段名列表,用于JNA结构体的字段映射。 | ||||
|  */ | ||||
| class FieldStructure : Structure() { | ||||
|     /** | ||||
|      * 获取结构体字段顺序列表 | ||||
|      * 通过反射遍历当前类及其父类的所有声明字段,过滤出公共非静态字段, | ||||
|      * 按照字段在类中声明的顺序返回字段名列表。 | ||||
|      * | ||||
|      * @return 包含字段名的列表,按声明顺序排列 | ||||
|      */ | ||||
|     override fun getFieldOrder(): MutableList<String> { | ||||
|         val fieldOrderList: MutableList<String> = ArrayList() | ||||
|         var cls: Class<*> = javaClass | ||||
|         while (cls != FieldStructure::class.java) { | ||||
|             val fields = cls.getDeclaredFields() | ||||
|             var modifiers: Int | ||||
|             for (field in fields) { | ||||
|                 modifiers = field.modifiers | ||||
|                 if (Modifier.isStatic(modifiers) || | ||||
|                     !Modifier.isPublic(modifiers) | ||||
|                 ) { | ||||
|                     continue | ||||
|                 } | ||||
|                 fieldOrderList.add(field.name) | ||||
|             } | ||||
|             cls = cls.getSuperclass() | ||||
|         } | ||||
|         return fieldOrderList | ||||
|     } | ||||
| } | ||||
							
								
								
									
										421
									
								
								src/main/kotlin/com/mingliqiye/utils/logger/Loggers.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										421
									
								
								src/main/kotlin/com/mingliqiye/utils/logger/Loggers.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,421 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile Loggers.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| @file:JvmName("Loggers") | ||||
| 
 | ||||
| package com.mingliqiye.utils.logger | ||||
| 
 | ||||
| 
 | ||||
| import org.slf4j.Logger | ||||
| import org.slf4j.LoggerFactory | ||||
| import org.slf4j.Marker | ||||
| import java.util.* | ||||
| 
 | ||||
| enum class MingLiLoggerLevel { | ||||
|     TRACE, | ||||
|     DEBUG, | ||||
|     INFO, | ||||
|     WARN, | ||||
|     ERROR | ||||
| } | ||||
| 
 | ||||
| class MingLiLogger : Logger { | ||||
|     override fun getName(): String { | ||||
|         return "MingLiLogger" | ||||
|     } | ||||
| 
 | ||||
|     override fun isTraceEnabled(): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(msg: String?) { | ||||
|         msg?.let { toPrintln(it, MingLiLoggerLevel.TRACE) } | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(format: String?, arg: Any?) { | ||||
|         format?.let { | ||||
|             val message = format1(it, arg) | ||||
|             toPrintln(message, MingLiLoggerLevel.TRACE) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(format: String?, arg1: Any?, arg2: Any?) { | ||||
|         format?.let { | ||||
|             val message = format2(it, arg1, arg2) | ||||
|             toPrintln(message, MingLiLoggerLevel.TRACE) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(format: String?, vararg arguments: Any?) { | ||||
|         format?.let { | ||||
|             val message = formatArray(it, arguments) | ||||
|             toPrintln(message, MingLiLoggerLevel.TRACE) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(msg: String?, t: Throwable?) { | ||||
|         msg?.let { | ||||
|             val message = if (t != null) "$it: ${t.message}" else it | ||||
|             toPrintln(message, MingLiLoggerLevel.TRACE) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun isTraceEnabled(marker: Marker?): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(marker: Marker?, msg: String?) { | ||||
|         trace(msg) | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(marker: Marker?, format: String?, arg: Any?) { | ||||
|         trace(format, arg) | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(marker: Marker?, format: String?, arg1: Any?, arg2: Any?) { | ||||
|         trace(format, arg1, arg2) | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(marker: Marker?, format: String?, vararg arguments: Any?) { | ||||
|         trace(format, *arguments) | ||||
|     } | ||||
| 
 | ||||
|     override fun trace(marker: Marker?, msg: String?, t: Throwable?) { | ||||
|         trace(msg, t) | ||||
|     } | ||||
| 
 | ||||
|     override fun isDebugEnabled(): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(msg: String?) { | ||||
|         msg?.let { toPrintln(it, MingLiLoggerLevel.DEBUG) } | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(format: String?, arg: Any?) { | ||||
|         format?.let { | ||||
|             val message = format1(it, arg) | ||||
|             toPrintln(message, MingLiLoggerLevel.DEBUG) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(format: String?, arg1: Any?, arg2: Any?) { | ||||
|         format?.let { | ||||
|             val message = format2(it, arg1, arg2) | ||||
|             toPrintln(message, MingLiLoggerLevel.DEBUG) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(format: String?, vararg arguments: Any?) { | ||||
|         format?.let { | ||||
|             val message = formatArray(it, arguments) | ||||
|             toPrintln(message, MingLiLoggerLevel.DEBUG) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(msg: String?, t: Throwable?) { | ||||
|         msg?.let { | ||||
|             val message = if (t != null) "$it: ${t.message}" else it | ||||
|             toPrintln(message, MingLiLoggerLevel.DEBUG) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun isDebugEnabled(marker: Marker?): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(marker: Marker?, msg: String?) { | ||||
|         debug(msg) | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(marker: Marker?, format: String?, arg: Any?) { | ||||
|         debug(format, arg) | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(marker: Marker?, format: String?, arg1: Any?, arg2: Any?) { | ||||
|         debug(format, arg1, arg2) | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(marker: Marker?, format: String?, vararg arguments: Any?) { | ||||
|         debug(format, *arguments) | ||||
|     } | ||||
| 
 | ||||
|     override fun debug(marker: Marker?, msg: String?, t: Throwable?) { | ||||
|         debug(msg, t) | ||||
|     } | ||||
| 
 | ||||
|     override fun isInfoEnabled(): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun info(msg: String?) { | ||||
|         msg?.let { toPrintln(it, MingLiLoggerLevel.INFO) } | ||||
|     } | ||||
| 
 | ||||
|     override fun info(format: String?, arg: Any?) { | ||||
|         format?.let { | ||||
|             val message = format1(it, arg) | ||||
|             toPrintln(message, MingLiLoggerLevel.INFO) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun info(format: String?, arg1: Any?, arg2: Any?) { | ||||
|         format?.let { | ||||
|             val message = format2(it, arg1, arg2) | ||||
|             toPrintln(message, MingLiLoggerLevel.INFO) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun info(format: String?, vararg arguments: Any?) { | ||||
|         format?.let { | ||||
|             val message = formatArray(it, arguments) | ||||
|             toPrintln(message, MingLiLoggerLevel.INFO) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun info(msg: String?, t: Throwable?) { | ||||
|         msg?.let { | ||||
|             val message = if (t != null) "$it: ${t.message}" else it | ||||
|             toPrintln(message, MingLiLoggerLevel.INFO) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun isInfoEnabled(marker: Marker?): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun info(marker: Marker?, msg: String?) { | ||||
|         info(msg) | ||||
|     } | ||||
| 
 | ||||
|     override fun info(marker: Marker?, format: String?, arg: Any?) { | ||||
|         info(format, arg) | ||||
|     } | ||||
| 
 | ||||
|     override fun info(marker: Marker?, format: String?, arg1: Any?, arg2: Any?) { | ||||
|         info(format, arg1, arg2) | ||||
|     } | ||||
| 
 | ||||
|     override fun info(marker: Marker?, format: String?, vararg arguments: Any?) { | ||||
|         info(format, *arguments) | ||||
|     } | ||||
| 
 | ||||
|     override fun info(marker: Marker?, msg: String?, t: Throwable?) { | ||||
|         info(msg, t) | ||||
|     } | ||||
| 
 | ||||
|     override fun isWarnEnabled(): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(msg: String?) { | ||||
|         msg?.let { toPrintln(it, MingLiLoggerLevel.WARN) } | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(format: String?, arg: Any?) { | ||||
|         format?.let { | ||||
|             val message = format1(it, arg) | ||||
|             toPrintln(message, MingLiLoggerLevel.WARN) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(format: String?, vararg arguments: Any?) { | ||||
|         format?.let { | ||||
|             val message = formatArray(it, arguments) | ||||
|             toPrintln(message, MingLiLoggerLevel.WARN) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(format: String?, arg1: Any?, arg2: Any?) { | ||||
|         format?.let { | ||||
|             val message = format2(it, arg1, arg2) | ||||
|             toPrintln(message, MingLiLoggerLevel.WARN) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(msg: String?, t: Throwable?) { | ||||
|         msg?.let { | ||||
|             val message = if (t != null) "$it: ${t.message}" else it | ||||
|             toPrintln(message, MingLiLoggerLevel.WARN) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun isWarnEnabled(marker: Marker?): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(marker: Marker?, msg: String?) { | ||||
|         warn(msg) | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(marker: Marker?, format: String?, arg: Any?) { | ||||
|         warn(format, arg) | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(marker: Marker?, format: String?, arg1: Any?, arg2: Any?) { | ||||
|         warn(format, arg1, arg2) | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(marker: Marker?, format: String?, vararg arguments: Any?) { | ||||
|         warn(format, *arguments) | ||||
|     } | ||||
| 
 | ||||
|     override fun warn(marker: Marker?, msg: String?, t: Throwable?) { | ||||
|         warn(msg, t) | ||||
|     } | ||||
| 
 | ||||
|     override fun isErrorEnabled(): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun error(msg: String?) { | ||||
|         msg?.let { toPrintln(it, MingLiLoggerLevel.ERROR) } | ||||
|     } | ||||
| 
 | ||||
|     override fun error(format: String?, arg: Any?) { | ||||
|         format?.let { | ||||
|             val message = format1(it, arg) | ||||
|             toPrintln(message, MingLiLoggerLevel.ERROR) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun error(format: String?, arg1: Any?, arg2: Any?) { | ||||
|         format?.let { | ||||
|             val message = format2(it, arg1, arg2) | ||||
|             toPrintln(message, MingLiLoggerLevel.ERROR) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun error(format: String?, vararg arguments: Any?) { | ||||
|         format?.let { | ||||
|             val message = formatArray(it, arguments) | ||||
|             toPrintln(message, MingLiLoggerLevel.ERROR) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun error(msg: String?, t: Throwable?) { | ||||
|         msg?.let { | ||||
|             val message = if (t != null) "$it: ${t.message}" else it | ||||
|             toPrintln(message, MingLiLoggerLevel.ERROR) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun isErrorEnabled(marker: Marker?): Boolean { | ||||
|         return true | ||||
|     } | ||||
| 
 | ||||
|     override fun error(marker: Marker?, msg: String?) { | ||||
|         error(msg) | ||||
|     } | ||||
| 
 | ||||
|     override fun error(marker: Marker?, format: String?, arg: Any?) { | ||||
|         error(format, arg) | ||||
|     } | ||||
| 
 | ||||
|     override fun error(marker: Marker?, format: String?, arg1: Any?, arg2: Any?) { | ||||
|         error(format, arg1, arg2) | ||||
|     } | ||||
| 
 | ||||
|     override fun error(marker: Marker?, format: String?, vararg arguments: Any?) { | ||||
|         error(format, *arguments) | ||||
|     } | ||||
| 
 | ||||
|     override fun error(marker: Marker?, msg: String?, t: Throwable?) { | ||||
|         error(msg, t) | ||||
|     } | ||||
| 
 | ||||
|     fun toPrintln(message: String, level: MingLiLoggerLevel) { | ||||
|         when (level) { | ||||
|             MingLiLoggerLevel.TRACE -> println("[TRACE] $message") | ||||
|             MingLiLoggerLevel.DEBUG -> println("[DEBUG] $message") | ||||
|             MingLiLoggerLevel.INFO -> println("[INFO] $message") | ||||
|             MingLiLoggerLevel.WARN -> println("[WARN] $message") | ||||
|             MingLiLoggerLevel.ERROR -> println("[ERROR] $message") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private fun format1(format: String, arg: Any?): String { | ||||
|         return if (format.contains("{}")) { | ||||
|             format.replaceFirst("{}", arg?.toString() ?: "null") | ||||
|         } else { | ||||
|             "$format $arg" | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private fun format2(format: String, arg1: Any?, arg2: Any?): String { | ||||
|         return format.replaceFirst("{}", arg1?.toString() ?: "null") | ||||
|             .replaceFirst("{}", arg2?.toString() ?: "null") | ||||
|     } | ||||
| 
 | ||||
|     private fun formatArray(format: String, arguments: Array<out Any?>): String { | ||||
|         var result = format | ||||
|         for (arg in arguments) { | ||||
|             result = result.replaceFirst("{}", arg?.toString() ?: "null") | ||||
|         } | ||||
|         return result | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| class MingLiLoggerFactory { | ||||
|     private var hasSLF4JImplementation: Boolean? = null | ||||
| 
 | ||||
|     // 线程安全的延迟初始化 | ||||
|     private fun checkSLF4JImplementation(): Boolean { | ||||
|         if (hasSLF4JImplementation == null) { | ||||
|             synchronized(this) { | ||||
|                 if (hasSLF4JImplementation == null) { | ||||
|                     hasSLF4JImplementation = try { | ||||
|                         // 更可靠的检测方式 | ||||
|                         ServiceLoader.load( | ||||
|                             Class.forName("org.slf4j.spi.SLF4JServiceProvider") | ||||
|                         ).iterator().hasNext() | ||||
|                     } catch (e: ClassNotFoundException) { | ||||
|                         false | ||||
|                     } catch (e: NoClassDefFoundError) { | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return hasSLF4JImplementation ?: false | ||||
|     } | ||||
| 
 | ||||
|     fun getLogger(name: String): Logger { | ||||
|         return if (checkSLF4JImplementation()) { | ||||
|             LoggerFactory.getLogger(name) | ||||
|         } else { | ||||
|             MingLiLogger() | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun getLogger(clazz: Class<*>): Logger { | ||||
|         return if (checkSLF4JImplementation()) { | ||||
|             LoggerFactory.getLogger(clazz) | ||||
|         } else { | ||||
|             MingLiLogger() | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| val mingLiLoggerFactory = MingLiLoggerFactory() | ||||
| 
 | ||||
| 
 | ||||
| @ -0,0 +1,102 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile DateTimeTypeHandler.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("DateTimeConvertor") | ||||
| 
 | ||||
| package com.mingliqiye.utils.mybatis.typehandler.datetime | ||||
| 
 | ||||
| import com.mingliqiye.utils.time.DateTime | ||||
| import org.apache.ibatis.type.BaseTypeHandler | ||||
| import org.apache.ibatis.type.JdbcType | ||||
| import org.apache.ibatis.type.MappedTypes | ||||
| import java.sql.CallableStatement | ||||
| import java.sql.PreparedStatement | ||||
| import java.sql.ResultSet | ||||
| import java.sql.SQLException | ||||
| import java.time.LocalDateTime | ||||
| 
 | ||||
| /** | ||||
|  * 将LocalDateTime对象转换为DateTime对象 | ||||
|  * | ||||
|  * @param localDateTime LocalDateTime对象 | ||||
|  * @return DateTime对象,如果localDateTime为null则返回null | ||||
|  */ | ||||
| fun toDateTime(localDateTime: LocalDateTime?): DateTime? { | ||||
|     return localDateTime?.let { DateTime.of(it) } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将DateTime对象转换为LocalDateTime对象 | ||||
|  * | ||||
|  * @param dateTime DateTime对象 | ||||
|  * @return LocalDateTime对象,如果dateTime为null则返回null | ||||
|  */ | ||||
| fun toLocalDateTime(dateTime: DateTime?): LocalDateTime? { | ||||
|     return dateTime?.toLocalDateTime() | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * DateTime类型处理器,用于在数据库和Java对象之间转换DateTime类型 | ||||
|  * @author MingLiQiye | ||||
|  * @date 2025/9/12 | ||||
|  */ | ||||
| @MappedTypes(DateTime::class) | ||||
| class DateTimeTypeHandler : BaseTypeHandler<DateTime>() { | ||||
| 
 | ||||
|     @Throws(SQLException::class) | ||||
|     override fun setNonNullParameter( | ||||
|         ps: PreparedStatement, | ||||
|         i: Int, | ||||
|         parameter: DateTime,  // 移除了 ?,因为这是 non-null 方法 | ||||
|         jdbcType: JdbcType | ||||
|     ) { | ||||
|         // 使用 setObject 允许传入 null,由数据库处理 | ||||
|         ps.setObject(i, toLocalDateTime(parameter)) | ||||
|     } | ||||
| 
 | ||||
|     @Throws(SQLException::class) | ||||
|     override fun getNullableResult(rs: ResultSet, columnName: String): DateTime? { | ||||
|         // 安全类型转换和空检查 | ||||
|         return when (val value = rs.getObject(columnName)) { | ||||
|             is LocalDateTime -> toDateTime(value) | ||||
|             null -> null | ||||
|             else -> throw SQLException("Expected LocalDateTime for column '$columnName', but got ${value.javaClass.name}") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Throws(SQLException::class) | ||||
|     override fun getNullableResult(rs: ResultSet, columnIndex: Int): DateTime? { | ||||
|         return when (val value = rs.getObject(columnIndex)) { | ||||
|             is LocalDateTime -> toDateTime(value) | ||||
|             null -> null | ||||
|             else -> throw SQLException("Expected LocalDateTime at column index $columnIndex, but got ${value.javaClass.name}") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Throws(SQLException::class) | ||||
|     override fun getNullableResult(cs: CallableStatement, columnIndex: Int): DateTime? { | ||||
|         return when (val value = cs.getObject(columnIndex)) { | ||||
|             is LocalDateTime -> toDateTime(value) | ||||
|             null -> null | ||||
|             else -> throw SQLException("Expected LocalDateTime at column index $columnIndex, but got ${value.javaClass.name}") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,279 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUIDTypeHandler.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("UUIDConvertor") | ||||
| 
 | ||||
| package com.mingliqiye.utils.mybatis.typehandler.uuid | ||||
| 
 | ||||
| import com.mingliqiye.utils.uuid.UUID | ||||
| import org.apache.ibatis.type.BaseTypeHandler | ||||
| import org.apache.ibatis.type.JdbcType | ||||
| import org.apache.ibatis.type.MappedJdbcTypes | ||||
| import org.apache.ibatis.type.MappedTypes | ||||
| import java.sql.CallableStatement | ||||
| import java.sql.PreparedStatement | ||||
| import java.sql.ResultSet | ||||
| import java.util.UUID as JUUID | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * 将字节数组转换为UUID对象 | ||||
|  * | ||||
|  * @param byteArray 字节数组 | ||||
|  * @return UUID对象,如果字节数组为null则返回null | ||||
|  */ | ||||
| fun byteArraytoUUID(byteArray: ByteArray?): UUID? { | ||||
|     return byteArray?.let { UUID.of(it) } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将UUID对象转换为字节数组 | ||||
|  * | ||||
|  * @param uuid UUID对象 | ||||
|  * @return 字节数组,如果UUID为null则返回null | ||||
|  */ | ||||
| fun uuidToByteArray(uuid: UUID?): ByteArray? { | ||||
|     return uuid?.toBytes() | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将字符串转换为UUID对象 | ||||
|  * | ||||
|  * @param str 字符串 | ||||
|  * @return UUID对象,如果字符串为null则返回null | ||||
|  */ | ||||
| fun stringToUUID(str: String?): UUID? { | ||||
|     return str?.let { UUID.of(it) } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将UUID对象转换为字符串 | ||||
|  * | ||||
|  * @param uuid UUID对象 | ||||
|  * @return 字符串,如果UUID为null则返回null | ||||
|  */ | ||||
| fun uuidToString(uuid: UUID?): String? { | ||||
|     return uuid?.getString() | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将java UUID转换为UUID对象 | ||||
|  * | ||||
|  * @param uuid JUUID java UUID | ||||
|  * @return UUID对象,如果字符串为null则返回null | ||||
|  */ | ||||
| fun juuidToUUID(uuid: JUUID?): UUID? { | ||||
|     return uuid?.let { UUID(it) } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将UUID对象转换为java UUID | ||||
|  * | ||||
|  * @param uuid UUID对象 | ||||
|  * @return java UUID,如果UUID为null则返回null | ||||
|  */ | ||||
| fun uuidToJuuid(uuid: UUID?): JUUID? { | ||||
|     return uuid?.getUuid() | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * JDBC类型转换 UUID BINARY | ||||
|  * @author MingLiQiye | ||||
|  * @date 2025/9/12 | ||||
|  */ | ||||
| @MappedTypes(UUID::class) | ||||
| @MappedJdbcTypes(JdbcType.BINARY) | ||||
| class UUIDBinaryTypeHandler : BaseTypeHandler<UUID>() { | ||||
|     /** | ||||
|      * 设置非空参数到PreparedStatement中 | ||||
|      * | ||||
|      * @param ps PreparedStatement对象 | ||||
|      * @param i 参数索引 | ||||
|      * @param parameter UUID参数值 | ||||
|      * @param jdbcType JDBC类型 | ||||
|      */ | ||||
|     override fun setNonNullParameter( | ||||
|         ps: PreparedStatement, i: Int, parameter: UUID, jdbcType: JdbcType | ||||
|     ) { | ||||
|         ps.setBytes(i, uuidToByteArray(parameter)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列名获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnName 列名 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         rs: ResultSet, columnName: String | ||||
|     ): UUID? { | ||||
|         return byteArraytoUUID(rs.getBytes(columnName)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult(rs: ResultSet, columnIndex: Int): UUID? { | ||||
|         return byteArraytoUUID(rs.getBytes(columnIndex)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从CallableStatement中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param cs CallableStatement对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         cs: CallableStatement, columnIndex: Int | ||||
|     ): UUID? { | ||||
|         return byteArraytoUUID(cs.getBytes(columnIndex)) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * JDBC类型转换 UUID String | ||||
|  * @author MingLiQiye | ||||
|  * @date 2025/9/12 | ||||
|  */ | ||||
| @MappedTypes(UUID::class) | ||||
| @MappedJdbcTypes(JdbcType.VARCHAR) | ||||
| class UUIDStringTypeHandler : BaseTypeHandler<UUID>() { | ||||
|     /** | ||||
|      * 设置非空参数到PreparedStatement中 | ||||
|      * | ||||
|      * @param ps PreparedStatement对象 | ||||
|      * @param i 参数索引 | ||||
|      * @param parameter UUID参数值 | ||||
|      * @param jdbcType JDBC类型 | ||||
|      */ | ||||
|     override fun setNonNullParameter( | ||||
|         ps: PreparedStatement, i: Int, parameter: UUID, jdbcType: JdbcType | ||||
|     ) { | ||||
|         ps.setString(i, uuidToString(parameter)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列名获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnName 列名 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         rs: ResultSet, columnName: String | ||||
|     ): UUID? { | ||||
|         return stringToUUID(rs.getString(columnName)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult(rs: ResultSet, columnIndex: Int): UUID? { | ||||
|         return stringToUUID(rs.getString(columnIndex)) | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从CallableStatement中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param cs CallableStatement对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         cs: CallableStatement, columnIndex: Int | ||||
|     ): UUID? { | ||||
|         return stringToUUID(cs.getString(columnIndex)) | ||||
|     } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * JDBC类型转换 UUID java UUID | ||||
|  * @author MingLiQiye | ||||
|  * @date 2025/9/12 | ||||
|  */ | ||||
| @MappedTypes(UUID::class) | ||||
| @MappedJdbcTypes(JdbcType.OTHER) | ||||
| class UUIDTypeHandler : BaseTypeHandler<UUID>() { | ||||
|     /** | ||||
|      * 设置非空参数到PreparedStatement中 | ||||
|      * | ||||
|      * @param ps PreparedStatement对象 | ||||
|      * @param i 参数索引 | ||||
|      * @param parameter UUID参数值 | ||||
|      * @param jdbcType JDBC类型 | ||||
|      */ | ||||
|     override fun setNonNullParameter( | ||||
|         ps: PreparedStatement, i: Int, parameter: UUID, jdbcType: JdbcType | ||||
|     ) { | ||||
|         ps.setObject(i, parameter.getUuid()) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列名获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnName 列名 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         rs: ResultSet, | ||||
|         columnName: String | ||||
|     ): UUID? { | ||||
|         return juuidToUUID(rs.getObject(columnName, JUUID::class.java) as JUUID) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult(rs: ResultSet, columnIndex: Int): UUID? { | ||||
|         return juuidToUUID(rs.getObject(columnIndex, JUUID::class.java) as JUUID) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从CallableStatement中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param cs CallableStatement对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         cs: CallableStatement, columnIndex: Int | ||||
|     ): UUID? { | ||||
|         return juuidToUUID(cs.getObject(columnIndex, JUUID::class.java) as JUUID) | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,119 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile MysqlUUIDBinaryTypeHandler.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.mybatis.typehandler.uuid.mysql | ||||
| 
 | ||||
| import com.mingliqiye.utils.uuid.UUID | ||||
| import com.mingliqiye.utils.uuid.mysqlToUuid | ||||
| import com.mingliqiye.utils.uuid.uuidToMysql | ||||
| import org.apache.ibatis.type.BaseTypeHandler | ||||
| import org.apache.ibatis.type.JdbcType | ||||
| import org.apache.ibatis.type.MappedTypes | ||||
| import java.sql.CallableStatement | ||||
| import java.sql.PreparedStatement | ||||
| import java.sql.ResultSet | ||||
| 
 | ||||
| /** | ||||
|  * JDBC类型转换 UUID | ||||
|  * @author MingLiQiye | ||||
|  * @date 2025/9/12 | ||||
|  */ | ||||
| @MappedTypes(UUID::class) | ||||
| class MysqlUUIDBinaryTypeHandler : BaseTypeHandler<UUID>() { | ||||
| 
 | ||||
|     /** | ||||
|      * 将字节数组转换为UUID对象 | ||||
|      * | ||||
|      * @param byteArray 字节数组 | ||||
|      * @return UUID对象,如果字节数组为null则返回null | ||||
|      */ | ||||
|     private fun toUUID(byteArray: ByteArray?): UUID? { | ||||
|         return byteArray?.let { UUID.of(mysqlToUuid(it)) } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 将UUID对象转换为字节数组 | ||||
|      * | ||||
|      * @param uuid UUID对象 | ||||
|      * @return 字节数组,如果UUID为null则返回null | ||||
|      */ | ||||
|     fun toByteArray(uuid: UUID?): ByteArray? { | ||||
|         return uuid?.let { uuidToMysql(it.toBytes()) } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 设置非空参数到PreparedStatement中 | ||||
|      * | ||||
|      * @param ps PreparedStatement对象 | ||||
|      * @param i 参数索引 | ||||
|      * @param parameter UUID参数值 | ||||
|      * @param jdbcType JDBC类型 | ||||
|      */ | ||||
|     override fun setNonNullParameter( | ||||
|         ps: PreparedStatement, | ||||
|         i: Int, | ||||
|         parameter: UUID, | ||||
|         jdbcType: JdbcType | ||||
|     ) { | ||||
|         ps.setBytes(i, toByteArray(parameter)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列名获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnName 列名 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         rs: ResultSet, | ||||
|         columnName: String | ||||
|     ): UUID? { | ||||
|         return toUUID(rs.getBytes(columnName)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从ResultSet中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param rs ResultSet对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult(rs: ResultSet, columnIndex: Int): UUID? { | ||||
|         return toUUID(rs.getBytes(columnIndex)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 从CallableStatement中根据列索引获取可空的UUID结果 | ||||
|      * | ||||
|      * @param cs CallableStatement对象 | ||||
|      * @param columnIndex 列索引 | ||||
|      * @return UUID对象或null | ||||
|      */ | ||||
|     override fun getNullableResult( | ||||
|         cs: CallableStatement, | ||||
|         columnIndex: Int | ||||
|     ): UUID? { | ||||
|         return toUUID(cs.getBytes(columnIndex)) | ||||
|     } | ||||
| 
 | ||||
| } | ||||
							
								
								
									
										174
									
								
								src/main/kotlin/com/mingliqiye/utils/string/StringUtils.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								src/main/kotlin/com/mingliqiye/utils/string/StringUtils.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,174 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile StringUtils.kt | ||||
|  * LastUpdate 2025-09-14 21:46:14 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("StringUtils") | ||||
| 
 | ||||
| package com.mingliqiye.utils.string | ||||
| 
 | ||||
| import com.mingliqiye.utils.logger.mingLiLoggerFactory | ||||
| 
 | ||||
| 
 | ||||
| val log = mingLiLoggerFactory.getLogger("StringUtils") | ||||
| 
 | ||||
| /** | ||||
|  * 判断`字符串`是否为空 | ||||
|  * | ||||
|  * @param str 待判断的字符串 | ||||
|  * @return `true`: 空 `false`: 非空 | ||||
|  */ | ||||
| fun isEmpty(str: String?): Boolean { | ||||
|     return str?.isEmpty() != null | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 格式化字符串,将字符串中的占位符{}替换为对应的参数值 | ||||
|  * | ||||
|  * `Kotlin`语言给我老老实实用`$`啊 | ||||
|  * | ||||
|  * @param str 需要格式化的字符串,包含{}占位符 \\{} 代表一个{} | ||||
|  * @param args 要替换占位符的参数列表 | ||||
|  * @return 格式化后的字符串 | ||||
|  */ | ||||
| fun format(str: String, vararg args: Any?): String { | ||||
|     var argIndex = 0 | ||||
|     val result = StringBuilder() | ||||
|     var lastIndex = 0 | ||||
| 
 | ||||
|     // 匹配所有非转义的 {} | ||||
|     val pattern = Regex("(?<!\\\\)\\{}") | ||||
|     val matches = pattern.findAll(str) | ||||
| 
 | ||||
|     for (match in matches) { | ||||
|         // 添加匹配前的文本 | ||||
|         result.append(str, lastIndex, match.range.first) | ||||
| 
 | ||||
|         // 替换占位符 | ||||
|         if (argIndex < args.size) { | ||||
|             result.append(args[argIndex].toString()) | ||||
|             argIndex++ | ||||
|         } | ||||
| 
 | ||||
|         lastIndex = match.range.last + 1 | ||||
|     } | ||||
| 
 | ||||
|     // 添加剩余文本 | ||||
|     if (lastIndex < str.length) { | ||||
|         result.append(str, lastIndex, str.length) | ||||
|     } | ||||
| 
 | ||||
|     // 处理转义的 \\{}(替换为 {}) | ||||
|     val finalResult = result.toString().replace("\\{}", "{}") | ||||
| 
 | ||||
|     // 检查参数数量 | ||||
|     val placeholderCount = matches.count() | ||||
|     if (argIndex != args.size) { | ||||
|         log.warn("Placeholder count: $placeholderCount, Argument count: ${args.size}") | ||||
|     } | ||||
| 
 | ||||
|     return finalResult | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * 将字符串转换为Unicode编码格式 | ||||
|  * | ||||
|  * @param str 待转换的字符串 | ||||
|  * @return 转换后的Unicode编码字符串,每个字符都以\\u开头的十六进制形式表示 | ||||
|  */ | ||||
| fun stringToUnicode(str: String): String { | ||||
|     val sb = java.lang.StringBuilder() | ||||
|     val c = str.toCharArray() | ||||
|     for (value in c) { | ||||
|         sb.append(stringToUnicode(value)) | ||||
|     } | ||||
|     return sb.toString() | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将字符转换为Unicode转义字符串 | ||||
|  * | ||||
|  * @param c 需要转换的字符 | ||||
|  * @return 返回格式为"\\uXXXX"的Unicode转义字符串,其中XXXX为字符的十六进制Unicode码点 | ||||
|  */ | ||||
| fun stringToUnicode(c: Char): String { | ||||
|     return "\\u" + String.format("%04x", c.code) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将整数转换为Unicode字符串表示形式 | ||||
|  * | ||||
|  * @param c 要转换的整数,表示Unicode码点 | ||||
|  * @return 返回格式为"\\uXXXX"的Unicode字符串,其中XXXX是参数c的十六进制表示 | ||||
|  */ | ||||
| fun stringToUnicode(c: Int): String { | ||||
|     return "\\u" + Integer.toHexString(c) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 将Unicode编码字符串转换为普通字符串 | ||||
|  * | ||||
|  * 该函数接收一个包含Unicode转义序列的字符串,将其解析并转换为对应的字符序列 | ||||
|  * | ||||
|  * @param unicode 包含Unicode转义序列的字符串,格式如"\\uXXXX",其中XXXX为十六进制数 | ||||
|  * @return 转换后的普通字符串,包含对应的Unicode字符 | ||||
|  */ | ||||
| fun unicodeToString(unicode: String): String { | ||||
|     val sb = java.lang.StringBuilder() | ||||
|     // 按照Unicode转义符分割字符串,得到包含十六进制编码的数组 | ||||
|     val hex: Array<String?> = unicode.split("\\\\u".toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray() | ||||
|     // 从索引1开始遍历,因为分割后的第一个元素是转义符前面的内容(可能为空) | ||||
|     for (i in 1..<hex.size) { | ||||
|         // 将十六进制字符串转换为整数,作为字符的Unicode码点 | ||||
|         val index = hex[i]!!.toInt(16) | ||||
|         // 将Unicode码点转换为对应字符并添加到结果中 | ||||
|         sb.append(index.toChar()) | ||||
|     } | ||||
|     return sb.toString() | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 创建一个指定初始容量的StringBuilder实例 | ||||
|  * | ||||
|  * @param i StringBuilder的初始容量 | ||||
|  * @return 指定初始容量的StringBuilder实例 | ||||
|  */ | ||||
| fun stringBuilder(i: Int): java.lang.StringBuilder { | ||||
|     return StringBuilder(i) | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 根据当前字符串创建一个StringBuilder实例 | ||||
|  * | ||||
|  * @return 包含当前字符串内容的StringBuilder实例 | ||||
|  */ | ||||
| fun String.stringBuilder(): java.lang.StringBuilder { | ||||
|     return StringBuilder(this) | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| fun split(str: String, separator: String): List<String> { | ||||
|     return str.split(separator) | ||||
| } | ||||
| 
 | ||||
| fun List<String>.join(separator: String): String { | ||||
|     return this.joinToString(separator) | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										53
									
								
								src/main/kotlin/com/mingliqiye/utils/uuid/MysqlUUIDv1.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/main/kotlin/com/mingliqiye/utils/uuid/MysqlUUIDv1.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | ||||
| /* | ||||
|  * Copyright 2025 mingliqiye | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile MysqlUUIDv1.kt | ||||
|  * LastUpdate 2025-09-14 18:19:29 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| @file:JvmName("MysqlUUIDv1") | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid | ||||
| 
 | ||||
| 
 | ||||
| fun uuidToMysql(uuid: ByteArray): ByteArray { | ||||
|     val reuuid = ByteArray(16) | ||||
|     reuuid[4] = uuid[0] | ||||
|     reuuid[5] = uuid[1] | ||||
|     reuuid[6] = uuid[2] | ||||
|     reuuid[7] = uuid[3] | ||||
|     reuuid[2] = uuid[4] | ||||
|     reuuid[3] = uuid[5] | ||||
|     reuuid[0] = uuid[6] | ||||
|     reuuid[1] = uuid[7] | ||||
|     System.arraycopy(uuid, 8, reuuid, 8, 8) | ||||
|     return reuuid | ||||
| } | ||||
| 
 | ||||
| fun mysqlToUuid(uuid: ByteArray): ByteArray { | ||||
|     val reuuid = ByteArray(16) | ||||
|     reuuid[6] = uuid[0] | ||||
|     reuuid[7] = uuid[1] | ||||
|     reuuid[4] = uuid[2] | ||||
|     reuuid[5] = uuid[3] | ||||
|     reuuid[0] = uuid[4] | ||||
|     reuuid[1] = uuid[5] | ||||
|     reuuid[2] = uuid[6] | ||||
|     reuuid[3] = uuid[7] | ||||
|     System.arraycopy(uuid, 8, reuuid, 8, 8) | ||||
|     return reuuid | ||||
| } | ||||
| @ -16,10 +16,9 @@ | ||||
|  * ProjectName mingli-utils | ||||
|  * ModuleName mingli-utils.main | ||||
|  * CurrentFile UUID.kt | ||||
|  * LastUpdate 2025-09-12 16:57:52 | ||||
|  * LastUpdate 2025-09-14 19:55:47 | ||||
|  * UpdateUser MingLiPro | ||||
|  */ | ||||
| 
 | ||||
| package com.mingliqiye.utils.uuid | ||||
| 
 | ||||
| import com.github.f4b6a3.uuid.UuidCreator | ||||
| @ -84,6 +83,16 @@ class UUID : Serializable { | ||||
|             return UUID(uuid) | ||||
|         } | ||||
| 
 | ||||
|         /** | ||||
|          * 从Java的UUID | ||||
|          * @param uuid 字符串 | ||||
|          * @return UUID | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         fun of(uuid: JUUID): UUID { | ||||
|             return UUID(uuid) | ||||
|         } | ||||
| 
 | ||||
|         /** | ||||
|          * 从字节码转换到UUID | ||||
|          * @param array 16字节 | ||||
| @ -93,6 +102,10 @@ class UUID : Serializable { | ||||
|         fun of(array: ByteArray): UUID { | ||||
|             return UUID(array) | ||||
|         } | ||||
| 
 | ||||
|         fun JUUID.toMlUUID(): UUID { | ||||
|             return of(this) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     internal constructor(msb: Long, lsb: Long) { | ||||
| @ -108,10 +121,11 @@ class UUID : Serializable { | ||||
|         this.uuid = JUUID(bb.getLong(), bb.getLong()) | ||||
|     } | ||||
| 
 | ||||
|     internal constructor(uuid: String) { | ||||
|     constructor(uuid: String) { | ||||
|         this.uuid = JUUID.fromString(uuid) | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     /** | ||||
|      * 获取对应的字节码 | ||||
|      * @return 字节码 | ||||
| @ -242,3 +256,5 @@ class UUID : Serializable { | ||||
|         return uuid.hashCode() | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -4,4 +4,4 @@ artifactId=$ARTIFACTID | ||||
| version=$VERSIONS | ||||
| buildJdkVersion=$JDKVERSIONS | ||||
| author=MingLiPro | ||||
| website=mingliqiye.com | ||||
| website=https://mingli-utils.mingliqiye.com | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user