java - 找不到功能 'default' 的元工厂;这通常意味着 axiom-impl.jar 不在类路径中

标签 java apache-axis axiom

来自下面链接中的答案Link

我发现可以通过将其添加到classpath中来解决。但我正在使用自定义类加载器来加载 jar axiom-impl-1.2.14。 有什么办法可以实现这一点吗?

axiom jar 正在使用 ClassLoader。枚举 getResources(String name) 在 jar 内部加载该 xml。在我们的例子中,XML 文件位于 jar 文件中。所以我正在寻找可以获取 XML 文件 URL 的解决方案。

源代码:

公共(public)类ExternalClassLoader扩展了ClassLoader {

private String jarFile = "";
private Hashtable<String, Class> classes = new Hashtable<String, Class>();

public ExternalClassLoader(String jarLocation) {
    super(ExternalClassLoader.class.getClassLoader());
    this.jarFile = jarLocation;
}

@Override
public Class loadClass(String className) throws ClassNotFoundException {
    return findClass(className);
}

@Override
public Class findClass(String className) {

    byte classByte[];
    Class result = null;
    System.out.println("CLASS : " + className);
    result = (Class) classes.get(className);
    if (result != null) {
        return result;
    }
    try {
        return findSystemClass(className);
    } catch (Exception e) {
    }
    JarFile jar = null;
    try {
        jar = new JarFile(jarFile);
        String classLocation = className.replace('.', '/');
        JarEntry entry = jar.getJarEntry(classLocation + ".class");
        InputStream is = jar.getInputStream(entry);
        ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
        int nextValue = is.read();

        while (-1 != nextValue) {
            byteStream.write(nextValue);
            nextValue = is.read();
        }

        classByte = byteStream.toByteArray();
        result = defineClass(className, classByte, 0, classByte.length, null);
        classes.put(className, result);
        return result;
    } catch (Exception e) {
        System.out.println("ERROR CLASS : " + className);
        return null;
    } finally {
        try {
            jar.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

}

@Override
public InputStream getResourceAsStream(String name) {
    try {
        System.out.println("RESOURCE : " + jarFile + "//" + name);
        JarFile jar = new JarFile(jarFile);
        JarEntry entry = jar.getJarEntry(name);
        return jar.getInputStream(entry);
    } catch (IOException e) {
        System.out.println("ERROR RESOURCE : " + jarFile + "//" + name);
        return null;
    }
}

}

最佳答案

由于您没有指定详细信息,我假设与类加载器的类路径中的另一个版本的 axiom-impl 发生冲突,应用程序的其余部分已加载(否则您可以只使用一个或多个 URLClassLoader 实例或更改应用程序类加载器的类加载策略)。

我还假设(正如您在评论中提到的)axiom-apiaxiom-impl 均由同一个自定义类加载器加载,或者您认为将这两个 JAR 中的类合并到一个 JAR 中(在这种情况下,我假设您不在同一个 JAR 中包含 axiom-dom,因为这会导致其他问题)。

如果这些假设成立,那么您需要的是一个类加载器,它从一个或多个 JAR 文件加载类,并使用父级最后作为类加载策略。为了实现这一点,您不需要像您在发布的代码中尝试那样重新实现 JAR 加载逻辑。相反,您可以使用 URLClassLoader,但您需要扩展它以将其类加载策略从默认的父级加载更改为父级加载最后。 Apache Axiom 本身的源代码中实际上有一个这样的示例:

http://svn.apache.org/repos/asf/webservices/axiom/tags/1.2.15/axiom-api/src/test/java/org/apache/axiom/util/stax/dialect/ParentLastURLClassLoader.java

您可能可以按原样使用该代码,尽管您可能希望删除 javax.* 上的包过滤器,因为在您的情况下这不是必需的。

关于java - 找不到功能 'default' 的元工厂;这通常意味着 axiom-impl.jar 不在类路径中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33099823/

相关文章:

java - XML 响应消息中的服务器名称?

java - 使用 ArangoDB 和 Java 实现最短路径

java - 在 Google Appengine 上配置 https ://yourdomain. com

java - 寻找允许修改源代码的 java (Android) 面向方面的框架

java - 如何使用 Jackson-Databind 序列化 java.time 类的类型信息?

java - 使用 Axiom 流式传输 XML 节点的内容

java - Apache Axis 1.x xml 有效负载

java - Web 服务抛出 ClassNotFoundException

伊莎贝尔:公理化和快速检查与自动解决直接

java - 如何通过给定标签名称从 XML 中删除特定元素。公理