-1

我有一个 AbstractBaseRepository。我所有的存储库都从这个类扩展。我创建了另一个类 RepositoryFactory 来创建 Repository 的任何实例。由于静态方法的早期绑定,我遇到了问题。

public abstract class AbstractBaseRepository {
    public static <T extends AbstractBaseRepository> T getNewInstance(EntityManagerFactory entityManagerFactory) {
        throw new RuntimeException("Override and provide valid initialization");
    }
    ...
}

public class RepositoryFactory {
    public static <T extends AbstractBaseRepository>  T getRepository(Class<T> cls) {       
        return T.getNewInstance(entityManagerFactory);
    }
    ...
}

一个示例子类

public class DeviceModelRepo extends AbstractBaseRepository {

    public static DeviceModelRepo getNewInstance(EntityManagerFactory entityManagerFactory) {
        return new DeviceModelRepo(entityManagerFactory);
    }
    ...
}

每当我使用 AbstractBaseRepository 的有效子类调用 getRepository() 时,都会引发运行时异常。这是由于静态方法的早期绑定。在编译期间,getNewInstance 与 AbstractBaseRepository 绑定,而不是在运行时与类的实际类型绑定。有什么好的解决方法吗?

4

2 回答 2

0

我的第一个建议是使用 Spring。很容易获得使用某个接口创建的所有 bean 的列表。

此外,如果您将 Repository 实例视为一种“插件”,您可能会看到 Java 的 ServiceLoader 类如何提供帮助。

此外,另一种方法是在工厂中使用 switch 语句并为每个案例创建实例,而不是在 Repository 子类上使用静态方法。

最后,我不推荐反射解决方案,但有一些方法可以根据其名称加载类并反射地创建一个新实例。

但是覆盖静态方法是不可能的。

于 2019-10-29T04:39:13.073 回答
0

通过查看您的代码,我了解到您希望拥有 AbstractBaseRepository 的不同实现,例如 DeviceModelRepo。然后你想要一个工厂类来创建 AbstractBaseRepository 的具体实现的实例。这里的主要问题是您尝试覆盖永远不会被覆盖但子类将隐藏父实现的静态方法。请不要使用静态方法进行覆盖。您可以如下所示更改您的实施,此问题将得到解决。

public abstract class AbstractBaseRepository {
    public AbstractBaseRepository(EntityManagerFactory entityManagerFactory){
        ...
    }
    //removed method getNewInstance(EntityManagerFactory entityManagerFactory) 
    ...
}

然后在子类的下面实现。

public class DeviceModelRepo extends AbstractBaseRepository {

    public DeviceModelRepo(EntityManagerFactory entityManagerFactory) {
        super(entityManagerFactory);
        ...
    }
    //removed method getNewInstance(EntityManagerFactory entityManagerFactory) 
    ...
}

现在我为您提供两个工厂类的实现。一种是对每个实现都有不同的方法,例如 getDeviceModelRepository()。另一种解决方案是使用反射并通过传递实现存储库类来获取存储库实例。

public class RepositoryFactory {
    //Solution-1, create separate method for each of repository like below
    public static AbstractBaseRepository getDeviceModelRepository() {       
        return new DeviceModelRepo(entityManagerFactory);
    }
    //Solution-2, use reflection to get instance of specific implementation
    //of AbstractBaseRepository
    public static <T extends AbstractBaseRepository> T 
        getRepository(Class<T> repoClass) throws Exception{

        return repoClass.getConstructor(EntityManagerFactory.class)
            .newInstance(entityManagerFactory);
    }
    ...
}

使用反射解决方案,您可以获得存储库实例,如下所示。

RepositoryFactory.getRepository(DeviceModelRepo.class)
于 2019-10-29T05:47:53.497 回答