值为null的引用类型?

时间:2011-04-19 07:17:48

标签: java null reference-type

根据: http://java.sun.com/docs/books/jls/second_edition/html/typesValues.doc.html
4.5.2参考类型的变量
引用类型可以包含null引用。

当指定的值为null时,是否可以检索引用类型的声明类型?

具体来说,在使用反射的方法中,我希望该方法是null安全的并且对原始声明的类型起作用(虽然我知道以下代码片段不起作用),例如:

String referenceType = null;
MyReflectionClass.reflectionMethod(referenceType);
...
public static void reflectionMethod(Object referenceType) {
   Class<?> declaredType = referenceType.getClass();
}

如果需要,我不会反对使用泛型将T而不是Object作为声明的参数类型。

编辑:我知道.getClass()适用于实例,而不是声明的类型。我想知道是否可以向引用询问它的声明类型。由于类层次结构是静态的,因此获取该信息应该没有问题。

编辑2:此处,情况已明确:Is Java "pass-by-reference" or "pass-by-value"?
Java只是按值传递,因此即使使用了引用类型,也总是处理它,就好像传递了值(对象实例)一样(即使内部只传递了一个对象指针)。这意味着Java实际上没有一个知道它的类型的引用类型(至少就程序员而言),它都在值实例中。
因此,无法确定任何null值的类型。

5 个答案:

答案 0 :(得分:8)

一句话,没有。但是,尝试查找空引用的成员并没有多大意义,因为null意味着没有任何东西。你可能最好不要将null作为输入,可能是抛出NullPointerException

如果必须处理null,也许最好的方法是显式传递类引用(或者甚至更好,直接引用相关方法)。

public static void reflectionMethod(Object param, Class<?> referenceType) // null is dis-allowed for referenceType
{
    // ... no need to try to back-track the type from an object
}

如果你只能获得空值,那么除非有一个非常幸运的猜测,否则很难(除非有其他限制),这很难做得更好。

答案 1 :(得分:7)

你误解了。 getClass()没有告诉您变量的声明类型,它会告诉您对象的类型(如果有的话)。让我们停下来,退后一分钟 - 你会指的是哪个变量,如果它那样做了?原始变量?方法参数?它沿途经过的任何其他变量?那将是彻底的疯狂。

答案 2 :(得分:3)

您无法找到null引用的声明类型。泛型也无法真正帮助,因为它们在编译时被删除了。如果你需要知道null情况,你必须在某种程度上在运行时传递类型。

答案 3 :(得分:0)

不,不是。在您的示例中,当尝试在空引用referenceType上调用方法时,您将获得空指针异常。

更具体地说,这是不可能的,因为在编译时不一定知道具体类型,例如,如果我声明Object类型的引用并将其分配给String,那么您的方法应该发现该对象是String而不是Object。

Object referenceType= new String("I'm a string");
MyReflectionClass.reflectionMethod(referenceType);
...
public static void reflectionMethod(Object referenceType) {
   Class<?> declaredType = referenceType.getClass(); // returns String.class
}

答案 4 :(得分:0)

我不确定这在您的上下文中是否有用,但您可以使用泛型来实现类似的东西 - Java 1.5+保留子类的类型信息。

package spikes;

import java.lang.reflect.ParameterizedType;

public class ReflectMethod {
    public abstract static class GenericType<T> {
        public Class typedClass() {
            ParameterizedType pType = (ParameterizedType) getClass().getGenericSuperclass();
            return (Class) pType.getActualTypeArguments()[0];
        }

        public void reflectMethod(T o) {
            System.out.println("Class: " + typedClass());
        }
    }

    /**
     * @param args
     */
    public static void main(String[] args) {
        GenericType<String> gs = new GenericType<String>() {

        };
        gs.reflectMethod("");
    }
}