web-dev-qa-db-ja.com

Javaのメソッドパラメータとして型を渡す方法

Javaでは、どのように型をパラメーターとして渡す(または変数として宣言する)ことができますか?

型のinstanceではなく、型自体(たとえば、int、Stringなど)を渡したくありません。

C#では、これを行うことができます。

private void foo(Type t)
{
    if (t == typeof(String)) { ... }
    else if (t == typeof(int)) { ... }
}

private void bar()
{
    foo(typeof(String));
}

Javaに型tのinstanceを渡さずに方法はありますか?
または、独自のint定数または列挙型を使用する必要がありますか?
それとももっと良い方法はありますか?

編集:fooの要件は次のとおりです。
タイプtに基づいて、異なる短いxml文字列を生成します。
if/elseのコードは非常に小さく(1行または2行)、いくつかのプライベートクラス変数を使用します。

70
user250781

_Class<T>_ を渡すことができます。

_private void foo(Class<?> cls) {
    if (cls == String.class) { ... }
    else if (cls == int.class) { ... }
}

private void bar() {
    foo(String.class);
}
_

Update:OOP方法は機能要件によって異なります。最善の方法は、foo()と2つの具体的な実装を定義するインターフェースですfoo()を実装してから、手元にある実装でfoo()を呼び出すだけです。別の方法は、actions.get(cls)で呼び出すことができる_Map<Class<?>, Action>_です。これは、インターフェイスと具象実装と簡単に組み合わせることができます:actions.get(cls).foo().

96
BalusC

同様の質問があったので、以下の完全な実行可能な回答を作成しました。私がする必要があるのは、クラス(C)を無関係なクラスのオブジェクト(O)に渡し、そのオブジェクト(O)がクラス(C)の新しいオブジェクトを要求したときに私に返すようにすることです。

以下の例は、これがどのように行われるかを示しています。 Projectileクラスのサブタイプ(Pebble、Bullet、NuclearMissle)とともにロードするMagicGunクラスがあります。興味深いのは、Projectileのサブタイプをロードしますが、そのタイプの実際のオブジェクトではありません。 MagicGunは、撮影するときに実際のオブジェクトを作成します。

出力

You've annoyed the target!
You've holed the target!
You've obliterated the target!
click
click

コード

import Java.util.ArrayList;
import Java.util.List;

public class PassAClass {
    public static void main(String[] args) {
        MagicGun gun = new MagicGun();
        gun.loadWith(Pebble.class);
        gun.loadWith(Bullet.class);
        gun.loadWith(NuclearMissle.class);
        //gun.loadWith(Object.class);   // Won't compile -- Object is not a Projectile
        for(int i=0; i<5; i++){
            try {
                String effect = gun.shoot().effectOnTarget();
                System.out.printf("You've %s the target!\n", effect);
            } catch (GunIsEmptyException e) {
                System.err.printf("click\n");
            }
        }
    }
}

class MagicGun {
    /**
     * projectiles holds a list of classes that extend Projectile. Because of erasure, it
     * can't hold be a List<? extends Projectile> so we need the SuppressWarning. However
     * the only way to add to it is the "loadWith" method which makes it typesafe. 
     */
    private @SuppressWarnings("rawtypes") List<Class> projectiles = new ArrayList<Class>();
    /**
     * Load the MagicGun with a new Projectile class.
     * @param projectileClass The class of the Projectile to create when it's time to shoot.
     */
    public void loadWith(Class<? extends Projectile> projectileClass){
        projectiles.add(projectileClass);
    }
    /**
     * Shoot the MagicGun with the next Projectile. Projectiles are shot First In First Out.
     * @return A newly created Projectile object.
     * @throws GunIsEmptyException
     */
    public Projectile shoot() throws GunIsEmptyException{
        if (projectiles.isEmpty())
            throw new GunIsEmptyException();
        Projectile projectile = null;
        // We know it must be a Projectile, so the SuppressWarnings is OK
        @SuppressWarnings("unchecked") Class<? extends Projectile> projectileClass = projectiles.get(0);
        projectiles.remove(0);
        try{
            // http://www.Java2s.com/Code/Java/Language-Basics/ObjectReflectioncreatenewinstance.htm
            projectile = projectileClass.newInstance();
        } catch (InstantiationException e) {
            System.err.println(e);
        } catch (IllegalAccessException e) {
            System.err.println(e);
        }
        return projectile;
    }
}

