“找不到或加载主类”是啥意思?

Posted

技术标签:

【中文标题】“找不到或加载主类”是啥意思?【英文标题】:What does "Could not find or load main class" mean?“找不到或加载主类”是什么意思? 【发布时间】:2019-02-20 17:38:22 【问题描述】:

新的 Java 开发人员遇到的一个常见问题是他们的程序无法运行并显示错误消息:Could not find or load main class ...

这是什么意思,是什么原因造成的,应该如何解决?

【问题讨论】:

请注意,这是一个“自我回答”问题,旨在为新 Java 用户提供通用参考问答。我找不到足以涵盖此问题的现有问答(IMO)。 【参考方案1】:

java <class-name> 命令语法

首先,您需要了解使用java(或javaw)命令启动程序的正确方法。

正常的语法1是这样的:

    java [ <options> ] <class-name> [<arg> ...]

其中&lt;option&gt; 是命令行选项(以“-”字符开头),&lt;class-name&gt; 是完全限定的Java 类名,&lt;arg&gt; 是传递给应用程序的任意命令行参数。


1 - 在此答案的末尾附近还描述了一些其他语法。

类的完全限定名 (FQN) 通常按照 Java 源代码编写;例如

    packagename.packagename2.packagename3.ClassName

但是,java 命令的某些版本允许您使用斜杠而不是句点;例如

    packagename/packagename2/packagename3/ClassName

which(令人困惑)看起来像一个文件路径名,但不是一个。请注意,完全限定名称这个术语是标准的 Java 术语......不是我只是为了让你感到困惑而编造的 :-)

下面是java 命令的示例:

    java -Xmx100m com.acme.example.ListUsers fred joe bert

以上将导致java 命令执行以下操作:

    搜索 com.acme.example.ListUsers 类的编译版本。 加载类。 检查该类是否具有main 方法,该方法具有由public static void main(String[]) 提供的签名返回类型修饰符。 (注意,方法参数的名称不是签名的一部分。) 调用该方法,将命令行参数(“fred”、“joe”、“bert”)作为String[] 传递给它。

Java 找不到类的原因

当您收到消息“Could not find or load main class ...”时,表示第一步失败。 java 命令找不到类。事实上,消息中的“...”将是java 正在寻找的完全限定的类名

那为什么会找不到类呢?

原因 #1 - 你在 classname 参数上犯了一个错误

第一个可能的原因是您可能提供了错误的类名。 (或者......正确的类名,但格式错误。)考虑到上面的例子,这里有多种错误的方法来指定类名:

示例 #1 - 一个简单的类名:

java ListUser

当类在com.acme.example这样的包中声明时,您必须使用完整的类名包括java命令中的包名;例如

java com.acme.example.ListUser

示例 #2 - 文件名或路径名而不是类名:

java ListUser.class
java com/acme/example/ListUser.class

示例 #3 - 大小写错误的类名:

java com.acme.example.listuser

示例 #4 - 一个错字

java com.acme.example.mistuser

示例 #5 - 源文件名(Java 11 或更高版本除外;见下文)

java ListUser.java

示例 #6 - 你完全忘记了类名

java lots of arguments

原因 #2 - 应用程序的类路径指定错误

第二个可能的原因是类名正确,但是java 命令找不到该类。要理解这一点,您需要了解“类路径”的概念。 Oracle 文档很好地解释了这一点

The java command documentation Setting the Classpath。 Java 教程 - PATH and CLASSPATH

所以...如果您正确指定了类名,接下来要检查的是您是否正确指定了类路径:

    阅读上面链接的三个文档。 (是的...阅读它们!Java 程序员了解至少 Java 类路径机制如何工作的基础知识是很重要的。) 查看命令行和/或运行java 命令时生效的CLASSPATH 环境变量。检查目录名和 JAR 文件名是否正确。 如果类路径中有相对路径名,请检查它们是否正确解析...从运行java 命令时生效的当前目录。 检查类(在错误消息中提到)是否可以位于有效的类路径中。 请注意,Windows 与 Linux 和 Mac OS 的类路径语法不同。 (类路径分隔符在 Windows 上是 ;,在其他系统上是 :。如果您为您的平台使用了错误的分隔符,您将不会收到明确的错误消息。相反,您将在将被静默忽略的路径。)

原因 #2a - 错误的目录位于类路径中

当您将目录放在类路径上时,它在理论上对应于限定名称空间的根目录。类位于根目录下的目录结构中,通过将完全限定名映射到路径名。因此,例如,如果“/usr/local/acme/classes”在类路径上,那么当 JVM 查找名为 com.acme.example.Foon 的类时,它将查找具有此路径名的“.class”文件:

  /usr/local/acme/classes/com/acme/example/Foon.class

