用Java手写jvm之实现查找class

写在前面

完成类加载器加载class的三阶段,加载,解析,初始化中的加载😀😀😀

源码 。
jvm想要运行class,是根据类全限定名称来从特定的位置基于类加载器来查找的,分别如下:

启动类加载器负责加载jre/lib目录
扩展类加载器负责加载jre/ext/lib目录
应用加载器负责加载应用程序classpath

不管是哪种类加载器,都要完成找到class查找这个工作。

具体类加载器的工作流程可以分为如下3个步骤:

1:加载根据类的全限定名称(包+类名称),定位并且读取字节码
2:链接(验证,准备,解析)验证字节码是否合法,给静态变量申请内存,符号引用(一个串)转直接引用(一个内存地址)
3:初始化执行初始化代码,如{}代码块,以及给静态变量赋值

本部分要来模拟的就是这里的1:加载

1:程序

因为分别要从如下三个目录来查找class:

1:jre/lib
2:jre/ext/lib
3:应用classpath

不管是那个目录,用户在指定class所在位置时都可能使用不同的形式,可能如下:

1:一个目录,如d:\test\
2:一个jar,zip等,如d:\test\aa.jar,d:\test\aa.zip
3:通配符形式,如d:\test\*
4:也可能是以上三种的组合形式

为了抽象不同的文件形式,我们先来定义如下文件入口形式的接口:

