如何从一组类创建一个类的实例?

时间:2013-06-19 22:55:59

标签: java class processing

我有一个类似的类列表,它们都是同一个抽象类的子类。我还有一个布尔数组,应该对应应该使用哪个类。

例如,我有一堆类,使用以下约定命名:

boolean[] classesOn = new boolean[4];
abstract class myClass {}

class myClass1 extends myClass { public void myClass1(float x, float y) ...}
class myClass2 extends myClass {}
class myClass3 extends myClass {}
class myClass4 extends myClass {}
...

我们的想法是只在classesOn中使用具有相应布尔值的类。为此,我使用for循环遍历classesOn并检查哪些是true。现在我有以下代码:

for (int i = 0; i < classesOn.length; i++) {
  if (classesOn[i]) {
    switch (i) {
      case 0: c = new myClass1(x, y); break;
      case 1: c = new myClass2(x, y); break;
...

现在这非常低效,当我添加myClass的新扩展时,我需要添加一个新案例。我希望能够只说if (classesOn[i]) { c = new "myClass" + (i + 1) ();}来创建该特定类的实例。

我该怎么做?

(顺便说一句,这些只是示例,每个类的实际实现差别很大)

我目前正在使用的用途实际上是在Processing中,其中有多个颜色方案,每个颜色方案都在一个类中表示。但我很好奇将来如何为所有类型的课程做这件事。

我现在正在处理的确切代码看起来像这样 - (但我对一般的答案感兴趣)

abstract class Scheme {
  float red,blue,green,x,y;
  String description;
  public void mousespot(){
    this.x = mouseX;
    this.y = mouseY;
    return;
  }
  public float getRed(){
    return this.red;
  }
  public float getBlue(){
    return this.blue;
  }
  public float getGreen(){
    return this.green;
  }
  public String getDescription(){
    fill(255,255,255);
    textSize(32);
    return this.description;
  }
}
class Scheme1 extends Scheme {
  public Scheme1(float x, float y) {
    this.description = "Green-Yellow-GW-Turqouise";
    this.red = map(x, 0, width, 0, 255);
    this.blue = map(y, 0, height, 0, 255);
    this.green = 255 * (float) dist(width/2, height/2, x, y) / (x / y);
  }
}

class Scheme2 extends Scheme {
  public Scheme2(float x, float y) {
    this.description = "Red-Yellow-Peach-Magenta";
    this.green = map(x, 0, width, 0, 255);
    this.blue = map(y, 0, height, 0, 255);
    this.red = 255 * (float) dist(width/2, height/2, x, y) / (x / y);
  }
}

并在mouseDragged()方法中:

for (i = 0; i < colorschemesOn.length;i++) {
      if (colorschemesOn[i]) {
        switch(i) {
          case 0: 
            public Scheme selectedScheme = new Scheme1(mouseX,mouseY);
            break;
          case 1: 
            public Scheme selectedScheme = new Scheme2(mouseX,mouseY);
            break;
          case 2:
            public Scheme selectedScheme = new Scheme3(mouseX,mouseY);
            break;
          case 3:
            public Scheme selectedScheme = new Scheme4(mouseX,mouseY);
            break;
          case 4:
            public Scheme selectedScheme = new Scheme5(mouseX,mouseY);
            break;
          case 5:
            public Scheme selectedScheme = new Scheme6(mouseX,mouseY);
            break;
          case 6:
            public Scheme selectedScheme = new Scheme7(mouseX,mouseY);
            break;
          case 7:
            public Scheme selectedScheme = new Scheme8(mouseX,mouseY);
            break;
          case 8:
            public Scheme selectedScheme = new Scheme9(mouseX,mouseY);
            break;
          case 9:
            public Scheme selectedScheme = new Scheme10(mouseX,mouseY);
            break;
          case 10:
            public Scheme selectedScheme = new Scheme11(mouseX,mouseY);
            break;
          case 11:
            public Scheme selectedScheme = new Scheme12(mouseX,mouseY);
            break;
          default:
            public Scheme selectedScheme = new Scheme1(mouseX,mouseY);
            break;
        }
    }
}

2 个答案:

答案 0 :(得分:4)

不要依赖命名约定,而是创建一个数组:

Class<? extends myClass>[] classes = new Class<? extends myClass>[] { 
    myClass1.class, myClass2.class, myClass3.class, myClass4.class
};
boolean[] classesOn = new boolean[classes.length];

然后你可以用反射来实例化它们:

if (classesOn[i])  { myClass c = classes[i].getConstructor().newInstance(); }

如果构造函数接受参数:

myClass c;
if (i < classesOn.lenght && classesOn[i])  { 
    c = classes[i]
        .getConstructor(float.class, float.class)
        .newInstance(mouseX, mouseY); 
} else {
    c = new myClass1(mouseX, mouseY);
}

答案 1 :(得分:2)

您可以使用反射

 if (classesOn[i]) { c = Class.forName("myClass" + (i + 1)).newInstance(); }

这很少是一个好主意,并且通常有更好的方法来构建您的类/代码,这意味着您不需要它。

更优雅的方式可能是使用Reflections库。 (不要与内置的混淆)

它支持通过注释或接口等方式查询类。您可以通过它来查找myClass的所有子类,无论它们被称为什么或它们所在的JAR。

如果构造函数接受两个参数,则可以将两个参数传递给它。

 if (classesOn[i]) { 
     c = Class.forName("myClass" + (i + 1))
              .getConstructor(float.class, float.class)
              .newInstance(mouseX, mouseY); 
 }