abstract class Projectile {
    public abstract String effectOnTarget();
}

class Pebble extends Projectile {
    @Override public String effectOnTarget() {
        return "annoyed";
    }
}

class Bullet extends Projectile {
    @Override public String effectOnTarget() {
        return "holed";
    }
}

class NuclearMissle extends Projectile {
    @Override public String effectOnTarget() {
        return "obliterated";
    }
}

class GunIsEmptyException extends Exception {
    private static final long serialVersionUID = 4574971294051632635L;
}
14
JohnnyLambada

Class...を渡す必要があります.

private void foo(Class<?> t){
    if(t == String.class){ ... }
    else if(t == int.class){ ... }
}

private void bar()
{
   foo(String.class);
}
9
Mark Elliot

ああ、でもそれはい、オブジェクト指向ではないコードです。 「if/else」および「typeof」を見ると、ポリモーフィズムを考えているはずです。これは間違った方法です。ジェネリックはあなたの友達だと思います。

いくつのタイプに対処する予定ですか?

更新:

Stringとintについて話しているだけの場合は、次の方法があります。インターフェースXmlGeneratorで開始します( "foo"で十分です):

package generics;

public interface XmlGenerator<T>
{
   String getXml(T value);
}

そして、具体的な実装XmlGeneratorImpl:

    package generics;

public class XmlGeneratorImpl<T> implements XmlGenerator<T>
{
    private Class<T> valueType;
    private static final int DEFAULT_CAPACITY = 1024;

    public static void main(String [] args)
    {
        Integer x = 42;
        String y = "foobar";

        XmlGenerator<Integer> intXmlGenerator = new XmlGeneratorImpl<Integer>(Integer.class);
        XmlGenerator<String> stringXmlGenerator = new XmlGeneratorImpl<String>(String.class);

        System.out.println("integer: " + intXmlGenerator.getXml(x));
        System.out.println("string : " + stringXmlGenerator.getXml(y));
    }

    public XmlGeneratorImpl(Class<T> clazz)
    {
        this.valueType = clazz;
    }

    public String getXml(T value)
    {
        StringBuilder builder = new StringBuilder(DEFAULT_CAPACITY);

        appendTag(builder);
        builder.append(value);
        appendTag(builder, false);

        return builder.toString();
    }

    private void appendTag(StringBuilder builder) { this.appendTag(builder, false); }

    private void appendTag(StringBuilder builder, boolean isClosing)
    {
        String valueTypeName = valueType.getName();
        builder.append("<").append(valueTypeName);
        if (isClosing)
        {
            builder.append("/");
        }
        builder.append(">");
    }
}

これを実行すると、次の結果が得られます。

integer: <Java.lang.Integer>42<Java.lang.Integer>
string : <Java.lang.String>foobar<Java.lang.String>

これがあなたが念頭に置いていたものかどうかはわかりません。

8
duffymo

型を渡したい場合、Javaの同等のものは

Java.lang.Class

弱く型付けされたメソッドを使用する場合は、単に使用します

Java.lang.Object

および対応する演算子

instanceof

例えば.

private void foo(Object o) {

  if(o instanceof String) {

  }

}//foo

ただし、Javaにはクラスではないプリミティブ型があります(つまり、この例のint)。したがって、注意する必要があります。

本当の質問は、ここで実際に達成したいことです。そうでなければ、答えるのは困難です。

または、より良い方法がありますか?

5
Dieter

タイプを表すJava.lang.Classのインスタンスを渡すことができます。

private void foo(Class cls)
0
ghallio