如果您将“/usr/local/acme/classes/com/acme/example”放在类路径中,那么 JVM 将无法找到该类。

原因 #2b - 子目录路径与 FQN 不匹配

如果你的类 FQN 是com.acme.example.Foon,那么 JVM 将在目录“com/acme/example”中查找“Foon.class”:

如果您的目录结构与上述模式中的包命名不匹配,则 JVM 将找不到您的类。

如果你尝试重命名一个类来移动它,那也会失败......但异常堆栈跟踪会有所不同。可能会这样说:

Caused by: java.lang.NoClassDefFoundError: <path> (wrong name: <name>)

因为类文件中的 FQN 与类加载器期望找到的不匹配。

举个具体的例子,假设:

你想运行com.acme.example.Foon类, 完整的文件路径是/usr/local/acme/classes/com/acme/example/Foon.class, 您当前的工作目录是/usr/local/acme/classes/com/acme/example/

然后:

# wrong, FQN is needed
java Foon

# wrong, there is no `com/acme/example` folder in the current working directory
java com.acme.example.Foon

# wrong, similar to above
java -classpath . com.acme.example.Foon

# fine; relative classpath set
java -classpath ../../.. com.acme.example.Foon

# fine; absolute classpath set
java -classpath /usr/local/acme/classes com.acme.example.Foon

注意事项:

在大多数 Java 版本中,-classpath 选项可以缩短为 -cp。检查javajavac 等的相应手册条目。 在类路径中选择绝对路径名和相对路径名时要仔细考虑。请记住,如果当前目录发生更改,相对路径名可能会“中断”。

原因 #2c - 类路径中缺少依赖项

类路径需要包含您的应用程序所依赖的所有其他(非系统)类。 (系统类是自动定位的,你很少需要关心这个。)为了正确加载主类,JVM需要找到:

类本身。 超类层次结构中的所有类和接口(例如,参见Java class is present in classpath but startup fails with Error: Could not find or load main class) 通过变量或变量声明、方法调用或字段访问表达式引用的所有类和接口。

(注意:JLS 和 JVM 规范允许 JVM 在一定范围内“延迟”加载类,这会影响何时引发类加载器异常。)

原因 #3 - 在错误的包中声明了该类

偶尔会发生有人将源代码文件放入 源代码树中的错误文件夹,或者他们遗漏了package 声明。如果您在 IDE 中执行此操作,IDE 的编译器会立即告诉您这一点。同样,如果您使用一个不错的 Java 构建工具,该工具将以检测问题的方式运行 javac。但是,如果您手动构建您的 Java 代码,您可以这样做,编译器不会注意到问题,并且生成的“.class”文件不在您期望的位置。

还是找不到问题?

要检查的东西很多,很容易漏掉一些东西。尝试将-Xdiag 选项添加到java 命令行(作为java 之后的第一件事)。它将输出有关类加载的各种信息,这可能会为您提供关于真正问题所在的线索。

另外,请考虑从网站、文档等复制和粘贴不可见或非 ASCII 字符可能导致的问题。并考虑“同形文字”,其中两个字母或符号看起来相同......但不是。

