从通用管理器类中,如何以静态方式访问有关其实例的信息?



我正在尝试创建一个在线竞技场游戏,玩家可以在其中创建自定义竞技场进行游戏。我希望将竞技场设置保存在MySQL数据库中,以便保存竞技场以供进一步玩。

问题是,阿里纳斯有不同的变体。因此,可能有丛林竞技场、沙漠竞技场等。所有这些不同的变体(抽象类Arena的子类)都必须存储在不同的表中,因为每种类型的竞技场的变量(以及列)都不同。

我的问题是:从下面描述的包含泛型类型数组的Manager类中,我如何从子(T)类及其父类中获得所需的列。我使用了一个泛型类型,因为我还想稍后使用这个类来管理玩家的信息(这些信息也需要存储在数据库中)。

这是经理类:

public class Manager<T extends Savable> {
private Connection connection;
private List<Column<?>> columns;
private List<T> instances;
public Manager() {
fillColumnArray()
try {
connection = getSQLConnection();
Statement s = connection.createStatement();
s.executeUpdate(getCreateQuery())
s.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
public void fillColumnArray() {
// Somehow, manages to get the needed columns from
// the parent class and the child class
}
public Connection getSQLConnection() {
// Creates a mysql connection to a database
// from parameters on a local file and returns it
}
public String getCreateQuery() {
// Creates the query from the previously found columns
}
/* */
}

以下是Manager中使用的Column类:

public class Column<U> {
private String name;
private U defaultValue;
/* */
}

以下是Arena类的可能变量和一个可能的儿童

public abstract class Arena implements Savable {
private Vector centerPosition;
private int radius;
private int gameLenght;
/* */
}
public class JungleArena extends Arena {
private int treeHeight;
private boolean daytime;
/* */
}

我想到了许多不同的可能性。其中之一是在Savable接口中添加一个requestColumns()方法,但这个选项不起作用,因为在创建Manager时,我只能以静态方式访问t类。

因此,我想在子类和父类中添加一个静态方法requestColumns(),但由于我使用的是泛型类型,编译器无法理解该方法的来源。

然后我想把T类型转换成特殊的Arena类型,但在我的项目中,当使用相同的类来存储玩家时,它就无法进一步工作了。

希望这一点足够清楚。

有人有让事情运转起来的想法吗?或者只是你理解的另一种做事方式吗?

感谢

考虑到您已经有了实例,并且可以使用该实例以通用的方式为您获取这些信息,我不确定为什么需要静态地执行此操作。我也不清楚你打算如何从类中获得列的值,如果你还没有,这个答案会处理这个问题。

就我个人而言,如果没有任何有价值的框架能够做到这一点,我会定义一个注释,放在列字段的公共getter上。像这样:

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Column {
String name() default "";
}

然后,您可以使用反射来获取这些getter,然后调用实例上的方法,从而获得值。例如,给定此类:

class Entity {
...
@Column(name = "Name")
public getName() { return name; }
@Column(name = "Height")
public getHeight() { return height; }
}

我可以得到这样的列名和值:

Entity e = new Entity("First", 42);
Method[] methods = e.getClass().getMethods();
Map<String, Method> colums = new HashMap<>();
for (Method m : methods) {
Column c = m.getAnnotation(Column.class);
if (c != null) {
colums.put(c.name(), m);
}
}
colums.entrySet().forEach(entry -> {
try {
System.out.println(entry.getKey() + ": " + entry.getValue().invoke(e));
} catch (IllegalAccessException | InvocationTargetException e1) {
e1.printStackTrace();
}
});

Output >
Height: 42
Name: First

最新更新