can't cast to implemented interface

丶灬走出姿态 提交于 2019-12-17 06:14:55

问题


i'm very confused...

I have a class which directly implements an interface:

public class Device implements AutocompleteResult
{...}

Here is proof that I'm looking at the right variables:

Object match = ...;
log.debug(match.getClass()); // Outputs 'Device'
log.debug(match.getClass().getInterfaces()[0]); // Outputs 'AutocompleteResult'

Yet when I try to cast an instance of the class to the interface:

AutocompleteResult result = (AutocompleteResult) match;

I get a ClassCastException!

ClassCastException: Device cannot be cast to AutocompleteResult

Also, isAssignableFrom returns false and i'm not sure why:

log.debug(AutocompleteResult.class.isAssignableFrom(Device.class));

from the doc:

Determines if the class or interface represented by this Class object is either the same as, or is a superclass or superinterface of, the class or interface represented by the specified Class parameter.

Shouldn't I always be able to cast a object to an interface its class implements?

Thanks.


回答1:


This can happen if two different classloaders load a class named AutocompleteResult.

These two classes are then treated as entirely different classes, even if they have the same package and name (and even implementation/fields/methods).

A common cause for this is if you use some kind of plugin system and both your base classes and the plugin classes provide the same class.

To check for this issue print the value returned by Class.getClassLoader() on both offending classes (i.e. the class of the interface implemented by Device and the result of AutocompleteResult.class).




回答2:


AKA when Java apparently doesn't Java.

I hit this problem recently with Play Framework 2.6.3, what helped me was this: https://www.playframework.com/documentation/2.6.x/ThreadPools#Application-class-loader

I leave this info here for the people that might have the same problem.

To make it clearer, what helps is:

Injecting Application on an Eager Singleton and then using its classloader to load the classes I was having issues with.

To make it clearer

public class Module {


 @Override
 public void configure {
   bind(TheClassLoaderLoader.class).asEagerSingleton()

public static class TheClassLoaderLoader {
  @Inject
        public TheClassLoaderLoader( Application application) {

         ClassLoader classloader = application.classloader();

                Class<?> interfaceClass = classloader.loadClass(InterfaceClass.class.getName());
                classloader.loadClass(ImplementsInterfaceClass.class.getName()).asSubclass(interfaceClass);

The example here https://playframework.com/documentation/2.6.x/JavaDependencyInjection#Configurable-bindings

That uses Environment often throws a frustrating ClassNotFoundException

Cheers



来源:https://stackoverflow.com/questions/8035877/cant-cast-to-implemented-interface

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!