如果您在META-INF/*.SF 中有无效或不正确的签名,您可能会遇到此问题。您可以尝试在您最喜欢的 ZIP 编辑器中打开 .jar,然后从 META-INF 中删除文件,直到您只剩下您的 MANIFEST.MF。但是,一般不建议这样做。 (无效签名可能是由于有人将恶意软件注入到原始签名的 JAR 文件中。如果您删除了无效签名,您就是在用恶意软件感染您的应用程序!)推荐的方法是获取具有有效的 JAR 文件签名,或从(真实的)原始源代码重建它们。

最后,如果MANIFEST.MF 文件中存在语法错误,您显然会遇到这个问题(请参阅https://***.com/a/67145190/139985)。


java 的替代语法

使用java command 启动Java 程序有三种替代语法。

    用于启动“可执行”JAR 文件的语法如下:

    java [ <options> ] -jar <jar-file-name> [<arg> ...]
    

    例如

    java -Xmx100m -jar /usr/local/acme-example/listuser.jar fred
    

    入口点类的名称(即com.acme.example.ListUser)和类路径在 JAR 文件的 MANIFEST 中指定。

    从模块(Java 9 及更高版本)启动应用程序的语法如下:

    java [ <options> ] --module <module>[/<mainclass>] [<arg> ...]
    

    入口点类的名称要么由&lt;module&gt; 本身定义,要么由可选的&lt;mainclass&gt; 给出。

    从 Java 11 开始,您可以使用 java 命令使用以下语法编译和运行单个源代码文件:

    java [ <options> ] <sourcefile> [<arg> ...]
    

    &lt;sourcefile&gt; 是(通常)后缀为“.java”的文件。

更多详细信息,请参阅您正在使用的 Java 版本的java 命令的官方文档。


集成开发环境

典型的 Java IDE 支持在 IDE JVM 本身或子 JVM 中运行 Java 应用程序。这些通常不受此特定异常的影响,因为 IDE 使用自己的机制来构造运行时类路径、识别主类并创建 java 命令行。

但是,如果您在 IDE 背后执行操作,则仍有可能发生此异常。例如,如果您之前在 Eclipse 中为您的 Java 应用程序设置了一个应用程序启动器,然后您将包含“main”类的 JAR 文件移动到文件系统中的不同位置而不告诉 Eclipse ,Eclipse 会无意中使用错误的类路径启动 JVM。

简而言之,如果您在 IDE 中遇到此问题,请检查过时的 IDE 状态、损坏的项目引用或损坏的启动器配置等。

IDE 也可能会简单地感到困惑。 IDE 是非常复杂的软件,包含许多交互部分。其中许多部分采用各种缓存策略,以使 IDE 作为一个整体响应。这些有时会出错,一个可能的症状是启动应用程序时出现问题。如果您怀疑这可能会发生,那么值得尝试其他方法,例如重新启动 IDE、重建项目等等。


其他参考

来自 Oracle Java 教程 - Common Problems (and Their Solutions)

【讨论】:

我在尝试使用 3rd 方库运行 Class 时遇到了这个问题。我这样调用java:java -cp ../third-party-library.jar com.my.package.MyClass;这不起作用,而是需要将本地文件夹也添加到类路径(由:分隔,像这样:java -cp ../third-party-library.jar:. com.my.package.MyClass,那么它应该可以工作 经过多年的 java 编程,我仍然设法结束了这个页面。对我来说,问题是the classpath syntax is OS-dependent。我对在 Windows 上编程有点陌生,不知道。 补充说明,第2点救救我!很遗憾看到java 并没有说它没有找到导入的类,而是说你试图运行的主类。这是一种误导,尽管我确信这是有原因的。我遇到了java 确切知道我的班级在哪里的情况,但是它找不到导入的班级之一。它没有这么说,而是抱怨找不到我的主要课程。真的,烦人。 我在 Eclipse 中遇到过两次这个问题。第一次 main() 的签名是错误的。第二次我重命名了一个 .jar,即使我将新的添加到构建路径中,Eclipse 也没有找到旧的,所以项目没有编译,出现这个错误。我不得不从 Project > Properties > Java Build Path > Libraries 中删除 .jar 文件。 我已经第三次遇到了。我从 Windows 10 批处理文件运行程序,并将 .jar 名称放入变量中(使用“-cp %jarname%;lib*”调用)。我错误地在 jarname 的末尾添加了一个额外的空格,这导致了错误。帽子戏法:)【参考方案2】:

如果您的源代码名称是 HelloWorld.java,那么您的编译代码将为 HelloWorld.class

如果您使用以下方式调用它,您将收到该错误:

java HelloWorld.class

改用这个:

java HelloWorld

【讨论】:

问题是这个解决方案只适用于在默认包中声明的没有JAR文件依赖的Java类。 (即便如此,也并非总是如此。)大多数 Java 程序并不是那么简单。 就像斯蒂芬说的那样,这只适用于“默认包” - 这意味着文件顶部的 no 包声明。为了快速测试一些代码,我做了:javac TestCode.java,然后是java TestCode 这对我不起作用。它仍然说,“无法找到或加载主类 HelloWorld” 我需要做java -classpath . HelloWorld @ChrisPrince - 是的......这工作......有时。要了解它何时有效,何时无效,请阅读投票最多的答案。【参考方案3】:

如果您的类在包中,那么您必须 cd 到项目的根目录并使用类的完全限定名称 (packageName.MainClassName) 运行。

例子:

我的课程在这里:

D:\project\com\cse\

我的主类的完全限定名称是:

com.cse.Main

于是我cd回到项目根目录:

D:\project

然后发出java 命令:

java com.cse.Main

这个答案是为了让新手 java 程序员摆脱一个常见错误造成的挫败感,我建议你阅读接受的答案,以更深入地了解 java 类路径。

【讨论】:

这个答案做了很多假设。还有其他方法可以实现这一目标。我建议人们不要盲目地遵循上述建议,而是花时间阅读我的答案中解释 Java 类路径如何工作的链接。最好了解你在做什么...... 这个答案做出了我需要的确切假设:) 我位于 .class 文件的目录中,而 java.exe 不起作用。一旦我在上面 cd-ed 并使用命令行中包含的包名运行它就可以了。 我同意尼克·康斯坦丁的观点。同样在这里。这是一个遵循我所做步骤的确切示例,它也对我有用。 java类路径处理有一定的逻辑,但我肯定不会这样设计的。 Tnanks,你的回答帮助我看看我使用 Main 和大写 _ 我在尝试这个之后写了这个评论,但它没有用。我有一个名为 Helloworld 的项目,它只包含 1 个 java 文件,Helloworld/src/com/firstpackage/Test.java(windows 10.intellij idea)。 我没有 CLASSPATH 环境变量,我想专门为这个项目设置类路径。在 Helloworld 目录中运行 java com.firstpackage.Test 不起作用,命令 java -classpath C:\Users\matuagkeetarp\IdeaProjects\Helloworld\src\com\first‌​package Test.java 也没有设置类路径变量。你能帮忙吗?【参考方案4】:

带包

如果你的源代码中有package关键字(主类定义在一个包中),你应该在分层目录上运行它,使用类的全名( packageName.MainClassName)。

假设有一个源代码文件(Main.java):

package com.test;

public class Main 

    public static void main(String[] args) 
        System.out.println("salam 2nya\n");
    

要运行此代码,您应该将Main.Class 放在 package like 目录中:

C:\Users\workspace\testapp\com\test\Main.Java

然后将终端的当前目录更改为项目的根目录:

cd C:\Users\workspace\testapp  

最后,运行代码:

java com.test.Main

无包装

如果您的源代码名称上没有任何包,则可能是您输入了错误的命令。假设你的java文件名是Main.java,编译后:

javac Main.java

您的编译代码将是Main.class

如果您使用以下方式调用它,您将收到该错误:

java Main.class

改用这个:

java Main

【讨论】:

请参阅我的答案的“附加说明 #1”。为了更好地解释这个问题。 @StephenC 是的,你的答案更完整(当然,+1),但这个特定的答案中有“包”这个词,这让我可以快速找到我需要的东西。它奏效了。所以+1拉扎维。 StephenC,你的缺少我需要的简单包示例,因为我是 Java 新手。 这正是我的问题。我一直在阅读大量的 Java 文档,而这个具体的例子正是我所需要的 是的,具体的例子很好,这很好用。我敢肯定,主要答案是非常彻底的,但是很难看到森林中的树。不错的@Razavi 我喜欢这个简短而有用的答案,而不是接受的答案!【参考方案5】:

当相同的代码在一台 PC 上运行,但在另一台 PC 上显示错误时,我发现的最佳解决方案是编译如下:

javac HelloWorld.java
java -cp . HelloWorld

【讨论】:

这不是一个好的建议。您取决于未设置的 CLASSPATH 环境变量,或者具有与“。”一致的值。是的,它在许多情况下都有效,但在其他情况下则无效。 当然javac -classpath . HelloWorld.java 会起作用的!在您的情况下,这是一个更好的解决方案。 如果您将 'package com.some.address' 作为第一行 - 这将不起作用。您需要注释掉“包裹地址”.. @Joe - hack(注释掉包)会起作用(在某些情况下),但这是个坏主意。更好的办法是学习/了解导致问题的原因并实施正确的解决方案。 取消设置类路径变量基本上对我有用。【参考方案6】:

帮助我的是在命令行上指定类路径,例如:

    新建文件夹C:\temp

    C:\temp 中创建文件 Temp.java,其中包含以下类:

    public class Temp 
        public static void main(String args[]) 
            System.out.println(args[0]);
        
    
    

    在文件夹C:\temp中打开命令行,编写以下命令编译Temp类:

    javac Temp.java
    

    运行编译后的 Java 类,添加 -classpath 选项让 JRE 知道在哪里可以找到该类:

    java -classpath C:\temp Temp Hello!
    

【讨论】:

在 Ubuntu 中,我还必须指定路径。不明白为什么默认情况下它不能使用当前工作目录。我确信 Java 是由键盘制造商赞助的! @gone - “.”的原因默认情况下不在 $PATH 中,这是一个安全陷阱。 seas.upenn.edu/cets/answers/dot-path.html 非常感谢......即使不知道为什么 java 在环境变量中设置后也无法找到类路径。 @akash89 - 最可能的原因是:1) java 没有查看 $CLASSPATH(因为您使用了 -classpath 或 -jar)或 2) 未在在运行java 的上下文中未生效的环境;例如因为您没有在正确的 shell 中“获取”添加 setenv 命令的文件。 我仍然收到错误:无法找到或加载主类 Temp 任何人都可以帮忙!【参考方案7】:

根据报错信息(“Could not find or load main class”),有两类问题:

    无法找到主类 无法加载主类(已接受的答案中未充分讨论这种情况)

无法找到主类,当完全限定的类名中有错字或语法错误或它在提供的类路径中不存在强>.

无法启动类时,无法加载主类,通常主类扩展了另一个类,并且该类不存在于提供的类路径中。 p>

例如:

public class YourMain extends org.apache.camel.spring.Main

如果不包含camel-spring,会报这个错误。

【讨论】:

“基本上”还有很多其他类别。缺少超类问题是一个非常不寻常的子案例。 (如此不寻常,我从未见过它......在这个网站上提出的问题中。) 有两个,因为错误显示“无法查找或加载主类”。如果还有其他类别,请告诉我。我已经看过了,所以只想在这里分享它,也许其他人会需要它。 我会将其修改为“您需要包含启动主类所需的所有类以避免此特定错误”。我不是想说服你。这只是我想看到的一种方式。我把答案留在这里只是为了那些可能喜欢以这种方式阅读内容的人。让我们不要进一步扩展这个讨论:) 我将我的声明更改为“在接受的答案中没有完全讨论”,希望你感觉更好。 此信息至关重要,值得明确提及(这是唯一提及 extends 的答案)。我刚刚了解到,当主类无法加载时,因为它扩展了另一个无法找到的类,java 不报告哪个实际类是未找到(与NoClassDefFoundError 不同)。所以是的,它确实发生了,当你不知道这一点时,这是一个令人毛骨悚然的情况。 在这种情况下,有什么方法可以准确判断哪个依赖类加载失败?【参考方案8】:

使用这个命令:

java -cp . [PACKAGE.]CLASSNAME

示例:如果您的类名是从 Hello.java 创建的 Hello.class,则使用以下命令:

java -cp . Hello

如果您的文件 Hello.java 在 com.demo 包中,则使用以下命令

java -cp . com.demo.Hello

在 JDK 8 中,类文件经常出现在同一个文件夹中,但 java 命令需要类路径,因此我们添加 -cp . 以获取当前文件夹作为类路径的参考。

【讨论】:

这只适用于简单的情况。更复杂的情况需要更复杂的类路径。 对于 >>really-cp . 是不必要的,因为如果 $CLASSPATH 未设置,那么 . 是默认的类路径。 没有斯蒂芬,很多时候在 Windows 默认类路径中不起作用。我在三台不同的机器上试过,你也可以试试。 这可能是因为您实际上在某处设置了 %CLASSPATH% 环境变量。如果你这样做,那么你没有使用默认的类路径。 (echo %CLASSPATH% 输出什么?)不,我无法检查,因为我没有 Windows PC。 当我尝试从命令行运行一个简单程序时,这对我有用【参考方案9】:

在这种情况下我遇到了这样的错误:

java -cp lib.jar com.mypackage.Main

它适用于 Windows 的 ; 和 Unix 的 :

java -cp lib.jar; com.mypackage.Main

【讨论】:

是的。这很可能是因为您的 Main 不在 JAR 文件中。 -cp lib.jar;-cp lib.jar;. 的含义相同,即当前目录包含在类路径中。 终于解决了unix的问题..谢谢(与:)合作【参考方案10】:

试试 -Xdiag

Steve C's answer 很好地涵盖了可能的情况,但有时要确定类是无法找到还是加载可能并不那么容易。使用 java -Xdiag(从 JDK 7 开始)。这会打印出一个很好的堆栈跟踪,它提供了消息 Could not find or load main class 消息含义的提示。

例如,它可以将您指向主类使用的其他类,这些类无法找到并阻止加载主类。

【讨论】:

【参考方案11】:

有时可能导致问题的原因与主类无关,我不得不以艰难的方式找出这一点。这是我移动的一个引用库,它给了我:

无法找到或加载主类 xxx Linux

我刚刚删除了那个引用,再次添加它,它又可以正常工作了。

【讨论】:

听起来问题是由于 IDE 项目中的“引用”损坏,您的类路径不正确。我将更新我的答案以涵盖该案例。 @StephenC 和 EduardoDennis,这里也缺少一个 jar,该 jar 包含一个接口,主类依赖于该接口被实例化。所以,错误信息太宽泛了。如果找不到类文件,我应该说“找不到”,如果缺少其他东西但不是文件本身,我应该说“无法加载(缺少依赖项)”,所以如果你只关注,错误消息太宽泛会误导在它的“查找”部分:( @AquariusPower - 对于“原因”异常应该有一个额外的“由”堆栈跟踪,表示缺少类。如果您想向 Java 开发人员建议他们更改 20 多年以来一直在说的错误消息......请随意。 (我认为错误消息是正确的。问题是 >>you @StephenC 我的意思是,如果主类文件可用或不可用,他们肯定可以访问信息,所以为什么不向我们显示一个更好的错误消息,说明缺少此类文件。另一方面,他们也可以说“找到了文件但无法加载”,我们会及时关注依赖关系,而不是浪费半天时间研究和测试来理解。只是我的意思是:)。他们可能会在 20 多年内以有限的方式做到这一点,但他们可以改进它,我们在这里通过我们的批评和投诉来保证这将发生! :D 请理解>>我 【参考方案12】:

我遇到了同样的问题,终于找到了我的错误:) 我使用这个命令进行编译,它工作正常:

javac -cp "/home/omidmohebbi/AAAATest/jars/core-1.7.jar:/home/omidmohebbi/AAAATest/jars/javase-1.7.jar:/home/omidmohebbi/AAAATest/jars/qrgen-1.2.jar" qrcode.java

但是这个命令对我不起作用(我无法找到或加载主类,qrcode):

java -cp "/home/omidmohebbi/AAAATest/jars/core-1.7.jar:/home/omidmohebbi/AAAATest/jars/javase-1.7.jar:/home/omidmohebbi/AAAATest/jars/qrgen-1.2.jar" qrcode

最后我只是在类路径末尾添加了':'字符,问题就解决了:

java -cp "/home/omidmohebbi/AAAATest/jars/core-1.7.jar:/home/omidmohebbi/AAAATest/jars/javase-1.7.jar:/home/omidmohebbi/AAAATest/jars/qrgen-1.2.jar:" qrcode

【讨论】:

【参考方案13】:

在这种情况下,您有:

找不到或无法加载主类?classpath

这是因为您使用的是“-classpath”,但破折号与java 在命令提示符下使用的破折号不同。我从Notepad 复制粘贴到cmd 时遇到了这个问题。

【讨论】:

哇!这是一个完全奇怪的原因! (但它适合您使用记事本而不是真正的文本编辑器:-))【参考方案14】:

就我而言,出现错误是因为我提供了源文件名而不是类名。

我们需要将包含 main 方法的类名提供给解释器。

【讨论】:

是的。请参阅我的示例 #2 指定类名的错误方法!【参考方案15】:

如果使用Maven构建JAR文件,请务必在pom.xml文件中指定主类:

<build>
    <plugins>
        <plugin>
            <artifactId>maven-jar-plugin</artifactId>
            <configuration>
                <archive>
                    <manifest>
                        <mainClass>class name us.com.test.abc.MyMainClass</mainClass>
                    </manifest>
                </archive>
            </configuration>
        </plugin>
    </plugins>
</build>

【讨论】:

【参考方案16】:

如果您的情况与我的情况特别相似,这可能会对您有所帮助:作为初学者,我在尝试运行 Java 程序时也遇到了这个问题。

我是这样编译的:

javac HelloWorld.java

我也尝试使用相同的扩展名运行:

java Helloworld.java

当我删除.java 并重写java HelloWorld 之类的命令时,程序运行完美。 :)

【讨论】:

这是因为您正在执行 .java 的编译版本。它实际上是在执行.class文件 作为记录,这与我的答案中的原因 #1、示例 #5 相同...【参考方案17】:

这里的所有答案似乎都是针对 Windows 用户的。对于 Mac,类路径分隔符是 :,而不是 ;。由于没有抛出使用; 设置类路径的错误,因此如果从 Windows 到 Mac,这可能很难发现。

这里是对应的Mac命令:

java -classpath ".:./lib/*" com.test.MyClass

在此示例中,包是com.test,并且lib 文件夹也将包含在类路径中。

【讨论】:

在 Linux 上就像在 Mac 上一样。 为什么需要/* 是通配符语法。 (这不是强制性的。如果需要,您可以明确列出 JAR。)【参考方案18】:

类文件位置: C:\test\com\company

文件名: Main.class

完全限定的类名: com.company.Main

命令行命令:

java  -classpath "C:\test" com.company.Main

请注意,类路径不包括 \com\company

【讨论】:

【参考方案19】:

我花了相当多的时间试图解决这个问题。我以为我以某种方式错误地设置了我的类路径,但问题是我输入了:

java -cp C:/java/MyClasses C:/java/MyClasses/utilities/myapp/Cool  

代替:

java -cp C:/java/MyClasses utilities/myapp/Cool   

我认为完全限定的意思是包含完整的路径名而不是完整的包名。

【讨论】:

我已经更新了我的答案,试图解决这种困惑。 这些都不正确。该类必须以utilities.myapp.Cool 或其包名(如果有)的形式给出。【参考方案20】:

这是一个具体的案例,但是由于我来到这个页面寻找解决方案并没有找到它,所以我将它添加到这里。

Windows(用 7 测试)不接受类和包名称中的特殊字符(如 á)。不过,Linux 可以。

我在 NetBeans 中构建 .jar 并尝试在命令行中运行它时发现了这一点。它在 NetBeans 中运行,但不在命令行中。

【讨论】:

【参考方案21】:

在我的案例中解决问题的是:

右击你想运行的项目/类,然后Run As->Run Configurations。然后,您应该修复现有配置或通过以下方式添加新配置:

打开Classpath标签,点击Advanced...按钮,然后添加您项目的bin文件夹

【讨论】:

【参考方案22】:

在 Windows 上,将.; 放在开头的 CLASSPATH 值中。

. (点)表示“查看当前目录”。这是一个永久的解决方案。

您也可以使用 set CLASSPATH=%CLASSPATH%;. 将其设置为“一次性”。只要您的 cmd 窗口打开,这将持续。

【讨论】:

这个建议可能有帮助,也可能没有帮助。如果类树包含当前目录中的类,这将有所帮助。如果他们不是,就不会。我实际上不会这样做。相反,我会创建一个单行包装脚本,无论用户是否在“正确”目录中,它都可以工作。【参考方案23】:

先用这个命令设置路径;

set path="paste the set path address"

然后你需要加载程序。在存储的驱动器中键入“cd(文件夹名称)”并编译它。例如,如果我的程序存储在 D 盘,输入“D:”按回车并输入“cd(文件夹名称)”。

【讨论】:

这没有帮助。这个问题是关于 Java 程序,而不是普通的可执行文件。 Java 不使用 PATH 来定位任何东西,如果“cd”有帮助,那么它是靠运气而不是靠判断。 if "cd" helps then it by luck rather than by judgement。这是错误的(我相信),因为 java 默认使用当前目录 . 作为类路径的一部分。 @GKFX - 这就是我的意思。除非您知道您使用的是默认类路径(或带有“.”的类路径),否则“cd”将不起作用。这个解决方案更多地靠运气(即猜测/希望“。”在类路径上)而不是通过判断(即检查“。”在类路径上)。此外,您对默认值不正确。 Java 使用“。”默认情况下作为类路径,而不是默认情况下作为类路径的一部分。【参考方案24】:

在 Java 中,当您有时使用 java 可执行文件从命令行运行 JVM 并尝试从具有 public static void main (PSVM) 的类文件启动程序时,您可能会遇到以下错误,即使JVM 的类路径参数是准确的,并且类文件存在于类路径中:

Error: main class not found or loaded

如果无法加载带有 PSVM 的类文件,则会发生这种情况。一个可能的原因是该类可能正在实现一个接口或扩展另一个不在类路径上的类。通常,如果一个类不在类路径上,则抛出的错误就是这样。但是,如果正在使用的类被扩展或实现,java 就无法加载该类本身。

参考:https://www.computingnotes.net/java/error-main-class-not-found-or-loaded/

【讨论】:

您是否阅读了接受的答案?你的回答有什么新内容吗? @StephenC 我试图通过查看“原因”类别及其要点在您的列表中找到原因。我在“原因 #1”和“原因 #2”标题中找不到匹配点,这与我的情况并不接近(因为我确信类路径本身没有问题)。我通过实验找到了原因,我很惊讶在我的例子中显示“找不到主类”错误,因为实现接口不在类路径上。当然你可以说“你应该阅读帖子中描述的所有内容”,但在我看来你的理由列表可以改进。【参考方案25】:

您确实需要从src 文件夹中执行此操作。在此处键入以下命令行:

[name of the package].[Class Name] [arguments]

假设您的课程名为CommandLine.class,代码如下所示:

package com.tutorialspoint.java;

    /**
     * Created by mda21185 on 15-6-2016.
     */

    public class CommandLine 
        public static void main(String args[])
            for(int i=0; i<args.length; i++)
                System.out.println("args[" + i + "]: " + args[i]);
            
        
    

