0

静的ブロックでハッシュマップを初期化したクラスがあります。キーを渡して、クラスである値を取得しました。このクラスのオブジェクトを作成するため。コンストラクター クラスを使用してコンストラクターを取得し、引数を渡してオブジェクトを作成しました。

ハッシュマップに2つのクラスがあります。EchoExpression のオブジェクトを作成するには、2 つの引数を渡す必要があり、OutExpression クラスには 1 つの引数 (文字列) のみを渡す必要があります。

質問:

キーによって返されたクラスに基づいて、取得して実装するコンストラクターを実行する必要があります。これは、引数が 1 つまたは 2 つのコンストラクターのどちらでもかまいません。

public class ExampleFactory {

  private static HashMap<String,Class<?>> hmap = new HashMap<String,Class<?>>();

  static
  {                   
      hmap.put("echo", EchoExpression.class);         
      hmap.put("Out", OutExpression.class);                       
  }

  public void getExpo(String key,String expression)
  {
    Class aClass =map.get(key);

    //Constructor implementation for OutExpression where only one argument string is passed

    Constructor constructor = aClass.getConstructor(new Class[]{String.class});

    Object object= constructor.newInstance(expression);

    //constructor for passing two arguments string for EchoExpression

    Constructor constructor = aClass.getConstructor(new Class[]{String.class,Class.class});

    Object object= constructor.newInstance(expression, Boolean.class);


    return null;        
  }                
}

if elseを使用せずに実装するクラスを値(クラス)から選択する方法は?

4

2 に答える 2

1

Enum を使用してオンにします。これは、リフレクションや例の構文に深く入り込むことなく、実行可能なスタブです。

package com.trip.test;

import java.util.HashMap;
import java.util.Map;

public class ExampleFactory {

    private static Map<String, Class<?>> hmap = new HashMap<String, Class<?>>();

    static {
        hmap.put("echo", EchoExpression.class);
        hmap.put("Out", OutExpression.class);
    }

    public static void getExpo(String key, String expression) {

        Class aClass = hmap.get(key);

        ClassMappingEnum myType = ClassMappingEnum.getClassMappingEnum(aClass);

        switch (myType) {
        case ECHO_EXPRESSION:{
            System.out.println(aClass.getName());
            // do something
            break;
        }
        case OUT_EXPRESSION:{
            System.out.println(aClass.getName());
            // do something
            break;          
        }
        case UNKNOWN:
        default:
            System.out.println("Bummer: " + aClass.getName());          
        }

    }

    public static void main(String[] args) {
        getExpo("echo", "B");
        getExpo("Out", "B");
    }   
}

enum ClassMappingEnum {
    ECHO_EXPRESSION(EchoExpression.class), OUT_EXPRESSION(OutExpression.class), UNKNOWN(null);

    private Class typeDes;

    private ClassMappingEnum(Class typeDes) {
        this.typeDes = typeDes;
    }

    public static ClassMappingEnum getClassMappingEnum(Class compare) {
        for (ClassMappingEnum cme : ClassMappingEnum.values()) {
            if (cme.typeDes.equals(compare)) {
                return cme;
            }
        }
        return UNKNOWN;
    }


}

class EchoExpression<T> {
    private String someString;
    private Class<T> someClass;
    public EchoExpression(String someString, Class<T> someClass) {
        super();
        this.someString = someString;
        this.someClass = someClass;
    }
    public String getSomeString() {
        return someString;
    }
    public void setSomeString(String someString) {
        this.someString = someString;
    }
    public Class<T> getSomeClass() {
        return someClass;
    }
    public void setSomeClass(Class<T> someClass) {
        this.someClass = someClass;
    }


}

class OutExpression {
    private String someString;

    public OutExpression(String someString) {
        super();
        this.someString = someString;
    }

    public String getSomeString() {
        return someString;
    }

    public void setSomeString(String someString) {
        this.someString = someString;
    }

}
于 2012-04-05T00:02:40.420 に答える
0

両方のコンストラクターが同じ署名を持つようにクラスを変更できる場合(同じ数/タイプの引数を同じ順序で受け入れる)、次のことができます。

Constructor constructor = aClass.getConstructor(new Class[]{String.class,Class.class});
Object object= constructor.newInstance(expression, Boolean.class);

両方のクラスで。

これはもちろん、現在追加のパラメータを必要としないクラスは、変更前に使用していなかった渡されたものを無視する必要があることを意味します

更新:Factoryクラスを使用してアイデアを実装するための可能な方法は次のとおりです。

public interface ObjectFactory
{
  Object create(String expr, Class cls);
}

public class EchoExpressionFactory implements ObjectFactory
{
  public EchoExpression create(String expr, Class cls)
  {
    return new EchoExpression(expr, cls);
  }
}

public class OutExpressionFactory implements ObjectFactory
{
  public OutExpression create(String expr, Class cls)
  {
    return new OutExpression(expr);
  }
}

public class ExampleFactory { 

  private static HashMap<String,ObjectFactory> hmap = new HashMap<String,ObjectFactory>(); 

  static 
  {                    
      hmap.put("echo", new EchoExpressionFactory());          
      hmap.put("Out", new OutExpressionFactory());                        
  } 

  public void getExpo(String key,String expression) 
  { 
    ObjectFactory factory = map.get(key); 

    //Constructor implementation for Expression
    Object object = factory.create(expression); 

    Object object= constructor.newInstance(expression, Boolean.class); 

    return;
  }                 
}
于 2012-04-04T22:47:56.190 に答える