我正在尝试在一个应用程序中使用Stanford NER和Stanford POS tagger。当我尝试运行POS标记方法时,我得到IncompatibleClassChangeError
。
我在类路径中有NER和POS标记器的jar文件。如果我从类路径中删除了一罐NER,那么这个错误就不会到来。我想在NER jar和POS jar中有一些常见的类,java无法确定在运行时使用哪个类。
以下是stacktrace:
java.lang.IncompatibleClassChangeError: Implementing class
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(Unknown Source)
at java.security.SecureClassLoader.defineClass(Unknown Source)
at java.net.URLClassLoader.defineClass(Unknown Source)
at java.net.URLClassLoader.access$000(Unknown Source)
at java.net.URLClassLoader$1.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClassInternal(Unknown Source)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.init(MaxentTagger.java:407)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.readModelAndInit(MaxentTagger.java:699)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.readModelAndInit(MaxentTagger.java:673)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.<init>(MaxentTagger.java:280)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.<init>(MaxentTagger.java:260)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.runTagger(MaxentTagger.java:1305)
at edu.stanford.nlp.tagger.maxent.MaxentTagger.main(MaxentTagger.java:1499)
at com.tcs.srl.stanford.POSWrapper.executePOSTagger(POSWrapper.java:39)
at com.tcs.srl.stanford.test.POSWrapperTester.ExecutePOSTagger(POSWrapperTester.java:19)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:76)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:49)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
我不知道为什么会出现这个错误。
答案 0 :(得分:1)
我认为你的诊断很接近但不太正确。 Java将始终在类路径中使用类的第一个实例。但是,如果您有类加载器的层次结构,并且子类加载器尝试定义父类已定义的类,则会出现这样的错误。
我不熟悉斯坦福NER或POS代码。但是,我可以提供一些建议。
如果你说服两个罐子中具有相同名称的类是相同的,那么只需将两个罐子组合成一个罐子 - 这将删除任何重复的。
如果您担心NER和POS代码可能不兼容,我会将两者的源代码移动到一个项目中,然后重新编译,仔细检查重复项。
另一种方法是从一个项目获取源代码并将其移动到另一个包中,以便类名保持不变,但包不再发生冲突。一个好的IDE应该可以毫不费力地做到这一点。
但是,对于像这样的问题,邮件列表通常是获得建议的最佳位置:http://www-nlp.stanford.edu/software/CRF-NER.shtml。我相信你不会是唯一一个一起使用这些软件的人。我上面提到的解决方案都不一定是必要的,所以从使用它们的其他人那里获得建议可能是最好的。
答案 1 :(得分:1)
似乎众所周知的问题和斯坦福大学的人都知道这个问题。我已经在NER解析器的邮件列表上发送了邮件,我收到了来自John的人的回复
这是一个已知的问题 固定在一个星期左右。在里面 同时,所有必要的课程 这两个图书馆都在 StanfordCoreNLP,
约翰
似乎他们正在尝试从他们的API中删除此错误,并且在解析器和标记器的下一个版本中,我们将获得干净的API。
目前我正在使用CoreNLP软件包here。这个CoreNLP软件包包含解析器,标记器,ner和其他一些东西。