那么你应该 cd 到 src 文件夹,你需要运行的命令如下所示:

java com.tutorialspoint.java.CommandLine this is a command line 200 -100

命令行的输出是:

args[0]: this
args[1]: is
args[2]: a
args[3]: command
args[4]: line
args[5]: 200
args[6]: -100

【讨论】:

一个类不能称为“CommandLine.class”。那将是一个 Java 语法错误。 (您的意思是包含已编译类的文件称为“CommandLine.class”...)。另一个问题是,“cd 到源目录”的指令只有在将代码 >> 编译到 在我的项目中,我的根目录下有 src 文件夹和 bin 文件夹。我必须将cd 转换为src,然后运行对我有用的命令java ../bin com.blah.blah.MyClass。所以谢谢你的提示!【参考方案26】:

当运行 java 和 Windows PowerShell 中所宣传的 -cp 选项时,您可能会收到类似于以下内容的错误:

The term `ClassName` is not recognized as the name of a cmdlet, function, script ...

为了让 PowerShell 接受命令,-cp 选项的参数必须包含在引号中,如下所示:

java -cp 'someDependency.jar;.' ClassName

以这种方式形成命令应该允许 Java 正确处理类路径参数。

【讨论】:

【参考方案27】:

我在测试 Java MongoDB JDBC 连接时也遇到了类似的错误。我认为最好简短地总结一下我的最终解决方案,以便将来任何人都可以直接查看这两个命令并继续进行下去。

