- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试实现用于教育目的的自定义类加载器。
我在 jar
文件中有模块“Weather”,我希望通过 JarClassLoader
从 App
类加载该模块。
来自here的类加载器(它从指定的 jar 加载所有类):
package com.example.classloading;
import java.io.IOException;
import java.io.InputStream;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
public class JarClassLoader extends ClassLoader {
private HashMap<String, Class<?>> cache = new HashMap<String, Class<?>>();
private String jarFileName;
private String packageName;
private static String WARNING = "Warning : No jar file found. Packet unmarshalling won't be possible. Please verify your classpath";
public JarClassLoader(String jarFileName, String packageName) {
this.jarFileName = jarFileName;
this.packageName = packageName;
cacheClasses();
}
private void cacheClasses() {
try {
JarFile jarFile = new JarFile(jarFileName);
Enumeration entries = jarFile.entries();
while (entries.hasMoreElements()) {
JarEntry jarEntry = (JarEntry) entries.nextElement();
// simple class validation based on package name
if (match(normalize(jarEntry.getName()), packageName)) {
byte[] classData = loadClassData(jarFile, jarEntry);
if (classData != null) {
Class<?> clazz = defineClass(stripClassName(normalize(jarEntry.getName())), classData, 0, classData.length);
cache.put(clazz.getName(), clazz);
System.out.println("== class " + clazz.getName() + " loaded in cache");
}
}
}
}
catch (IOException IOE) {
System.out.println(WARNING);
}
}
public synchronized Class<?> loadClass(String name) throws ClassNotFoundException {
Class<?> result = cache.get(name);
if (result == null)
result = cache.get(packageName + "." + name);
if (result == null)
result = super.findSystemClass(name);
System.out.println("== loadClass(" + name + ")");
return result;
}
private String stripClassName(String className) {
return className.substring(0, className.length() - 6);
}
private String normalize(String className) {
return className.replace('/', '.');
}
private boolean match(String className, String packageName) {
return className.startsWith(packageName) && className.endsWith(".class");
}
private byte[] loadClassData(JarFile jarFile, JarEntry jarEntry) throws IOException {
long size = jarEntry.getSize();
if (size == -1 || size == 0)
return null;
byte[] data = new byte[(int)size];
InputStream in = jarFile.getInputStream(jarEntry);
in.read(data);
return data;
}
}
接口(interface)和实现(只是模板,没有任何具体逻辑):
package com.example.classloading;
public interface Module {
public void demo(String str);
}
package com.example.classloading;
public class Weather implements Module {
public void demo(String str) {
System.out.println("hello from weather module");
}
}
应用程序类别:
import com.example.classloading.JarClassLoader;
import com.example.classloading.Module;
public class App {
public static void main(String[] args) {
JarClassLoader jarClassLoader = new JarClassLoader("classloading/weather-module/target/weather-module-1.0-SNAPSHOT.jar", "com.example.classloading");
try {
Class<?> clas = jarClassLoader.loadClass("com.example.classloading.Weather");
Module sample = (Module) clas.newInstance();
sample.demo("1");
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
问题:当我运行 main 方法时,我得到以下输出:
== loadClass(java.lang.Object)
== class com.example.classloading.Module loaded in cache
== class com.example.classloading.Weather loaded in cache
== loadClass(com.example.classloading.Weather)
Exception in thread "main" java.lang.ClassCastException: com.example.classloading.Weather cannot be cast to com.example.classloading.Module
at App.main(App.java:12)
逻辑或语法有问题吗? 模块
没有被应用程序类加载器加载?
文件树(稍微简化):
├───classloading
│ │ pom.xml
│ │
│ ├───menu-module
│ │ │ pom.xml
│ │ │
│ │ ├───src
│ │ │ ├───main
│ │ │ │ ├───java
│ │ │ │ │ │ App.java
│ │ │ │ │ │
│ │ │ │ │ └───com
│ │ │ │ │ └───example
│ │ │ │ │ └───classloading
│ │ │ │ │ JarClassLoader.java
│ │ │ │ │ Module.java
│ │ │ │ │
│ │ │ │ └───resources
│ │ │ └───test
│ │ │ └───java
│ │ └───target
│ │ ├───classes
│ │ │ │ App.class
│ │ │ │
│ │ │ └───com
│ │ │ └───example
│ │ │ └───classloading
│ │ │ JarClassLoader.class
│ │ │ Module.class
│ │ │
│ │ └───generated-sources
│ │ └───annotations
│ └───weather-module
│ │ pom.xml
│ │
│ ├───src
│ │ ├───main
│ │ │ ├───java
│ │ │ │ └───com
│ │ │ │ └───example
│ │ │ │ └───classloading
│ │ │ │ Module.java
│ │ │ │ Weather.java
│ │ │ │
│ │ │ └───resources
│ │ └───test
│ │ └───java
│ └───target
│ │ weather-module-1.0-SNAPSHOT.jar
│ │
│ ├───classes
│ │ │ Module.class
│ │ │ Weather.class
│ │ │
│ │ └───com
│ │ └───example
│ │ └───classloading
│ │ Module.class
│ │ Weather.class
│ │
│ ├───maven-archiver
│ │ pom.properties
│ │
│ └───maven-status
│ └───maven-compiler-plugin
│ ├───compile
│ │ └───default-compile
│ │ createdFiles.lst
│ │ inputFiles.lst
│ │
│ └───testCompile
│ └───default-testCompile
│ inputFiles.lst
│
└───
更新:我在 JarClassLoader
cacheClasses()
if (match(normalize(jarEntry.getName()), packageName))
至
if (match(normalize(jarEntry.getName()), packageName)
&& !normalize(jarEntry.getName()).contains("Module"))
这是解决方法。如何以正确的方式做到这一点?
更新:据我了解,可以从模块 Weather
中删除 Module
接口(interface),然后“将“菜单”模块声明为依赖项用于天气模块”@Costi Ciudatu .
现在我有以下 pom.xml
文件:
菜单模块
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>classloading</artifactId>
<groupId>java-tasks</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>menu</artifactId>
<dependencies>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
<version>2.8.1</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
<version>2.8.1</version>
</dependency>
</dependencies>
</project>
天气模块
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>java-tasks</groupId>
<artifactId>weather-module</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>java-tasks</groupId>
<artifactId>menu</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
</dependencies>
</project>
类加载
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>java-tasks</groupId>
<artifactId>classloading</artifactId>
<packaging>pom</packaging>
<version>1.0-SNAPSHOT</version>
<modules>
<module>weather-module</module>
<module>menu-module</module>
</modules>
</project>
问题:我尝试打包 weather-module
并收到错误:
[INFO] Scanning for projects...
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] Building weather-module 1.0-SNAPSHOT
[INFO] ------------------------------------------------------------------------
[WARNING] The POM for java-tasks:menu:jar:1.0-SNAPSHOT is missing, no dependency information available
[INFO] ------------------------------------------------------------------------
[INFO] BUILD FAILURE
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 0.471 s
[INFO] Finished at: 2017-04-07T09:15:38+03:00
[INFO] Final Memory: 8M/245M
[INFO] ------------------------------------------------------------------------
[ERROR] Failed to execute goal on project weather-module: Could not resolve dependencies for project java-tasks:weather-module:jar:1.0-SNAPSHOT: Could not find artifact java-tasks:menu:jar:1.0-SNAPSHOT -> [Help 1]
[ERROR]
[ERROR] To see the full stack trace of the errors, re-run Maven with the -e switch.
[ERROR] Re-run Maven using the -X switch to enable full debug logging.
[ERROR]
[ERROR] For more information about the errors and possible solutions, please read the following articles:
[ERROR] [Help 1] http://cwiki.apache.org/confluence/display/MAVEN/DependencyResolutionException
我应该如何配置 maven pom.xml
文件才能正确工作?
最佳答案
您的天气模块不应包含 Module
类的副本。否则,您最终会得到该类的两个副本,这是 ClassCastException
的根本原因。
使天气模块依赖于菜单模块或将 Module
类提取到一个单独的类中。最重要的是,您应该确保您的类路径中最终只有一个版本的 Module
。
关于java - 从 jar 加载自定义类时出现 ClassCastException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43265983/
我正在编写一个 mapreduce 应用程序,它接受(键,值)格式的输入并只显示与 reducer 输出相同的数据。 这是示例输入: 1500s 1 1960s 1 Aldus 1 在下面
我不明白,我有一个典型的消息源 content.Language 我可以得到它就像 @Autowire protec
我已经为抽屉导航编写了一个运行良好的程序,但最近我尝试为个人资料图片和 TextView 放置一个 ImageView,之后它给了我一个 ClassCastException。 main_activi
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
@Override public void onPause() { super.onPause(); save(notes.itemSelected); } @Override pub
描述 我正在尝试创建一种自定义语言,我想将词法分析器规则与解析器规则分开。此外,我的目标是将词法分析器和解析器规则进一步划分为特定文件(例如,通用词法分析器规则和关键字规则)。 但我似乎无法让它发挥作
我正在尝试使用以下代码为给定的 Runnable 对象创建代理: public class WorkInvocationHandler implements InvocationHandler {
我有两个非常简单的类,一个扩展了另一个: public class LocationType implements Parcelable { protected int locid =
我遇到了 ClassCastException :Cannot cast my.package.classA to my.package.classA.请注意,(规范)类(名称)是相同的。 我知道这应
我有一个代码试图将一个函数包装在另一个执行动态类型检查的函数中: class Base class Foo extends Base class Bar extends Base object Mai
我使用hadoop 0.18.3遇到以下错误 java.lang.ClassCastException:org.apache.hadoop.io.Text无法转换为org.apache.hadoop.
在 org.dozer.BeanFactory.createBean(Object, Class, String) 的实现中我尝试将对象转换为它的类型。如果我部署所有 bundle ,即关闭并启动所有
我有这个代码: package Maven_test.Maven_project_test; public class Test { class A { int i = 10;
我一直在尝试对 Wicket 的 WebSession 进行子类化,以便可以实现基本的身份验证系统。我已遵循 Wicket 引用库中的指南。当我在网页中尝试以下操作时,出现 ClassCastExce
我正在构建一个 kotlin AAR 库,我需要在发布之前对其进行混淆。我有以下结构: package com.example.token interface TokenManager { s
Kotlin 引入了 here 中描述的声明站点差异. 在某些情况下,泛型参数的 out/in 关键字可能会导致 ClassCastException。我的程序如下所示。 fun main(args:
我正在 AnyLogic 中进行基于代理的传染病流行模拟。我的模型中有两种代理类型 - 人员和建筑物。我正在尝试编写一个函数来计算代理类型“人员”在任何给定时间点所具有的传染性接触数量。下面是我的函数
我有一个 EditContact 类。打开后,它会显示哪些复选框已选中或未选中。这是通过我的适配器中的一些代码完成的,它可以正常工作: //This is for EditContact, t
这个问题已经有答案了: 奥 git _a (2 个回答) 已关闭 5 年前。 我正在获取 ClassCastException 。这两个类来自不同的 jar,但是JettyContinuationPr
我想在 Java 中使用一组对,但是当我调用 contains() 来查看它是否已包含特定对时,我总是会得到 ClassCastException >。有没有办法避免这种行为? 它的实例化如下: pr
我是一名优秀的程序员,十分优秀!