1

このようなメソッドがあるとします

 public void readAndInitialize(StringReader reader, Class className)
  {
    // Here, I want to to something like this - 

   // obj = new (object of the specified className)

      obj.readAndInitialize(reader);
  }

Javaでこれを行う方法は?

4

6 に答える 6

4

誰もが指摘するのは簡単Class.forName("com.myorg.MyClass");で、関連するnewInstance()メソッドですが、パラメーターをとらないデフォルトのコンストラクターのみを呼び出すことを覚えておくことが重要です。

クラスの特定のコンストラクターを呼び出す必要がある場合は、リフレクションを使用して正しいコンストラクターを見つけてから呼び出す必要があります。

import java.lang.reflect.Constructor;
import java.lang.reflect.Type;
import static java.lang.System.out;

public class ConstructorSift {
    public static void main(String... args) {
    try {
        Class<?> cArg = Class.forName(args[1]);

        Class<?> c = Class.forName(args[0]);
        Constructor[] allConstructors = c.getDeclaredConstructors();
        for (Constructor ctor : allConstructors) {
        Class<?>[] pType  = ctor.getParameterTypes();
        for (int i = 0; i < pType.length; i++) {
            if (pType[i].equals(cArg)) {
            out.format("%s%n", ctor.toGenericString());

            Type[] gpType = ctor.getGenericParameterTypes();
            for (int j = 0; j < gpType.length; j++) {
                char ch = (pType[j].equals(cArg) ? '*' : ' ');
                out.format("%7c%s[%d]: %s%n", ch,
                       "GenericParameterType", j, gpType[j]);
            }
            break;
            }
        }
        }

        // production code should handle this exception more gracefully
    } catch (ClassNotFoundException x) {
        x.printStackTrace();
    }
    }
}

にはすべてのコンストラクターがリストされています。チュートリアルはこちらから入手できます

目的のコンストラクターが見つかったら、次のように呼び出すことができます

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import static java.lang.System.out;

class EmailAliases {
    private Set<String> aliases;
    private EmailAliases(HashMap<String, String> h) {
    aliases = h.keySet();
    }

    public void printKeys() {
    out.format("Mail keys:%n");
    for (String k : aliases)
        out.format("  %s%n", k);
    }
}

public class RestoreAliases {

    private static Map<String, String> defaultAliases = new HashMap<String, String>();
    static {
    defaultAliases.put("Duke", "duke@i-love-java");
    defaultAliases.put("Fang", "fang@evil-jealous-twin");
    }

    public static void main(String... args) {
    try {
        Constructor ctor = EmailAliases.class.getDeclaredConstructor(HashMap.class);
        ctor.setAccessible(true);
        EmailAliases email = (EmailAliases)ctor.newInstance(defaultAliases);
        email.printKeys();

        // production code should handle these exceptions more gracefully
    } catch (InstantiationException x) {
        x.printStackTrace();
    } catch (IllegalAccessException x) {
        x.printStackTrace();
    } catch (InvocationTargetException x) {
        x.printStackTrace();
    } catch (NoSuchMethodException x) {
        x.printStackTrace();
    }
    }
}

これも、リフレクションに関するチュートリアルからのものです。

于 2012-06-07T20:09:54.860 に答える
2

あなたの例では、classNameをClassオブジェクトとして渡していることに注意してください。

className.newInstance();

しかし、クラス名を として渡す方法を探しているのではないかと思います。Stringその場合の手順は (ここでclassNameは a ですString):

Class class = Class.forName(className());
Object obj = class.newInstance();

Exceptions両方の呼び出しによってスローされたも処理する必要があることに注意してください。

これは最も単純なケースであり、特定のコンストラクターなどは対象外です。

クラスのドキュメントを参照してください。

于 2012-06-07T20:06:16.260 に答える
1
public void readAndInitialize(StringReader reader, String className)  //className changed to String
  {
    // Here, I want to to something like this - 
   // obj = new (object of the specified className)

      //try this:
      Class myclass = Class.forName(className);
      Object obj = myclass.newInstance();

      obj.readAndInitialize(reader);
  }
于 2012-06-07T20:06:18.303 に答える
1

私が質問を正しく理解していれば。特定のクラスのオブジェクトは、次のように作成する必要があります。

if (yourClass != null) {
    Object yourObject = null;
    try {
        yourObject = yourClass.newInstance();
    } catch (InstantiationException e) {
        LOGGER.error("", e);            
    } catch (IllegalAccessException e) {
        LOGGER.error("", e);

    }
}
于 2012-06-07T20:08:12.843 に答える
0

オブジェクトを既に取得しているか、クラス名を持つ文字列のみを取得しているかに応じて、次のことを行います。

  1. クラス名の文字列を使用して、リフレクションを使用して、上記のようにオブジェクトをインスタンス化します。すでにオブジェクトを取得している場合は、この手順をスキップできます
  2. オブジェクトが、instanceof を介してパブリック メソッド readAndAnalyze(...) を提供するクラスまたはインターフェイスから継承されていることを確認します。
  3. instanceof を介して、1 ステップ上でチェックしたクラスにオブジェクトをキャストします。
  4. メソッドを呼び出して続行する
于 2012-06-07T20:37:09.143 に答える
0

Class.newInstance() (ゼロ引数コンストラクターのみ) または Constructor.newInstance() を使用して、必要なことを行うことができます。実際の例については、こちらを参照してください。

于 2012-06-07T20:09:36.487 に答える