假设您位于您的 Java 文件和外部依赖项(JAR 文件)所在的目录中。

编译:

javac -cp mongo-java-driver-3.4.1.jar JavaMongoDBConnection.java
-cp - 类路径参数;一个一个地传递所有依赖的 JAR 文件 *.java - 这是具有 main 方法的 Java 类文件。 sdsd

运行:

java -cp mongo-java-driver-3.4.1.jar: JavaMongoDBConnection
请注意所有依赖 JAR 文件结束后的冒号 (Unix) / 逗号 (Windows) 最后,观察没有任何扩展名的主类名(没有.class或.java)

【讨论】:

这一切都假设 1) JavaMongoDBConnection 没有包,并且 2) 你没有更改目录。至少可以说,它是脆弱的。由于不解释问题,它会导致新手尝试这种方法在它不起作用的情况下。简而言之,它鼓励“巫毒编程技术”:en.wikipedia.org/wiki/Voodoo_programming【参考方案28】:

好吧,已经有很多答案了,但是没有人提到文件权限可能是罪魁祸首的情况。

运行时,用户可能无法访问 JAR 文件或路径的目录之一。例如,考虑:

/dir1/dir2/dir3/myjar.jar中的jar文件

拥有 JAR 文件的用户 1 可以这样做:

# Running as User1
cd /dir1/dir2/dir3/
chmod +r myjar.jar

