【问题标题】:Class not found when I pass command line arguments传递命令行参数时找不到类
【发布时间】:2013-11-13 03:42:35
【问题描述】:

我有一个 Java 类,它的 main 方法需要参数。如果我在没有参数的情况下运行该类,我会收到这样的错误,这意味着 Java 找到了该类,运行它,并生成一个运行时 ArrayIndexOutOfBoundsException。

zookeeper@zookeeper-virtual-machine:~/zookeeper-3.4.5$ java -cp .:zookeeper-3.4.5.jar org.zookeeper.LsGroup
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 0
    at org.zookeeper.LsGroup.main(LsGroup.java:50)

但是,当我尝试使用必需的命令行参数 (zookeeper@zookeeper-virtual-machine:~/zookeeper-3.4.5$ java -cp .:zookeeper-3.4.5.jar org.zookeeper.LsGroup Test) 执行相同的语句时,我得到一个找不到类的错误:

zookeeper@zookeeper-virtual-machine:~/zookeeper-3.4.5$ java -cp .:zookeeper-3.4.5.jar org.zookeeper.LsGroup Test
Exception in thread "main" java.lang.NoClassDefFoundError: org/slf4j/LoggerFactory
    at org.apache.zookeeper.ZooKeeper.<clinit>(ZooKeeper.java:94)
    at org.zookeeper.LsGroup.connect(LsGroup.java:19)
    at org.zookeeper.LsGroup.main(LsGroup.java:50)
Caused by: java.lang.ClassNotFoundException: org.slf4j.LoggerFactory
    at java.net.URLClassLoader$1.run(URLClassLoader.java:217)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:205)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:321)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:294)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:266)
    ... 3 more
zookeeper@zookeeper-virtual-machine:~/zookeeper-3.4.5$ 

我今天早上早些时候遇到了类似的问题,但该线程上的特定问题已解决。 Why can't java find my class?

当我传递命令行参数时,为什么会出现找不到类的错误?我该如何解决这个问题?

这是java代码:

package org.zookeeper;
import java.io.IOException;
import java.util.List;
import java.util.concurrent.CountDownLatch;

import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.Watcher.Event.KeeperState;

public class LsGroup implements Watcher {

    private static final int SESSION_TIMEOUT = 5000;
    private ZooKeeper zk;
    private CountDownLatch connectedSignal = new CountDownLatch(1);

    public void connect(String hosts) throws IOException, InterruptedException {
        zk = new ZooKeeper(hosts, SESSION_TIMEOUT, this);
        connectedSignal.await();
    }

    @Override
    public void process(WatchedEvent event) { // Watcher interface
        if (event.getState() == KeeperState.SyncConnected) {
            connectedSignal.countDown();
        }
    }

    public void ls(String groupName) throws KeeperException, InterruptedException {
        String path = "/" + groupName;
        try {
            List<String> children = zk.getChildren(path, false);
            for (String child : children) {
                System.out.println(path+"/"+child);
                System.out.println(zk.getChildren(path +"/"+ child, false));
            }
        } catch (KeeperException.NoNodeException e) {
            System.out.printf("Group %s does not exist\n", groupName);
            System.exit(1);
        }
    }

    public void close() throws InterruptedException {
        zk.close();
    }

    public static void main(String[] args) throws Exception {
        LsGroup lsGroup = new LsGroup();
        lsGroup.connect(args[0]);
        lsGroup.ls(args[1]);
        lsGroup.close();
    }
}

【问题讨论】:

  • 缺少依赖的情况,将slf4j api jar和其他依赖添加到一个目录中,并加载特定目录下的所有jar

标签: java apache-zookeeper


【解决方案1】:

您的ZooKeeper 类似乎依赖于slf4j。如果你没有指定任何命令行参数,这一行

lsGroup.connect(args[0]);

将因您看到的ArrayIndexOutOfBoundsException 而失败。因此,ZooKeeper 类将永远不会被初始化,并且对slf4j 的依赖将永远不会被 JVM 解决。

如果你确实指定了命令行参数。 JVM 将尝试加载您的类但失败,因为slf4j 不在类路径中。您需要将相关的 jars 添加到您的类路径中。

【讨论】:

    【解决方案2】:

    java 类路径缺少 slf4j 实现(我猜是 slf4j.jar)。

    这个 jar 是否捆绑到您的 JAR 中? 还是这个 JAR 在类路径中的某个地方?

    您需要提供所有类的依赖项才能运行它

    【讨论】:

    • 其实是问题
    【解决方案3】:

    将所有 api jar 打包到一个目录中

    在 linux 环境中使用以下选项

    java -cp 'PATH_TO_DIR/*' 类名

    【讨论】:

      猜你喜欢
      • 2016-03-25
      • 2010-11-11
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2020-08-24
      • 2018-04-07
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多