java - 枚举抽象问题

标签 java enums abstraction

我目前正在努力解决 Java 抽象问题。我有这样的东西:

public interface State {
};

public interface Dynamics {
  getObservationChance(State state, Observation observation);
};

class SpecialState implements State {
};

enum SpecialObservation() {
  FREE, WALL, etc.
}

class SpecialDynamics implements Dynamics {
   getObservationChance(State state, Observation observation) {
       // state should be SpecialState, observation should be SpecialObservation!
   }
};

class Main {
  Main(State state, Observation observation, Dynamics dynamics) {
      dynamics.getObservationChance(state, observation);
   }
};

SecialObservation 应该是可能的观察结果的枚举(或类似的东西),但我想要问题的抽象表示。所以我想要一个应该包含观察的观察 一个返回所有可能观察列表的函数。最后一件事对于我正在实现的算法非常重要,因为我必须总结所有可能的观察结果。

谢谢!

最佳答案

您在这里需要参数化类型 - 您每个都有树类型系列:状态、观察和动态。

如果我们将观察枚举作为参数类型,我们可以将您的类型转换为如下形式:

public interface Observation<O extends Observation<O>> {
    ...
}

public interface State<O extends Observation<O>> {
}

public interface Dynamics<O extends Observation<O>> {
  getObservationChance(State<O> state, O observation);
}

enum SpecialObservation implements Observation<SpecialObservation> {
  FREE, WALL, etc.
}

class SpecialState implements State<SpecialObservation> {
}


class SpecialDynamics implements Dynamics<SpecialObservation> {
   getObservationChance(State<SpecialObservation> state, SpecialObservation observation) {
       // state should be SpecialState, observation should be SpecialObservation!
   }
}

class Main<O extends Observation> {
  Main(State<O> state, O observation, Dynamics<O> dynamics) {
      dynamics.getObservationChance(state, observation);
   }
}

当然,只有当我们的 State 接口(interface)的方法足以满足 getObservationChance 方法时,这种方法才有效。

一种更通用的方法是对所有三种类型进行参数化:

public interface Observation<O extends Observation<O, S, D>,
                             S extends State<O,S,D>,
                             D extends Dynamics<O,S,D>>
{
    ...
}

public interface State<O extends Observation<O,S,D>,
                       S extends State<O,S,D>,
                       D extends Dynamics<O,S,D>> {
}

public interface Dynamics<O extends Observation<O,S,D>,
                          S extends State<O,S,D>,
                          D extends Dynamics<O,S,D>> {
  getObservationChance(S state, O observation);
}

然后我们可以这样定义实现:

enum SpecialObservation implements Observation<SpecialObservation, SpecialState, SpecialDynamics> {
  FREE, WALL, etc.
}

class SpecialState implements State<SpecialObservation, SpecialState, SpecialDynamics> {
}


class SpecialDynamics implements Dynamics<SpecialObservation, SpecialState, SpecialDynamics> {
   getObservationChance(SpecialObservation state, SpecialObservation observation) {
       // state should be SpecialState, observation should be SpecialObservation!
   }
}

当然,主类需要所有三个参数:

class Main<O extends Observation<O,S,D>,
           S extends State<O,S,D>,
           D extends Dynamics<O,S,D>> {
  Main(S state, O observation, D dynamics) {
      dynamics.getObservationChance(state, observation);
   }
}

实际上,在您的情况下,动态仅取决于观察和状态,而不是相反(并且它们彼此不依赖),所以另一种方式是这样的:

public interface Observation {
    ...
}

public interface State {
}

public interface Dynamics<S extends State,
                          O extends Observation> {
  getObservationChance(S state, O observation);
}

enum SpecialObservation implements Observation {
  FREE, WALL, etc.
}

class SpecialState implements State {
}


class SpecialDynamics implements Dynamics<SpecialState, SpecialObservation> {
   getObservationChance(SpecialState state, SpecialObservation observation) {
       // state should be SpecialState, observation should be SpecialObservation!
   }
}

class Main<S extends State, O extends Observation> {
  Main(S state, O observation, Dynamics<S, O> dynamics) {
      dynamics.getObservationChance(state, observation);
   }
}

编辑: 关于 getAllObservations 方法: 只要你能以某种方式使你的类型参数具体化,这里就没有真正的问题。 要访问特定类型的枚举常量列表,您需要访问此类型 - 直接访问 (SpecialObservation.values()),或使用类对象,如下所示:

class Main<S extends State, O extends Observation> {

  public O[] getAllObservations(Class<O> oClass) {
     return oClass.getEnumConstants();
  }

  Main(S state, Dynamics<S, O> dynamics, Class<O> observationClass) {
      O[] observations = getAllObservations(observationClass);
      for(O o : observations) {
         dynamics.getObservationChance(state, observation);
      }
   }
}

(当然,这只有在 O 是一个枚举类时才有效。)

如果您有一个混合列表,它会变得更加复杂,并且类型安全地匹配 Dynamics、Action、Observation 和 State 类也不是很容易。

关于java - 枚举抽象问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5539064/

相关文章:

java - 运行 sitemesh 是否需要 .tld?

java - 接口(interface)定义的关系的术语

java - 我是否需要通过 CDI 中的生产者进行所有对象实例化

java - 如何将字符串值转换为 LatLng() 对象?

java - 使用带有自定义序列化的 Gson 序列化枚举映射

java - 如何查找与特定枚举类型相关的 name() 的所有用法

java - 函数参数 : abstract class or interface?

c# - CA1008 : Enums should have zero value and Flag enums

scala - 对隐式对象或值中的参数类型进行抽象?

java - 定义抽象对象的实现