但还是不行:

# Running as User2
java -cp "/dir1/dir2/dir3:/dir1/dir2/javalibs" MyProgram
Error: Could not find or load main class MyProgram

这是因为正在运行的用户 (User2) 无权访问 dir1、dir2 或 javalibs 或 dir3。当用户 1 可以看到文件并且可以访问它们时,它可能会让人抓狂,但用户 2 仍然会发生错误。

【讨论】:

【参考方案29】:

执行mvn eclipse:eclipse 后出现此错误 这有点搞砸了我的.classpath 文件。

必须将.classpath 中的行从

<classpathentry kind="src" path="src/main/java" including="**/*.java"/>
<classpathentry kind="src" path="src/main/resources" excluding="**/*.java"/>

<classpathentry kind="src" path="src/main/java" output="target/classes" />
<classpathentry kind="src" path="src/main/resources" excluding="**"  output="target/classes" />

【讨论】:

【参考方案30】:

我无法使用此处所述的解决方案来解决此问题(尽管所述答案无疑已经清除了我的概念)。我曾两次遇到这个问题,每次都尝试了不同的解决方案(在 Eclipse IDE 中)。

首先,我在项目的不同类中遇到了多个main 方法。因此,我从后续课程中删除了main 方法。 其次,我尝试了以下解决方案:
    右键单击我的主项目目录。 前往源代码,然后清理并坚持使用默认设置并点击完成。在完成一些后台任务后,您将被定向到您的主项目目录。 之后我关闭了我的项目,重新打开它,然后繁荣,我终于解决了我的问题。

【讨论】:

删除 main 方法不会解决问题。具有多个入口点的应用程序在技术上没有任何问题。

以上是关于“找不到或加载主类”是啥意思?的主要内容,如果未能解决你的问题,请参考以下文章

“找不到或加载主类”是啥意思?

通过 jpackage 打包找不到或加载主类

Spring Boot gradle“找不到或加载主类”

Java JAR:找不到或加载主类

openmpi + java,找不到或加载主类

IntelliJ 构建错误的 JAR:找不到或加载主类