如何在Java中将类型作为方法参数传递

71 java types

在Java中,如何将类型作为参数传递(或声明为变量)?

我不想传递类型的实例,而是传递类型本身(例如int,String等).

在C#中,我可以这样做:

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

private void bar()
{
    foo(typeof(String));
}
Run Code Online (Sandbox Code Playgroud)

Java中有没有传递类型为t 的实例的方法?
或者我必须使用自己的int常量或枚举?
或者,还有更好的方法?

编辑:这是foo的要求:
基于类型t,它生成一个不同的短xml字符串.
if/else中的代码将非常小(一行或两行)并将使用一些私有类变量.

Bal*_*usC 98

你可以传入Class<T>.

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

private void bar() {
    foo(String.class);
}
Run Code Online (Sandbox Code Playgroud)

更新:OOP方式取决于功能要求.最好的选择是定义接口foo()和实现两个具体实现foo(),然后调用foo()您手头的实现.另一种方式可能是Map<Class<?>, Action>你可以打电话actions.get(cls).这很容易与接口和具体实现相结合:actions.get(cls).foo().


Joh*_*ada 15

我有一个类似的问题,所以我在下面编写了一个完整的可运行答案.我需要做的是将一个类(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
Run Code Online (Sandbox Code Playgroud)

代码

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;
}
Run Code Online (Sandbox Code Playgroud)

  • 我知道你正试图将一个类型传递给一个方法,我一般都喜欢这个例子,但是这个简单的问题就是为什么你不能用新的射弹实例加载魔法枪而不是像这个?也不要压制警告用`List <Class <?修复它们?延伸弹丸>>弹丸=新的ArrayList <Class <?延伸弹丸>>()`.Projectile类应该是一个接口,并且您的示例不需要序列, (2认同)

Mar*_*iot 9

你应该通过Class......

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

private void bar()
{
   foo(String.class);
}
Run Code Online (Sandbox Code Playgroud)


duf*_*ymo 8

哦,但那是丑陋的,非面向对象的代码.你看到"if/else"和"typeof"的那一刻,你应该考虑多态性.这是错误的方式.我认为仿制药是你的朋友.

您打算处理多少种类型?

更新:

如果您只是在谈论String和int,那么这是您可以采用的一种方式.从界面XmlGenerator开始(足够用"foo"):

package generics;

public interface XmlGenerator<T>
{
   String getXml(T value);
}
Run Code Online (Sandbox Code Playgroud)

并具体实现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(">");
    }
}
Run Code Online (Sandbox Code Playgroud)

如果我运行它,我得到以下结果:

integer: <java.lang.Integer>42<java.lang.Integer>
string : <java.lang.String>foobar<java.lang.String>
Run Code Online (Sandbox Code Playgroud)

我不知道这是不是你的想法.

  • "任何时候你发现自己编写表单的代码"如果对象是T1类型,那么做一些事情,但如果它是T2类型,那么做一些别的事情,"打自己吧." http://www.javapractices.com/topic/TopicAction.do?Id=31 (4认同)

Die*_*ter 5

如果要传递类型,则Java中的等效项将是

java.lang.Class
Run Code Online (Sandbox Code Playgroud)

如果要使用弱类型方法,则只需使用

java.lang.Object
Run Code Online (Sandbox Code Playgroud)

和相应的运算符

instanceof
Run Code Online (Sandbox Code Playgroud)

例如

private void foo(Object o) {

  if(o instanceof String) {

  }

}//foo
Run Code Online (Sandbox Code Playgroud)

但是,在Java中,有一些原始类型,它们不是类(即示例中的int),因此您需要小心。

真正的问题是您实际上想要在这里实现什么,否则很难回答:

或者,还有更好的方法?