public interface Entry {byte[] readClass(String className) throws IOException;static Entry create(String path) {//File.pathSeparator;路径分隔符(win\linux)if (path.contains(File.pathSeparator)) {return new CompositeEntry(path);}if (path.endsWith("*")) {return new WildcardEntry(path);}if (path.endsWith(".jar") || path.endsWith(".JAR") ||path.endsWith(".zip") || path.endsWith(".ZIP")) {return new ZipEntry(path);}return new DirEntry(path);}}

其中抽象方法readClass用来完成读取字节码的工作,也就是本文要做的内容,静态方法create用来根据给定的路径生成对应的Entry对象,这里根据路径形式的不同我们需要定义Entry不同的子类,分别是目录形式的子类DirectoryEntry:

package com.dahuyou.find.clazz.entry.impl;import com.dahuyou.find.clazz.entry.Entry;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;/*** 目录形式的entry 如 d:\test*/
public class DirectoryEntry implements Entry {private Path absolutePath;public DirectoryEntry(String path) {//获取绝对路径this.absolutePath = Paths.get(path).toAbsolutePath();}@Overridepublic byte[] readClass(String className) throws IOException {return Files.readAllBytes(absolutePath.resolve(className));}@Overridepublic String toString() {return this.absolutePath.toString();}
}

jar和zip形式的entry:

package com.dahuyou.find.clazz.entry.impl;import com.dahuyou.find.clazz.entry.Entry;import java.io.IOException;
import java.nio.file.*;/*** jar,zip形式的entry*/
public class JarOrZipEntry implements Entry {private Path absolutePath;public JarOrZipEntry(String path) {//获取绝对路径this.absolutePath = Paths.get(path).toAbsolutePath();}@Overridepublic byte[] readClass(String className) throws IOException {try (FileSystem zipFs = FileSystems.newFileSystem(absolutePath, null)) {return Files.readAllBytes(zipFs.getPath(className));}}@Overridepublic String toString() {return this.absolutePath.toString();}}

这里因为通配符形式的路径其实是组合形式的一种特殊写法,如d:\test\目录下有a.jar,b.jar,则组合形式就是d:\test\a.jar,d:\test\b.jar,而通配符形式就是d:\test\*,通配符形式是可以转换为组合形式的,只不过多了一个转换的工作而已,所以这里,我们单独定义组合形式子类,然后再继承组合形式定义通配符形式。

则,组合形式子类如下:

package com.dahuyou.find.clazz.entry.impl;import com.dahuyou.find.clazz.entry.Entry;import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;/*** 一组路径的组合形式entry*/
public class CompositeEntry implements Entry {private final List<Entry> entryList = new ArrayList<>();public CompositeEntry(String pathList) {String[] paths = pathList.split(File.pathSeparator);for (String path : paths) {entryList.add(Entry.create(path));}}@Overridepublic byte[] readClass(String className) throws IOException {for (Entry entry : entryList) {try {return entry.readClass(className);} catch (Exception ignored) {//ignored}}throw new IOException("class not found " + className);}@Overridepublic String toString() {String[] strs = new String[entryList.size()];for (int i = 0; i < entryList.size(); i++) {strs[i] = entryList.get(i).toString();}return String.join(File.pathSeparator, strs);}}

通配符entry如下:

package com.dahuyou.find.clazz.entry.impl;import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.stream.Collectors;/*** 通配符entry,继承CompositeEntry,只需要将通配符转换为一组路径,就转换为组合entry了,* 所以这里直接继承组合entry,妙啊!!!*/
public class WildcardEntry extends CompositeEntry {public WildcardEntry(String path) {super(toPathList(path));}private static String toPathList(String wildcardPath) {String baseDir = wildcardPath.replace("*", ""); // remove *try {return Files.walk(Paths.get(baseDir)).filter(Files::isRegularFile).map(Path::toString).filter(p -> p.endsWith(".jar") || p.endsWith(".JAR")).collect(Collectors.joining(File.pathSeparator));} catch (IOException e) {return "";}}}

UML图:
在这里插入图片描述

又因为类最终其实是从classpath下查找的,所以我们还需要定义一个类来代表classpath,使用各种entry来完成具体查找class的工作:

package com.dahuyou.find.clazz.classpath;import com.dahuyou.find.clazz.classpath.entry.Entry;
import com.dahuyou.find.clazz.classpath.entry.impl.WildcardEntry;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Paths;/*** classpath类(jre entry+jre ext entry+应用entry)*/
public class Classpath {// jre路径的entry(jvm中启动类加载器使用)private Entry bootstrapClasspath;// jre扩展路径entry(jvm中扩展类加载器使用)private Entry extensionClasspath;// 应用class路径entry(jvm中app类加载器使用)private Entry userClasspath;public Classpath(String jreOption, String cpOption) {//启动类(D:\programs\javas\java1.8\jre\lib)&扩展类(D:\programs\javas\java1.8\jre\lib\ext)bootstrapAndExtensionClasspath(jreOption);//用户类 d:\test\target\classesparseUserClasspath(cpOption);}private void bootstrapAndExtensionClasspath(String jreOption) {String jreDir = getJreDir(jreOption);//..jre/lib/* jre核心类库初始化String jreLibPath = Paths.get(jreDir, "lib") + File.separator + "*";bootstrapClasspath = new WildcardEntry(jreLibPath);//..jre/lib/ext/* jre ext初始化String jreExtPath = Paths.get(jreDir, "lib", "ext") + File.separator + "*";extensionClasspath = new WildcardEntry(jreExtPath);}private static String getJreDir(String jreOption) {// 用户直接设置了jre目录 D:\programs\javas\java1.8\jreif (jreOption != null && Files.exists(Paths.get(jreOption))) {return jreOption;}if (Files.exists(Paths.get("./jre"))) {return "./jre";}String jh = System.getenv("JAVA_HOME");if (jh != null) {return Paths.get(jh, "jre").toString();}throw new RuntimeException("Can not find JRE folder!");}private void parseUserClasspath(String cpOption) {if (cpOption == null) {cpOption = ".";}userClasspath = Entry.create(cpOption);}public byte[] readClass(String className) throws Exception {className = className + ".class";// 先使用jre 的entry加载(对应jvm先通过启动类加载器加载)try {return bootstrapClasspath.readClass(className);} catch (Exception ignored) {//ignored}// jre的entry加载不到,使用ext对应的entry加载(对应jvm 启动类加载器加载不到,使用ext类加载器加载)try {return extensionClasspath.readClass(className);} catch (Exception ignored) {//ignored}// 最后通过用户类 entry加载(对应jvm最后猜使用app应用类加载器)return userClasspath.readClass(className);}}

总结以上代码就是,为了对应不同形式的class,即不同入口形式的class,定义了entry接口,并根据不同的入口形式定义了对应的entry实现,然后定义了classpath类,该类分别针对jre,jre ext,用户类,定义了对应的entry,这样的当需要查找某个class时,就可以按照顺序分别从jre entry->jre ext entry->用户entry来查找了,可参考如下图:
在这里插入图片描述

接着我们还需要定义接收命令行参数的类,这个如果你看过前面的文章就比较熟悉了,这里贴下源码:

package com.dahuyou.find.clazz.cmd;import com.beust.jcommander.JCommander;
import com.beust.jcommander.Parameter;import java.util.List;public class Cmd {@Parameter(names = {"-?", "-help"}, description = "print help message", order = 3, help = true)boolean helpFlag = false;@Parameter(names = "-version", description = "print version and exit", order = 2)boolean versionFlag = false;@Parameter(names = {"-cp", "-classpath"}, description = "classpath", order = 1)String classpath;@Parameter(names = "-Xthejrepath", description = "path to jre", order = 4)String thejrepath;// 要查找的目标类@Parameter(names = "-Xthetargetclazz", description = "path to jre", order = 4)public String thetargetclazz;@Parameter(description = "main class and args")List<String> mainClassAndArgs;boolean ok;String getMainClass() {return mainClassAndArgs != null && !mainClassAndArgs.isEmpty()? mainClassAndArgs.get(0): null;}List<String> getAppArgs() {return mainClassAndArgs != null && mainClassAndArgs.size() > 1? mainClassAndArgs.subList(1, mainClassAndArgs.size()): null;}static Cmd parse(String[] argv) {Cmd args = new Cmd();JCommander cmd = JCommander.newBuilder().addObject(args).build();cmd.parse(argv);args.ok = true;return args;}}

通过-Xthejrepath来指定jre路径,当然也可以不指定,程序也会读取JAVA_HOME环境变量配置。-Xthetargetclazz指定要查找的目标类。

定义主函数类:

package com.dahuyou.find.clazz;import com.dahuyou.find.clazz.classpath.Classpath;
import com.dahuyou.find.clazz.cmd.Cmd;
import java.util.Arrays;/*** program arguments:-Xthejrepath D:\programs\javas\java1.8/jre -Xthetargetclazz D:\test\itstack-demo-jvm-master\find-class-from-classpath\target\test-classes\com\dahuyou\find\clazz\test\ShowMeByteCode*/
public class Main {public static void main(String[] args) {Cmd cmd = Cmd.parse(args);if (!cmd.ok || cmd.helpFlag) {System.out.println("Usage: <main class> [-options] class [args...]");return;}if (cmd.versionFlag) {//注意案例测试都是基于1.8,另外jdk1.9以后使用模块化没有rt.jarSystem.out.println("java version \"1.8.0\"");return;}startJVM(cmd);}private static void startJVM(Cmd cmd) {// 创建classpathClasspath cp = new Classpath(cmd.thejrepath, cmd.classpath);System.out.printf("classpath:%s class:%s args:%s\n", cp, cmd.getMainClass(), cmd.getAppArgs());//获取className
//        String className = cmd.getMainClass().replace(".", "/");try {
//            byte[] classData = cp.readClass(className);byte[] classData = cp.readClass(cmd.thetargetclazz.replace(".", "/"));System.out.println(Arrays.toString(classData));System.out.println("classData:");for (byte b : classData) {//16进制输出System.out.print(String.format("%02x", b & 0xff) + " ");}} catch (Exception e) {System.out.println("Could not find or load main class " + cmd.getMainClass());e.printStackTrace();}}}

为了测试,我们来定义一个要查找的类:

package com.dahuyou.find.clazz.test;public class ShowMeByteCode {public static void main(String[] args) {}
}

运行,首先配置参数:
在这里插入图片描述

-Xthejrepath
D:\programs\javas\java1.8/jre
-Xthetargetclazz
D:\test\itstack-demo-jvm-master\find-class-from-classpath\target\test-classes\com\dahuyou\find\clazz\test\ShowMeByteCode

-Xthejrepath设置jre的路径,-Xthetargetclazz设置要查找的目标class,运行如下:

classpath:com.dahuyou.find.clazz.classpath.Classpath@bebdb06 class:null args:null
[-54, -2, -70, -66, 0, 0, 0, 52, 0, 20, 10, 0, 3, 0, 17, 7, 0, 18, 7, 0, 19, 1, 0, 6, 60, 105, 110, 105, 116, 62, 1, 0, 3, 40, 41, 86, 1, 0, 4, 67, 111, 100, 101, 1, 0, 15, 76, 105, 110, 101, 78, 117, 109, 98, 101, 114, 84, 97, 98, 108, 101, 1, 0, 18, 76, 111, 99, 97, 108, 86, 97, 114, 105, 97, 98, 108, 101, 84, 97, 98, 108, 101, 1, 0, 4, 116, 104, 105, 115, 1, 0, 44, 76, 99, 111, 109, 47, 100, 97, 104, 117, 121, 111, 117, 47, 102, 105, 110, 100, 47, 99, 108, 97, 122, 122, 47, 116, 101, 115, 116, 47, 83, 104, 111, 119, 77, 101, 66, 121, 116, 101, 67, 111, 100, 101, 59, 1, 0, 4, 109, 97, 105, 110, 1, 0, 22, 40, 91, 76, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 83, 116, 114, 105, 110, 103, 59, 41, 86, 1, 0, 4, 97, 114, 103, 115, 1, 0, 19, 91, 76, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 83, 116, 114, 105, 110, 103, 59, 1, 0, 10, 83, 111, 117, 114, 99, 101, 70, 105, 108, 101, 1, 0, 19, 83, 104, 111, 119, 77, 101, 66, 121, 116, 101, 67, 111, 100, 101, 46, 106, 97, 118, 97, 12, 0, 4, 0, 5, 1, 0, 42, 99, 111, 109, 47, 100, 97, 104, 117, 121, 111, 117, 47, 102, 105, 110, 100, 47, 99, 108, 97, 122, 122, 47, 116, 101, 115, 116, 47, 83, 104, 111, 119, 77, 101, 66, 121, 116, 101, 67, 111, 100, 101, 1, 0, 16, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 79, 98, 106, 101, 99, 116, 0, 33, 0, 2, 0, 3, 0, 0, 0, 0, 0, 2, 0, 1, 0, 4, 0, 5, 0, 1, 0, 6, 0, 0, 0, 47, 0, 1, 0, 1, 0, 0, 0, 5, 42, -73, 0, 1, -79, 0, 0, 0, 2, 0, 7, 0, 0, 0, 6, 0, 1, 0, 0, 0, 3, 0, 8, 0, 0, 0, 12, 0, 1, 0, 0, 0, 5, 0, 9, 0, 10, 0, 0, 0, 9, 0, 11, 0, 12, 0, 1, 0, 6, 0, 0, 0, 43, 0, 0, 0, 1, 0, 0, 0, 1, -79, 0, 0, 0, 2, 0, 7, 0, 0, 0, 6, 0, 1, 0, 0, 0, 6, 0, 8, 0, 0, 0, 12, 0, 1, 0, 0, 0, 1, 0, 13, 0, 14, 0, 0, 0, 1, 0, 15, 0, 0, 0, 2, 0, 16]
classData:
ca fe ba be 00 00 00 34 00 14 0a 00 03 00 11 07 00 12 07 00 13 01 00 06 3c 69 6e 69 74 3e 01 00 03 28 29 56 01 00 04 43 6f 64 65 01 00 0f 4c 69 6e 65 4e 75 6d 62 65 72 54 61 62 6c 65 01 00 12 4c 6f 63 61 6c 56 61 72 69 61 62 6c 65 54 61 62 6c 65 01 00 04 74 68 69 73 01 00 2c 4c 63 6f 6d 2f 64 61 68 75 79 6f 75 2f 66 69 6e 64 2f 63 6c 61 7a 7a 2f 74 65 73 74 2f 53 68 6f 77 4d 65 42 79 74 65 43 6f 64 65 3b 01 00 04 6d 61 69 6e 01 00 16 28 5b 4c 6a 61 76 61 2f 6c 61 6e 67 2f 53 74 72 69 6e 67 3b 29 56 01 00 04 61 72 67 73 01 00 13 5b 4c 6a 61 76 61 2f 6c 61 6e 67 2f 53 74 72 69 6e 67 3b 01 00 0a 53 6f 75 72 63 65 46 69 6c 65 01 00 13 53 68 6f 77 4d 65 42 79 74 65 43 6f 64 65 2e 6a 61 76 61 0c 00 04 00 05 01 00 2a 63 6f 6d 2f 64 61 68 75 79 6f 75 2f 66 69 6e 64 2f 63 6c 61 7a 7a 2f 74 65 73 74 2f 53 68 6f 77 4d 65 42 79 74 65 43 6f 64 65 01 00 10 6a 61 76 61 2f 6c 61 6e 67 2f 4f 62 6a 65 63 74 00 21 00 02 00 03 00 00 00 00 00 02 00 01 00 04 00 05 00 01 00 06 00 00 00 2f 00 01 00 01 00 00 00 05 2a b7 00 01 b1 00 00 00 02 00 07 00 00 00 06 00 01 00 00 00 03 00 08 00 00 00 0c 00 01 00 00 00 05 00 09 00 0a 00 00 00 09 00 0b 00 0c 00 01 00 06 00 00 00 2b 00 00 00 01 00 00 00 01 b1 00 00 00 02 00 07 00 00 00 06 00 01 00 00 00 06 00 08 00 00 00 0c 00 01 00 00 00 01 00 0d 00 0e 00 00 00 01 00 0f 00 00 00 02 00 10 
Process finished with exit code 0

可以使用sublime等工具打开class,查看十六进制字节码对比程序输出结果:
在这里插入图片描述

写在后面

参考文章列表

【Java 基础篇】Java类加载器详解 。

jvm之启动参数 。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/49743.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【SQL 新手教程 2/20】关系模型 -- 主键

&#x1f497; 关系数据库建立在关系模型上⭐ 关系模型本质上就是若干个存储数据的二维表 记录 (Record)&#xff1a; 表的每一行称为记录&#xff08;Record&#xff09;&#xff0c;记录是一个逻辑意义上的数据 字段 (Column)&#xff1a;表的每一列称为字段&#xff08;Colu…

吴恩达的TranslationAgent学习

TranslationAgent构成 整个[TranslationAgent (github.com)]在流程上分为短文本的一次性翻译和长文本的分chunk翻译&#xff08;按照Token进行划分&#xff09;。 但是不论长文本翻译还是短文本翻译&#xff0c;总体流程遵循执行、纠正再执行的逻辑循环实现。 这种按照自省思路…

【数字IC/FPGA】使用Verdi对比两个波形

步骤一&#xff1a; 使用verdi打开第一个波形 bsub verdi -ssf 1.fsdb添加需要观察的信号&#xff0c;如下图所示&#xff1a; 步骤二&#xff1a; 新建容器&#xff0c;依次点击Window --> Dock to --> New Container Window。 然后输入容器的名字&#xff0c;如下图所…

SQL数据库:通过在视频监控平台服务器上直接使用SQL存储过程,在海量记录中查询特定时间段内-某个摄像头的所有视频片段

目录 一、背景 1、存储过程 2、视频监控系统 二、需求和数据表 1、具体要求 2、数据表 3、部分数据 三、实现 1、目标 2、创建存储过程 &#xff08;1&#xff09;存储过程代码 &#xff08;2&#xff09;创建成功 3、存储过程的解释 4、SQL命令调用方式 5、调用…

Java----队列(Queue)

目录 1.队列&#xff08;Queue&#xff09; 1.1概念 1.2队列的使用 1.3队列的模拟实现 1.4循环队列 1.4.1循环队列下标偏移 1.4.2如何区分队列是空还是满 1.5双端队列 (Deque) 1.队列&#xff08;Queue&#xff09; 1.1概念 队列&#xff1a;只允许在一端进行插入数据…

Linux Redhat ens33不显示IP问题

优质博文&#xff1a;IT-BLOG-CN 【第一步】&#xff1a;查看系统网卡设备 : ip addr show 【第二步】&#xff1a;修改网卡配置参数 cd /etc/sysconfig/network-scripts/ vi ifcfg-ens33 修改ONBOOT参数为yes 【第三步】&#xff1a;重启网卡&#xff0c;然后ping检测…

奇怪的Excel单元格字体颜色格式

使用VBA代码修改单元格全部字符字体颜色是个很简单的任务&#xff0c;例如设置A1单元格字体颜色为红色。 Range("A1").Font.Color RGB(255, 0, 0)有时需要修改部分字符的颜色&#xff0c;如下图所示&#xff0c;将红色字符字体颜色修改为蓝色。代码将会稍许复杂&am…

Linux:Linux进程控制

目录 1. 进程概念 1.1 并行和并发 2. 进程创建 2.1 fork()函数初识 2.2 写时拷贝 2.3 fork常规用法 2.4 fork调用失败的原因 3. 进程终止 3.1 进程场景 3.2 进程常见退出方法 4. 进程等待 4.1 进程等待必要性 4.2 进程等待的方法 4.2.1 wait方法&#xff1a; 4.…

2024年起重信号司索工(建筑特殊工种)证模拟考试题库及起重信号司索工(建筑特殊工种)理论考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年起重信号司索工(建筑特殊工种)证模拟考试题库及起重信号司索工(建筑特殊工种)理论考试试题是由安全生产模拟考试一点通提供&#xff0c;起重信号司索工(建筑特殊工种)证模拟考试题库是根据起重信号司索工(建筑特…

2.9.GoogLeNet

GoogLeNet ​ 主要解决了什么样大小的卷积核是最合适的&#xff1a;有时使用不同大小的卷积核组合是有利的 1.Inception块 ​ Inception块由四条并行路径组成。 前三条路径使用窗口大小为11、33和55的卷积层&#xff0c;从不同空间大小中提取信息。 ​ 中间的两条路径在输入…

谷粒商城实战-58-商品服务-API-三级分类-删除-批量删除小结

文章目录 一&#xff0c;增加一个批量删除的按钮并绑定事件二&#xff0c;全栈工程师三&#xff0c;逆向工程在全栈开发中的应用提升效率的方式&#xff1a;使用案例&#xff1a; 这一节的主要内容是开发批量删除分类的功能。 一&#xff0c;增加一个批量删除的按钮并绑定事件 …

zh echarts样式

记录一下&#xff1a; 一个图的配置 在echarts官网demo界面 option {title: {text: },legend: {data: [xxx前, xxx后]},radar: {// shape: circle,name: {// 雷达图各类别名称文本颜色textStyle: {color: #000,fontSize: 16}},indicator: [{ name: 完整性, max: 1 },{ name:…

【无标题】shell脚本的基本命令+编写shell脚本

shell脚本 一.shell基础 1.shell概念 2.shell脚本 3.shell脚本编写注意事项 二.编写shell脚本 1.编写一个helloworld脚本&#xff0c;运行脚本 [rootshell ~]# vim helloworld.sh #!/bin/bash //声明 echo "hello world!" ls -lh /etc/ 运行脚本(四种方式)&…

C语言字符函数与字符串函数超详解

文章目录 前言1. 字符分类函数2. 字符转换函数3. strlen3. 1 strlen 的使用3. 2 strlen 的模拟实现 4. strcpy4. 1 strcpy 的使用4. 2 strcpy 的模拟实现 5. strcat5. 1 strcat 的使用5. 2 strcat 的模拟实现 6. strcmp6. 1 strcmp 的使用6. 2 strcmp 的模拟实现 7. strncpy 函…

VI/VIM编辑器及三种模式

目录 1. 三种模式 2. 使用 VIM 3. i/ a/ o 进入输入模式 VI/VIM是 visual interface 的缩写是 Linux 中最经典的文本编辑器&#xff1b; VIM是 VI 的增强版本&#xff0c;兼容 VI 的所有指令&#xff0c;不仅能够编辑文本&#xff0c;还具有 shell 程序编辑的功能&#xff…

maven引入了jar包但在class文件里找不到jar包里的类

在工作当中遇到的这个问题&#xff0c;别人引入的jar包&#xff0c;我代码里报错 maven clean 和 maven install 都不管用 检查过了pom文件 检查了maven仓库路径下是否有这个cn.hutool的jar包 都没有找到问题 最终解决办法是手动引入 步骤一&#xff1a;点击左上角file->…

3.4-GRU

1网络结构 1.1与LSTM相比 LSTM里面有三个门&#xff0c;还有一个增加信息的tanh单元&#xff0c;参数量相较于RNN显著增加&#xff1b; 因此GRU在参数上比LSTM要少&#xff1b; 另外&#xff0c;LSTM 将必要信息记录在记忆单元中&#xff0c;并基于记忆单元的信息计算隐藏状…

MySQL数据库(基础篇)

&#x1f30f;个人博客主页&#xff1a;心.c 前言&#xff1a;今天讲解的是MySQL的详细知识点的&#xff0c;希望大家可以收货满满&#xff0c;话不多说&#xff0c;直接开始搞&#xff01; &#x1f525;&#x1f525;&#x1f525;文章专题&#xff1a;MySQL &#x1f63d;感…

1.c#(winform)编程环境安装

目录 安装vs创建应用帮助查看器安装与使用&#xff08; msdn&#xff09; 安装vs 安装什么版本看个人心情&#xff0c;或者公司开发需求需要 而本栏全程使用vs2022进行开发c#&#xff0c;着重讲解winform桌面应用开发 使用***.net framework***开发 那先去官网安装企业版的vs…

AI绘画入门实践 | Midjourney:使用 --chaos 给图像风格来点惊喜

在 Midjourney 中&#xff0c;--chaos 影响初始图像网格的多样性&#xff0c;指 MJ 每次出的4张图之间的差异性。 默认值为0&#xff0c;值越高&#xff0c;差异性越大。 使用格式&#xff1a;--chaos 0-100的整数值 使用演示 a lot of flowers --chaos 0 --v 6.0a lot of fl…